From ec38a58a9ffc337e1bcb522c7ecfce8176214665 Mon Sep 17 00:00:00 2001 From: daiscog Date: Mon, 29 Jan 2024 17:23:23 +0000 Subject: [PATCH] fix(js): fix missing top-level dependencies in publishable libs (#17730) --- .../js/src/utils/buildable-libs-utils.spec.ts | 95 +++++++++++++++++++ packages/js/src/utils/buildable-libs-utils.ts | 5 +- 2 files changed, 99 insertions(+), 1 deletion(-) diff --git a/packages/js/src/utils/buildable-libs-utils.spec.ts b/packages/js/src/utils/buildable-libs-utils.spec.ts index cf865b0c6dcef..bf987d362344e 100644 --- a/packages/js/src/utils/buildable-libs-utils.spec.ts +++ b/packages/js/src/utils/buildable-libs-utils.spec.ts @@ -161,6 +161,101 @@ describe('calculateProjectDependencies', () => { ], }); }); + + it('should include all top-level dependencies, even ones that are also transitive', async () => { + const graph: ProjectGraph = { + nodes: { + example: { + type: 'lib', + name: 'example', + data: { + root: '/root/example', + targets: { + build: { + executor: 'x', + }, + }, + }, + }, + example2: { + type: 'lib', + name: 'example2', + data: { + root: '/root/example2', + targets: { + build: { + executor: 'x', + }, + }, + }, + }, + }, + externalNodes: { + 'npm:formik': { + type: 'npm', + name: 'npm:formik', + data: { + packageName: 'formik', + version: '0.0.0', + }, + }, + 'npm:foo': { + type: 'npm', + name: 'npm:foo', + data: { + packageName: 'foo', + version: '0.0.0', + }, + }, + }, + dependencies: { + example: [ + // when example2 dependency is listed first + { + source: 'example', + target: 'example2', + type: DependencyType.static, + }, + { + source: 'example', + target: 'npm:formik', + type: DependencyType.static, + }, + ], + example2: [ + // and example2 also depends on npm:formik + { + source: 'example2', + target: 'npm:formik', + type: DependencyType.static, + }, + { + source: 'example2', + target: 'npm:foo', + type: DependencyType.static, + }, + ], + }, + }; + + const results = calculateProjectDependencies( + graph, + 'root', + 'example', + 'build', + undefined + ); + expect(results).toMatchObject({ + target: { + name: 'example', + }, + topLevelDependencies: [ + // expect example2 and formik as top-level deps, but not foo + expect.objectContaining({ name: 'example2' }), + expect.objectContaining({ name: 'formik' }), + ], + }); + }); }); describe('calculateDependenciesFromTaskGraph', () => { diff --git a/packages/js/src/utils/buildable-libs-utils.ts b/packages/js/src/utils/buildable-libs-utils.ts index 1615417c0a98d..d6c0461ff2089 100644 --- a/packages/js/src/utils/buildable-libs-utils.ts +++ b/packages/js/src/utils/buildable-libs-utils.ts @@ -170,7 +170,8 @@ function collectDependencies( areTopLevelDeps = true ): { name: string; isTopLevel: boolean }[] { (projGraph.dependencies[project] || []).forEach((dependency) => { - if (!acc.some((dep) => dep.name === dependency.target)) { + const existingEntry = acc.find((dep) => dep.name === dependency.target); + if (!existingEntry) { // Temporary skip this. Currently the set of external nodes is built from package.json, not lock file. // As a result, some nodes might be missing. This should not cause any issues, we can just skip them. if ( @@ -184,6 +185,8 @@ function collectDependencies( if (!shallow && isInternalTarget) { collectDependencies(dependency.target, projGraph, acc, shallow, false); } + } else if (areTopLevelDeps && !existingEntry.isTopLevel) { + existingEntry.isTopLevel = true; } }); return acc;