| Version | Supported | 
|---|---|
| 1.1.3 | ✅ | 
Please report sensitive security issues via GitHub Issues.
Vulnerabilities are handled and published through GitHub Security Advisories.
In the event of a vulnerability the runbook for the maintainers is as follows:
- Create a new draft security advisory. The values and descriptions don't need to be perfect to begin with as they can be edited later. For severity, use the "Assess severity using CVSS" and refer to the guide for help.
- Request a CVE identification number. It can take up to 72h for one to be assigned so be sure to do this early on.
- If there is not already a patch for the vulnerability, collaborate on one in a private fork:
- Head to the security advisory on GitHub and create a private fork
- Invite any collaborators from outside the maintainer team that should be part of creating a fix.
- Create one or multiple Pull Requests with fixes towards the fork. Note that these PRs will not have CI checks run on them, so more care should be taken to run local validation. The PRs are also not merged like normal PRs, but are instead merged straight into the main repo all at once when the merge button is hit on the security advisory.
 
- Once the fix is ready in a PR or private fork and it is time to release it, there are a couple of options. Either merge into the main branch with a changeset and wait for a regular release, or do a quick release by manually bumping the version in package.jsonof the affected package, along with a manualCHANGELOG.mdentry. Note that a quick release will only work if the package does not have any other pending changes that depend on pending changes in other packages, so be sure to manually check that first, and fall back to an early regular release if needed. In general it's best to stick with the regular release flow, with the quick release being used only for time sensitive fixes.
- Finalize and publish the security advisory. Note that once you hit the publish button it's no longer possible to edit the advisory. Just like the CVE number this can take up to 72h, and expect it to be slower than the CVE number request.
This sample repository uses a mixed bag of CodeQL, SonarCloud, Dependabot and RenovateBot to find and fix vulnerabilities.
In this section we highlight patterns where particular care needs to be taken in order to avoid security vulnerabilities, as well as the best practices to follow.
A common pitfall in backend packages is to resolve local file paths based on user input, without sufficiently protecting against input that traverses outside the intended directory.
For example, consider the following code:
// WARNING: DO NOT DO THIS
import { join } from "path";
import fs from "fs-extra";
function writeTemporaryFile(tmpDir: string, name: string, content: string) {
  const filePath = join(tmpDir, name);
  await fs.writeFile(filePath, content);
}If the name of the file is controlled by the user, they can for example enter ../../../../etc/hosts as the name of the file. This can lead to a file being written outside the intended directory, which in turn can be used to inject malicious code or other form of attacks.
The recommended solution to this is to use resolveSafeChildPath from @backstage/backend-common to resolve the file path instead. It makes sure that the resolved path does not fall outside the provided directory. If you simply want to validate whether a file path is safe, you can use isChildPath instead.
The insecure example above should instead be written like this:
// THIS IS GOOD, DO THIS
import { resolveSafeChildPath } from "@backstaghe/backend-common";
import fs from "fs-extra";
function writeTemporaryFile(tmpDir: string, name: string, content: string) {
  const filePath = resolveSafeChildPath(tmpDir, name);
  await fs.writeFile(filePath, content);
}