Skip to content
This repository has been archived by the owner on Jan 20, 2022. It is now read-only.

Commit

Permalink
Install peerOptionals if explicitly requested, or dev
Browse files Browse the repository at this point in the history
Fix npm/cli#2005

PR-URL: #166
Credit: @isaacs
Close: #166
Reviewed-by: @ruyadorno
  • Loading branch information
isaacs authored and ruyadorno committed Oct 23, 2020
1 parent 1d6b057 commit daa5cf0
Show file tree
Hide file tree
Showing 6 changed files with 133 additions and 9 deletions.
2 changes: 2 additions & 0 deletions lib/add-rm-pkg-deps.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ const removeFromOthers = (name, type, pkg) => {
break
case 'dev':
others.delete('devDependencies')
others.delete('peerDependencies')
others.delete('peerDependenciesMeta')
break
case 'optional':
others.delete('optionalDependencies')
Expand Down
9 changes: 6 additions & 3 deletions lib/arborist/build-ideal-tree.js
Original file line number Diff line number Diff line change
Expand Up @@ -856,9 +856,12 @@ This is a one-time fix-up, please be patient...
if (edge.to && edge.to.inShrinkwrap)
return false

// If the edge has no destination, that's a problem.
if (!edge.to)
return edge.type !== 'peerOptional'
// If the edge has no destination, that's a problem, unless
// if it's peerOptional and not explicitly requested.
if (!edge.to) {
return edge.type !== 'peerOptional' ||
this[_explicitRequests].has(edge.name)
}

// If the edge has an error, there's a problem.
if (!edge.valid)
Expand Down
12 changes: 6 additions & 6 deletions lib/node.js
Original file line number Diff line number Diff line change
Expand Up @@ -502,6 +502,12 @@ class Node {
this[_loadDepType](this.package.optionalDependencies, 'optional')
this[_loadDepType](this.package.dependencies, 'prod')

// Linked targets that are disconnected from the tree are tops,
// but don't have a 'path' field, only a 'realpath', because we
// don't know their canonical location. We don't need their devDeps.
if (this.isTop && this.path)
this[_loadDepType](this.package.devDependencies, 'dev')

const pd = this.package.peerDependencies
if (pd && typeof pd === 'object' && !this.legacyPeerDeps) {
const pm = this.package.peerDependenciesMeta || {}
Expand All @@ -516,12 +522,6 @@ class Node {
this[_loadDepType](peerDependencies, 'peer')
this[_loadDepType](peerOptional, 'peerOptional')
}

// Linked targets that are disconnected from the tree are tops,
// but don't have a 'path' field, only a 'realpath', because we
// don't know their canonical location. We don't need their devDeps.
if (this.isTop && this.path)
this[_loadDepType](this.package.devDependencies, 'dev')
}

[_loadDepType] (obj, type) {
Expand Down
91 changes: 91 additions & 0 deletions tap-snapshots/test-arborist-build-ideal-tree.js-TAP.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -65110,6 +65110,97 @@ Node {
}
`

exports[`test/arborist/build-ideal-tree.js TAP peerOptionals that are devDeps or explicit request > should install the abbrev dep 1`] = `
Node {
"children": Map {
"abbrev" => Node {
"edgesIn": Set {
Edge {
"from": "",
"name": "abbrev",
"spec": "",
"type": "peerOptional",
},
},
"location": "node_modules/abbrev",
"name": "abbrev",
"optional": true,
"peer": true,
"resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz",
},
"wrappy" => Node {
"dev": true,
"edgesIn": Set {
Edge {
"from": "",
"name": "wrappy",
"spec": "",
"type": "dev",
},
},
"location": "node_modules/wrappy",
"name": "wrappy",
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
},
},
"edgesOut": Map {
"abbrev" => Edge {
"name": "abbrev",
"spec": "",
"to": "node_modules/abbrev",
"type": "peerOptional",
},
"wrappy" => Edge {
"name": "wrappy",
"spec": "",
"to": "node_modules/wrappy",
"type": "dev",
},
},
"location": "",
"name": "peer-optional-installs",
"resolved": null,
}
`

exports[`test/arborist/build-ideal-tree.js TAP peerOptionals that are devDeps or explicit request > should install the wrappy dep, and not remove from peerDeps 1`] = `
Node {
"children": Map {
"wrappy" => Node {
"dev": true,
"edgesIn": Set {
Edge {
"from": "",
"name": "wrappy",
"spec": "",
"type": "dev",
},
},
"location": "node_modules/wrappy",
"name": "wrappy",
"resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
},
},
"edgesOut": Map {
"abbrev" => Edge {
"name": "abbrev",
"spec": "",
"to": null,
"type": "peerOptional",
},
"wrappy" => Edge {
"name": "wrappy",
"spec": "",
"to": "node_modules/wrappy",
"type": "dev",
},
},
"location": "",
"name": "peer-optional-installs",
"resolved": null,
}
`

exports[`test/arborist/build-ideal-tree.js TAP push conflicted peer deps deeper in to the tree to solve > must match snapshot 1`] = `
Node {
"children": Map {
Expand Down
11 changes: 11 additions & 0 deletions test/arborist/build-ideal-tree.js
Original file line number Diff line number Diff line change
Expand Up @@ -2114,3 +2114,14 @@ t.test('carbonium eslint conflicts', async t => {
],
}))
})

t.test('peerOptionals that are devDeps or explicit request', async t => {
const path = resolve(fixtures, 'peer-optional-installs')
t.matchSnapshot(await printIdeal(path, {
add: ['abbrev'],
}), 'should install the abbrev dep')
t.matchSnapshot(await printIdeal(path, {
add: ['wrappy'],
saveType: 'dev',
}), 'should install the wrappy dep, and not remove from peerDeps')
})
17 changes: 17 additions & 0 deletions test/fixtures/peer-optional-installs/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
{
"peerDependencies": {
"abbrev": "",
"wrappy": ""
},
"devDependencies": {
"wrappy": ""
},
"peerDependenciesMeta": {
"wrappy": {
"optional": true
},
"abbrev": {
"optional": true
}
}
}

0 comments on commit daa5cf0

Please sign in to comment.