1
- const { delimiter, dirname, resolve } = require ( 'path' )
1
+ 'use strict'
2
+
2
3
const { promisify } = require ( 'util' )
3
- const read = promisify ( require ( 'read' ) )
4
4
5
5
const Arborist = require ( '@npmcli/arborist' )
6
6
const ciDetect = require ( '@npmcli/ci-detect' )
7
+ const crypto = require ( 'crypto' )
7
8
const log = require ( 'proc-log' )
8
- const npmlog = require ( 'npmlog' )
9
9
const mkdirp = require ( 'mkdirp-infer-owner' )
10
10
const npa = require ( 'npm-package-arg' )
11
+ const npmlog = require ( 'npmlog' )
11
12
const pacote = require ( 'pacote' )
13
+ const read = promisify ( require ( 'read' ) )
14
+ const semver = require ( 'semver' )
12
15
13
- const cacheInstallDir = require ( './cache-install-dir.js' )
14
16
const { fileExists, localFileExists } = require ( './file-exists.js' )
15
17
const getBinFromManifest = require ( './get-bin-from-manifest.js' )
16
18
const noTTY = require ( './no-tty.js' )
17
19
const runScript = require ( './run-script.js' )
18
20
const isWindows = require ( './is-windows.js' )
19
- const _localManifest = Symbol ( 'localManifest' )
20
21
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
+ }
25
72
26
73
const exec = async ( opts ) => {
27
74
const {
@@ -32,18 +79,16 @@ const exec = async (opts) => {
32
79
locationMsg = undefined ,
33
80
globalBin = '' ,
34
81
output,
35
- packages : _packages = [ ] ,
82
+ // dereference values because we manipulate it later
83
+ packages : [ ...packages ] = [ ] ,
36
84
path = '.' ,
37
85
runPath = '.' ,
38
86
scriptShell = isWindows ? process . env . ComSpec || 'cmd' : 'sh' ,
39
87
yes = undefined ,
40
88
...flatOptions
41
89
} = opts
42
90
43
- // dereferences values because we manipulate it later
44
- const packages = [ ..._packages ]
45
- const pathArr = [ ...PATH ]
46
- const _run = ( ) => runScript ( {
91
+ const run = ( ) => runScript ( {
47
92
args,
48
93
call,
49
94
color,
@@ -56,120 +101,87 @@ const exec = async (opts) => {
56
101
scriptShell,
57
102
} )
58
103
59
- // nothing to maybe install, skip the arborist dance
104
+ // interactive mode
60
105
if ( ! call && ! args . length && ! packages . length ) {
61
- return await _run ( )
106
+ return run ( )
62
107
}
63
108
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 )
69
112
if ( needPackageCommandSwap ) {
70
- let binExists = false
71
113
const dir = dirname ( dirname ( localBin ) )
72
- const localBinPath = await localFileExists ( dir , args [ 0 ] )
114
+ const localBinPath = await localFileExists ( dir , args [ 0 ] , '/' )
73
115
if ( localBinPath ) {
74
- pathArr . unshift ( localBinPath )
75
- binExists = true
116
+ // @npmcli /run-script adds local bin to $PATH itself
117
+ return await run ( )
76
118
} else if ( await fileExists ( `${ globalBin } /${ args [ 0 ] } ` ) ) {
77
119
pathArr . unshift ( globalBin )
78
- binExists = true
79
- }
80
-
81
- if ( binExists ) {
82
- return await _run ( )
120
+ return await run ( )
83
121
}
84
122
123
+ // We swap out args[0] with the bin from the manifest later
85
124
packages . push ( args [ 0 ] )
86
125
}
87
126
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 } )
93
128
const localTree = await localArb . loadActual ( )
94
129
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 } )
124
137
}
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
- } )
132
138
} ) )
133
139
134
140
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 )
136
153
}
137
154
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
143
158
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 )
145
167
await mkdirp ( installDir )
146
- const arb = new Arborist ( {
168
+ const npxArb = new Arborist ( {
147
169
...flatOptions ,
148
170
path : installDir ,
149
171
} )
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
+ }
159
182
}
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
+ } ) )
171
184
172
- // no need to install if already present
173
185
if ( add . length ) {
174
186
if ( ! yes ) {
175
187
// set -n to always say no
@@ -196,15 +208,15 @@ const exec = async (opts) => {
196
208
}
197
209
}
198
210
}
199
- await arb . reify ( {
211
+ await npxArb . reify ( {
200
212
...flatOptions ,
201
213
add,
202
214
} )
203
215
}
204
216
pathArr . unshift ( resolve ( installDir , 'node_modules/.bin' ) )
205
217
}
206
218
207
- return await _run ( )
219
+ return await run ( )
208
220
}
209
221
210
222
module . exports = exec
0 commit comments