Skip to content

Commit ea44995

Browse files
wraithgarnlf
authored andcommittedAug 2, 2022
fix: properly find locally/globally/npxCache packages
Lots of bugfixes here, we properly parse ranges and versions, and we also now work with git repos and gists, and know when they are already installed.
1 parent d0be9a2 commit ea44995

File tree

11 files changed

+480
-155
lines changed

11 files changed

+480
-155
lines changed
 

‎docs/content/commands/npm-exec.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ $ npm exec -- foo@latest bar --package=@npmcli/foo
127127
* Default:
128128
* Type: String (can be set multiple times)
129129

130-
The package to install for [`npm exec`](/commands/npm-exec)
130+
The package or packages to install for [`npm exec`](/commands/npm-exec)
131131

132132
<!-- automatically generated, do not edit manually -->
133133
<!-- see lib/utils/config/definitions.js -->

‎docs/content/using-npm/config.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1244,7 +1244,7 @@ Directory in which `npm pack` will save tarballs.
12441244
* Default:
12451245
* Type: String (can be set multiple times)
12461246

1247-
The package to install for [`npm exec`](/commands/npm-exec)
1247+
The package or packages to install for [`npm exec`](/commands/npm-exec)
12481248

12491249
<!-- automatically generated, do not edit manually -->
12501250
<!-- see lib/utils/config/definitions.js -->

‎lib/commands/exec.js

