Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

npm workspace improvements #1659

Merged
merged 2 commits into from
Feb 27, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 19 additions & 0 deletions lib/cli/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -2488,8 +2488,19 @@ export async function createNodejsBom(path, options) {
if (mgrData) {
mgr = mgrData.split("@")[0];
}
// Try harder to identify the correct package manager
if (supPkgMgrs.includes(mgr)) {
pkgMgr = mgr;
} else if (pkgData?.engines?.yarn) {
pkgMgr = "yarn";
} else if (
isPackageManagerAllowed("yarn", ["npm", "pnpm", "rush"], options)
) {
pkgMgr = "yarn";
} else if (
isPackageManagerAllowed("pnpm", ["npm", "yarn", "rush"], options)
) {
pkgMgr = "pnpm";
}
let installCommand = "install";
if (pkgMgr === "npm" && isSecureMode && pkgJsonLockFiles?.length > 0) {
Expand Down Expand Up @@ -2524,6 +2535,11 @@ export async function createNodejsBom(path, options) {
installArgs.push("--package-lock");
}
}
if (pkgMgr !== "npm") {
thoughtLog(
`**PACKAGE MANAGER**: About to invoke '${pkgMgr}' with the arguments '${installArgs.join(" ")}' to generate the needed lock files.`,
);
}
console.log(
`Executing '${pkgMgr} ${installArgs.join(" ")}' in`,
basePath,
Expand All @@ -2538,6 +2554,9 @@ export async function createNodejsBom(path, options) {
console.error(
`${pkgMgr} install has failed. Generated SBOM will be empty or with a lower precision.`,
);
thoughtLog(
"It looks like the install command has failed. I'm considering some troubleshooting ideas.",
);
if (DEBUG_MODE && result.stdout) {
if (result.stdout.includes("EBADENGINE Unsupported engine")) {
console.log(
Expand Down
47 changes: 35 additions & 12 deletions lib/helpers/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -1038,11 +1038,8 @@ export async function parsePkgJson(pkgJsonFile, simple = false) {
const pkgData = JSON.parse(readFileSync(pkgJsonFile, "utf8"));
const pkgIdentifier = parsePackageJsonName(pkgData.name);
let name = pkgIdentifier.fullName || pkgData.name;
if (DEBUG_MODE && !name && !pkgJsonFile.includes("node_modules")) {
name = dirname(pkgJsonFile);
console.log(
`${pkgJsonFile} doesn't contain the package name. Consider using the 'npm init' command to create a valid package.json file for this project. Assuming the name as '${name}'.`,
);
if (!name && !pkgJsonFile.includes("node_modules")) {
name = basename(dirname(pkgJsonFile));
}
const group = pkgIdentifier.scope || "";
const purl = new PackageURL(
Expand Down Expand Up @@ -1232,11 +1229,23 @@ export async function parsePkgLock(pkgLockFile, options = {}) {
"bom-ref": decodeURIComponent(purlString),
};
if (node.resolved) {
pkg.properties.push({
name: "ResolvedUrl",
value: node.resolved,
});
pkg.distribution = { url: node.resolved };
if (node.resolved.startsWith("file:")) {
pkg.properties.push({
name: "cdx:npm:resolvedPath",
value: node.realpath
? relative(dirname(pkgLockFile), node.realpath)
: relative(
dirname(pkgLockFile),
resolve(node.resolved.replace("file:", "")),
),
});
} else {
pkg.properties.push({
name: "ResolvedUrl",
value: node.resolved,
});
pkg.distribution = { url: node.resolved };
}
}
if (node.location) {
pkg.properties.push({
Expand Down Expand Up @@ -1328,7 +1337,6 @@ export async function parsePkgLock(pkgLockFile, options = {}) {
});
}
pkgList.push(pkg);

// retrieve workspace node pkglists
const workspaceDependsOn = [];
if (node.fsChildren && node.fsChildren.size > 0) {
Expand All @@ -1346,7 +1354,7 @@ export async function parsePkgLock(pkgLockFile, options = {}) {
);
pkgList = pkgList.concat(childPkgList);
dependenciesList = dependenciesList.concat(childDependenciesList);
const depWorkspacePurlString = decodeURIComponent(
let depWorkspacePurlString = decodeURIComponent(
new PackageURL(
"npm",
"",
Expand All @@ -1358,6 +1366,21 @@ export async function parsePkgLock(pkgLockFile, options = {}) {
.toString()
.replace(/%2F/g, "/"),
);
let purlStringFromPkgid;
if (workspaceNode.pkgid) {
purlStringFromPkgid = `pkg:npm/${workspaceNode.pkgid.replace(`${workspaceNode.name}@npm:`, "")}`;
}
if (
purlStringFromPkgid &&
purlStringFromPkgid !== depWorkspacePurlString
) {
if (DEBUG_MODE) {
console.log(
`Internal warning: Got two different refs for this workspace node: ${depWorkspacePurlString} and ${purlStringFromPkgid}. Assuming the bom-ref as ${purlStringFromPkgid} based on pkgid.`,
);
}
depWorkspacePurlString = purlStringFromPkgid;
}
if (decodeURIComponent(purlString) !== depWorkspacePurlString) {
workspaceDependsOn.push(depWorkspacePurlString);
}
Expand Down
16 changes: 16 additions & 0 deletions lib/helpers/utils.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,7 @@ import {
toGemModuleNames,
yarnLockToIdentMap,
} from "./utils.js";
import { validateRefs } from "./validator.js";

test("SSRI test", () => {
// gopkg.lock hash
Expand Down Expand Up @@ -3393,6 +3394,21 @@ test("parsePkgLock v3", async () => {
expect(parsedList.dependenciesList.length).toEqual(161);
});

test("parsePkgLock theia", async () => {
const parsedList = await parsePkgLock(
"./test/data/package-json/theia/package-lock.json",
{},
);
expect(parsedList.pkgList.length).toEqual(2410);
expect(parsedList.dependenciesList.length).toEqual(2410);
expect(
validateRefs({
components: parsedList.pkgList,
dependencies: parsedList.dependenciesList,
}),
).toBeTruthy();
});

test("parseBowerJson", async () => {
const deps = await parseBowerJson("./test/data/bower.json");
expect(deps.length).toEqual(1);
Expand Down
14 changes: 10 additions & 4 deletions lib/managers/docker.js
Original file line number Diff line number Diff line change
Expand Up @@ -639,8 +639,11 @@ export const getImage = async (fullImageName) => {
}
if (needsCliFallback()) {
let dockerCmd = process.env.DOCKER_CMD || "docker";
if (!process.env.DOCKER_CMD && detectRancherDesktop()) {
dockerCmd = "nerdctl";
if (!process.env.DOCKER_CMD) {
detectColima() || detectRancherDesktop();
if (isNerdctl) {
dockerCmd = "nerdctl";
}
}
let result = spawnSync(dockerCmd, ["pull", fullImageName], {
encoding: "utf-8",
Expand Down Expand Up @@ -1116,8 +1119,11 @@ export const exportImage = async (fullImageName, options) => {
if (needsCliFallback()) {
const imageTarFile = join(tempDir, "image.tar");
let dockerCmd = process.env.DOCKER_CMD || "docker";
if (!process.env.DOCKER_CMD && detectRancherDesktop()) {
dockerCmd = "nerdctl";
if (!process.env.DOCKER_CMD) {
detectColima() || detectRancherDesktop();
if (isNerdctl) {
dockerCmd = "nerdctl";
}
}
console.log(
`About to export image ${fullImageName} to ${imageTarFile} using ${dockerCmd} cli`,
Expand Down
Loading
Loading