From 646b6b5d05de937beb8202e5fd8b8daf3e58e902 Mon Sep 17 00:00:00 2001 From: nlf Date: Wed, 1 Jun 2022 14:13:33 -0700 Subject: [PATCH] fix(arborist): use rawSpec for bundled and shrinkwrapped deps (#4963) --- workspaces/arborist/lib/edge.js | 7 +- workspaces/arborist/test/edge.js | 110 +++++++++++++++++++++++++++++++ 2 files changed, 116 insertions(+), 1 deletion(-) diff --git a/workspaces/arborist/lib/edge.js b/workspaces/arborist/lib/edge.js index d72f312569466..a04404f226563 100644 --- a/workspaces/arborist/lib/edge.js +++ b/workspaces/arborist/lib/edge.js @@ -92,7 +92,12 @@ class Edge { return false } - return depValid(node, this.spec, this.accept, this.from) + // NOTE: this condition means we explicitly do not support overriding + // bundled or shrinkwrapped dependencies + const spec = (node.hasShrinkwrap || node.inShrinkwrap || node.inBundle) + ? this.rawSpec + : this.spec + return depValid(node, spec, this.accept, this.from) } explain (seen = []) { diff --git a/workspaces/arborist/test/edge.js b/workspaces/arborist/test/edge.js index 2df989eb82f89..8bd12bbc1da91 100644 --- a/workspaces/arborist/test/edge.js +++ b/workspaces/arborist/test/edge.js @@ -808,3 +808,113 @@ t.same(bundledEdge.explain(), { bundled: true, from: bundleParent.explain(), }, 'bundled edge.explain as expected') + +t.test('shrinkwrapped and bundled deps are not overridden and remain valid', (t) => { + const overrides = new OverrideSet({ + overrides: { + bar: '^2.0.0', + }, + }) + + const root = { + name: 'root', + packageName: 'root', + edgesOut: new Map(), + edgesIn: new Set(), + explain: () => 'root node explanation', + package: { + name: 'root', + version: '1.2.3', + dependencies: { + foo: '^1.0.0', + }, + overrides: { + bar: '^2.0.0', + }, + }, + get version () { + return this.package.version + }, + isTop: true, + parent: null, + overrides, + resolve (n) { + return n === 'foo' ? foo : null + }, + addEdgeOut (edge) { + this.edgesOut.set(edge.name, edge) + }, + addEdgeIn (edge) { + this.edgesIn.add(edge) + }, + } + + const foo = { + name: 'foo', + packageName: 'foo', + edgesOut: new Map(), + edgesIn: new Set(), + explain: () => 'foo node explanation', + hasShrinkwrap: true, + package: { + name: 'foo', + version: '1.2.3', + dependencies: { + bar: '^1.0.0', + }, + }, + get version () { + return this.package.version + }, + parent: root, + resolve (n) { + return n === 'bar' ? bar : this.parent.resolve(n) + }, + addEdgeOut (edge) { + this.edgesOut.set(edge.name, edge) + }, + addEdgeIn (edge) { + this.edgesIn.add(edge) + }, + } + foo.overrides = overrides.getNodeRule(foo) + + const bar = { + name: 'bar', + packageName: 'bar', + edgesOut: new Map(), + edgesIn: new Set(), + explain: () => 'bar node explanation', + inShrinkwrap: true, + package: { + name: 'bar', + version: '1.2.3', + dependencies: {}, + }, + get version () { + return this.package.version + }, + parent: foo, + resolve (n) { + return this.parent.resolve(n) + }, + addEdgeOut (edge) { + this.edgesOut.set(edge.name, edge) + }, + addEdgeIn (edge) { + this.edgesIn.add(edge) + }, + } + bar.overrides = foo.overrides.getNodeRule(bar) + + const edge = new Edge({ + from: foo, + type: 'prod', + spec: '^1.0.0', + name: 'bar', + overrides: overrides.getEdgeRule({ name: 'bar', spec: '^1.0.0' }), + }) + + t.ok(edge.valid, 'edge is valid') + t.end() +})