+2
Original file line numberDiff line numberDiff line change
@@ -49,8 +49,10 @@ class Exec extends BaseCommand {
4949
static isShellout = true
5050

5151
async exec (_args, { locationMsg, runPath } = {}) {
52+
// This is where libnpmexec will look for locally installed packages
5253
const path = this.npm.localPrefix
5354

55+
// This is where libnpmexec will actually run the scripts from
5456
if (!runPath) {
5557
runPath = process.cwd()
5658
}

‎lib/utils/config/definitions.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1470,7 +1470,7 @@ define('package', {
14701470
hint: '<package-spec>',
14711471
type: [String, Array],
14721472
description: `
1473-
The package to install for [\`npm exec\`](/commands/npm-exec)
1473+
The package or packages to install for [\`npm exec\`](/commands/npm-exec)
14741474
`,
14751475
flatten,
14761476
})

‎tap-snapshots/test/lib/utils/config/definitions.js.test.cjs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1316,7 +1316,7 @@ exports[`test/lib/utils/config/definitions.js TAP > config description for packa
13161316
* Default:
13171317
* Type: String (can be set multiple times)
13181318
1319-
The package to install for [\`npm exec\`](/commands/npm-exec)
1319+
The package or packages to install for [\`npm exec\`](/commands/npm-exec)
13201320
`
13211321

13221322
exports[`test/lib/utils/config/definitions.js TAP > config description for package-lock 1`] = `

‎workspaces/libnpmexec/lib/cache-install-dir.js

-20
This file was deleted.

‎workspaces/libnpmexec/lib/file-exists.js

+12-10
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,25 @@
11
const { resolve } = require('path')
2-
const { promisify } = require('util')
3-
const stat = promisify(require('fs').stat)
2+
const fs = require('@npmcli/fs')
43
const walkUp = require('walk-up-path')
54

6-
const fileExists = (file) => stat(file)
7-
.then((res) => res.isFile())
8-
.catch(() => false)
9-
10-
const localFileExists = async (dir, binName, root = '/') => {
11-
root = resolve(root).toLowerCase()
5+
const fileExists = async (file) => {
6+
try {
7+
const res = await fs.stat(file)
8+
return res.isFile()
9+
} catch {
10+
return false
11+
}
12+
}
1213

13-
for (const path of walkUp(resolve(dir))) {
14+
const localFileExists = async (dir, binName, root) => {
15+
for (const path of walkUp(dir)) {
1416
const binDir = resolve(path, 'node_modules', '.bin')
1517

1618
if (await fileExists(resolve(binDir, binName))) {
1719
return binDir
1820
}
1921

20-
if (path.toLowerCase() === root) {
22+
if (path.toLowerCase() === resolve(root).toLowerCase()) {
2123
return false
2224
}
2325
}

‎workspaces/libnpmexec/lib/index.js

+114-102
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,74 @@
1-
const { delimiter, dirname, resolve } = require('path')
1+
'use strict'
2+
23
const { promisify } = require('util')
3-
const read = promisify(require('read'))
44

55
const Arborist = require('@npmcli/arborist')
66
const ciDetect = require('@npmcli/ci-detect')
7+
const crypto = require('crypto')
78
const log = require('proc-log')
8-
const npmlog = require('npmlog')
99
const mkdirp = require('mkdirp-infer-owner')
1010
const npa = require('npm-package-arg')
11+
const npmlog = require('npmlog')
1112
const pacote = require('pacote')
13+
const read = promisify(require('read'))
14+
const semver = require('semver')
1215

13-
const cacheInstallDir = require('./cache-install-dir.js')
1416
const { fileExists, localFileExists } = require('./file-exists.js')
1517
const getBinFromManifest = require('./get-bin-from-manifest.js')
1618
const noTTY = require('./no-tty.js')
1719
const runScript = require('./run-script.js')
1820
const isWindows = require('./is-windows.js')
19-
const _localManifest = Symbol('localManifest')
2021

21-
/* istanbul ignore next */
22-
const PATH = (
23-
process.env.PATH || process.env.Path || process.env.path
24-
).split(delimiter)
22+
const { delimiter, dirname, resolve } = require('path')
23+
24+
const pathArr = process.env.PATH.split(delimiter)
25+
26+
// when checking the local tree we look up manifests, cache those results by
27+
// spec.raw so we don't have to fetch again when we check npxCache
28+
const manifests = new Map()
29+
30+
// Returns the required manifest if the spec is missing from the tree
31+
const missingFromTree = async ({ spec, tree, pacoteOpts }) => {
32+
if (spec.registry && (spec.rawSpec === '' || spec.type !== 'tag')) {
33+
// registry spec that is not a specific tag.
34+
const nodesBySpec = tree.inventory.query('packageName', spec.name)
35+
for (const node of nodesBySpec) {
36+
if (spec.type === 'tag') {
37+
// package requested by name only
38+
return
39+
} else if (spec.type === 'version') {
40+
// package requested by specific version
41+
if (node.pkgid === spec.raw) {
42+
return
43+
}
44+
} else {
45+
// package requested by version range, only remaining registry type
46+
if (semver.satisfies(node.package.version, spec.rawSpec)) {
47+
return
48+
}
49+
}
50+
}
51+
if (!manifests.get(spec.raw)) {
52+
manifests.set(spec.raw, await pacote.manifest(spec, pacoteOpts))
53+
}
54+
return manifests.get(spec.raw)
55+
} else {
56+
// non-registry spec, or a specific tag. Look up manifest and check
57+
// resolved to see if it's in the tree.
58+
if (!manifests.get(spec.raw)) {
59+
manifests.set(spec.raw, await pacote.manifest(spec, pacoteOpts))
60+
}
61+
const manifest = manifests.get(spec.raw)
62+
const nodesByManifest = tree.inventory.query('packageName', manifest.name)
63+
for (const node of nodesByManifest) {
64+
if (node.package.resolved === manifest._resolved) {
65+
// we have a package by the same name and the same resolved destination, nothing to add.
66+
return
67+
}
68+
}
69+
return manifest
70+
}
71+
}
2572

2673
const exec = async (opts) => {
2774
const {
@@ -32,18 +79,16 @@ const exec = async (opts) => {
3279
locationMsg = undefined,
3380
globalBin = '',
3481
output,
35-
packages: _packages = [],
82+
// dereference values because we manipulate it later
83+
packages: [...packages] = [],
3684
path = '.',
3785
runPath = '.',
3886
scriptShell = isWindows ? process.env.ComSpec || 'cmd' : 'sh',
3987
yes = undefined,
4088
...flatOptions
4189
} = opts
4290

43-
// dereferences values because we manipulate it later
44-
const packages = [..._packages]
45-
const pathArr = [...PATH]
46-
const _run = () => runScript({
91+
const run = () => runScript({
4792
args,
4893
call,
4994
color,
@@ -56,120 +101,87 @@ const exec = async (opts) => {
56101
scriptShell,
57102
})
58103

59-
// nothing to maybe install, skip the arborist dance
104+
// interactive mode
60105
if (!call && !args.length && !packages.length) {
61-
return await _run()
106+
return run()
62107
}
63108

64-
const needPackageCommandSwap = args.length && !packages.length
65-
// if there's an argument and no package has been explicitly asked for
66-
// check the local and global bin paths for a binary named the same as
67-
// the argument and run it if it exists, otherwise fall through to
68-
// the behavior of treating the single argument as a package name
109+
const pacoteOpts = { ...flatOptions, perferOnline: true }
110+
111+
const needPackageCommandSwap = (args.length > 0) && (packages.length === 0)
69112
if (needPackageCommandSwap) {
70-
let binExists = false
71113
const dir = dirname(dirname(localBin))
72-
const localBinPath = await localFileExists(dir, args[0])
114+
const localBinPath = await localFileExists(dir, args[0], '/')
73115
if (localBinPath) {
74-
pathArr.unshift(localBinPath)
75-
binExists = true
116+
// @npmcli/run-script adds local bin to $PATH itself
117+
return await run()
76118
} else if (await fileExists(`${globalBin}/${args[0]}`)) {
77119
pathArr.unshift(globalBin)
78-
binExists = true
79-
}
80-
81-
if (binExists) {
82-
return await _run()
120+
return await run()
83121
}
84122

123+
// We swap out args[0] with the bin from the manifest later
85124
packages.push(args[0])
86125
}
87126

88-
// figure out whether we need to install stuff, or if local is fine
89-
const localArb = new Arborist({
90-
...flatOptions,
91-
path,
92-
})
127+
const localArb = new Arborist({ ...flatOptions, path })
93128
const localTree = await localArb.loadActual()
94129

95-
const getLocalManifest = ({ tree, name }) => {
96-
// look up the package name in the current tree inventory,
97-
// if it's found then return that normalized pkg data
98-
const [node] = tree.inventory.query('packageName', name)
99-
100-
if (node) {
101-
return {
102-
_id: node.pkgid,
103-
...node.package,
104-
[_localManifest]: true,
105-
}
106-
}
107-
}
108-
109-
// If we do `npm exec foo`, and have a `foo` locally, then we'll
110-
// always use that, so we don't really need to fetch the manifest.
111-
// So: run npa on each packages entry, and if it is a name with a
112-
// rawSpec==='', then try to find that node name in the tree inventory
113-
// and only pacote fetch if that fails.
114-
const manis = await Promise.all(packages.map(async p => {
115-
const spec = npa(p, path)
116-
if (spec.type === 'tag' && spec.rawSpec === '') {
117-
const localManifest = getLocalManifest({
118-
tree: localTree,
119-
name: spec.name,
120-
})
121-
if (localManifest) {
122-
return localManifest
123-
}
130+
// Find anything that isn't installed locally
131+
const needInstall = []
132+
await Promise.all(packages.map(async pkg => {
133+
const spec = npa(pkg, path)
134+
const manifest = await missingFromTree({ spec, tree: localTree, pacoteOpts })
135+
if (manifest) {
136+
needInstall.push({ spec, manifest })
124137
}
125-
// Force preferOnline to true so we are making sure to pull in the latest
126-
// This is especially useful if the user didn't give us a version, and
127-
// they expect to be running @latest
128-
return await pacote.manifest(p, {
129-
...flatOptions,
130-
preferOnline: true,
131-
})
132138
}))
133139

134140
if (needPackageCommandSwap) {
135-
args[0] = getBinFromManifest(manis[0])
141+
// Either we have a scoped package or the bin of our package we inferred
142+
// from arg[0] is not identical to the package name
143+
let commandManifest
144+
if (needInstall.length === 0) {
145+
commandManifest = await pacote.manifest(args[0], {
146+
...flatOptions,
147+
preferOnline: true,
148+
})
149+
} else {
150+
commandManifest = needInstall[0].manifest
151+
}
152+
args[0] = getBinFromManifest(commandManifest)
136153
}
137154

138-
// are all packages from the manifest list installed?
139-
const needInstall =
140-
manis.some(manifest => !manifest[_localManifest])
141-
142-
if (needInstall) {
155+
const add = []
156+
if (needInstall.length > 0) {
157+
// Install things to the npx cache, if needed
143158
const { npxCache } = flatOptions
144-
const installDir = cacheInstallDir({ npxCache, packages })
159+
if (!npxCache) {
160+
throw new Error('Must provide a valid npxCache path')
161+
}
162+
const hash = crypto.createHash('sha512')
163+
.update(packages.sort((a, b) => a.localeCompare(b, 'en')).join('\n'))
164+
.digest('hex')
165+
.slice(0, 16)
166+
const installDir = resolve(npxCache, hash)
145167
await mkdirp(installDir)
146-
const arb = new Arborist({
168+
const npxArb = new Arborist({
147169
...flatOptions,
148170
path: installDir,
149171
})
150-
const tree = await arb.loadActual()
151-
152-
// inspect the npx-space installed tree to check if the package is already
153-
// there, if that's the case also check that it's version matches the same
154-
// version expected by the user requested pkg returned by pacote.manifest
155-
const filterMissingPackagesFromInstallDir = (mani) => {
156-
const localManifest = getLocalManifest({ tree, name: mani.name })
157-
if (localManifest) {
158-
return localManifest.version !== mani.version
172+
const npxTree = await npxArb.loadActual()
173+
await Promise.all(needInstall.map(async ({ spec }) => {
174+
const manifest = await missingFromTree({ spec, tree: npxTree, pacoteOpts })
175+
if (manifest) {
176+
// Manifest is not in npxCache, we need to install it there
177+
if (!spec.registry) {
178+
add.push(manifest._from)
179+
} else {
180+
add.push(manifest._id)
181+
}
159182
}
160-
return true
161-
}
162-
163-
// at this point, we have to ensure that we get the exact same
164-
// version, because it's something that has only ever been installed
165-
// by npm exec in the cache install directory
166-
const add = manis
167-
.filter(mani => !mani[_localManifest])
168-
.filter(filterMissingPackagesFromInstallDir)
169-
.map(mani => mani._id || mani._from)
170-
.sort((a, b) => a.localeCompare(b, 'en'))
183+
}))
171184

172-
// no need to install if already present
173185
if (add.length) {
174186
if (!yes) {
175187
// set -n to always say no
@@ -196,15 +208,15 @@ const exec = async (opts) => {
196208
}
197209
}
198210
}
199-
await arb.reify({
211+
await npxArb.reify({
200212
...flatOptions,
201213
add,
202214
})
203215
}
204216
pathArr.unshift(resolve(installDir, 'node_modules/.bin'))
205217
}
206218

207-
return await _run()
219+
return await run()
208220
}
209221

210222
module.exports = exec

‎workspaces/libnpmexec/test/cache-install-dir.js

-12
This file was deleted.

‎workspaces/libnpmexec/test/index.js

+347-6
Large diffs are not rendered by default.

‎workspaces/libnpmexec/test/registry/content/ruyadorno/create-index.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -78,4 +78,4 @@
7878
"readmeFilename": "README.md",
7979
"_cached": false,
8080
"_contentLength": 0
81-
}
81+
}

0 commit comments

Comments
 (0)
Please sign in to comment.