1'use strict' 2 3exports = module.exports = lifecycle 4exports.makeEnv = makeEnv 5exports._incorrectWorkingDirectory = _incorrectWorkingDirectory 6 7// for testing 8const platform = process.env.__TESTING_FAKE_PLATFORM__ || process.platform 9const isWindows = platform === 'win32' 10const spawn = require('./lib/spawn') 11const path = require('path') 12const Stream = require('stream').Stream 13const fs = require('graceful-fs') 14const chain = require('slide').chain 15const uidNumber = require('uid-number') 16const umask = require('umask') 17const which = require('which') 18const byline = require('byline') 19const resolveFrom = require('resolve-from') 20 21const DEFAULT_NODE_GYP_PATH = resolveFrom(__dirname, 'node-gyp/bin/node-gyp') 22const hookStatCache = new Map() 23 24let PATH = isWindows ? 'Path' : 'PATH' 25exports._pathEnvName = PATH 26const delimiter = path.delimiter 27 28// windows calls its path 'Path' usually, but this is not guaranteed. 29// merge them all together in the order they appear in the object. 30const mergePath = env => 31 Object.keys(env).filter(p => /^path$/i.test(p) && env[p]) 32 .map(p => env[p].split(delimiter)) 33 .reduce((set, p) => set.concat(p.filter(p => !set.includes(p))), []) 34 .join(delimiter) 35exports._mergePath = mergePath 36 37const setPathEnv = (env, path) => { 38 // first ensure that the canonical value is set. 39 env[PATH] = path 40 // also set any other case values, because windows. 41 Object.keys(env) 42 .filter(p => p !== PATH && /^path$/i.test(p)) 43 .forEach(p => { env[p] = path }) 44} 45exports._setPathEnv = setPathEnv 46 47function logid (pkg, stage) { 48 return pkg._id + '~' + stage + ':' 49} 50 51function hookStat (dir, stage, cb) { 52 const hook = path.join(dir, '.hooks', stage) 53 const cachedStatError = hookStatCache.get(hook) 54 55 if (cachedStatError === undefined) { 56 return fs.stat(hook, function (statError) { 57 hookStatCache.set(hook, statError) 58 cb(statError) 59 }) 60 } 61 62 return setImmediate(() => cb(cachedStatError)) 63} 64 65function lifecycle (pkg, stage, wd, opts) { 66 return new Promise((resolve, reject) => { 67 while (pkg && pkg._data) pkg = pkg._data 68 if (!pkg) return reject(new Error('Invalid package data')) 69 70 opts.log.info('lifecycle', logid(pkg, stage), pkg._id) 71 if (!pkg.scripts) pkg.scripts = {} 72 73 if (stage === 'prepublish' && opts.ignorePrepublish) { 74 opts.log.info('lifecycle', logid(pkg, stage), 'ignored because ignore-prepublish is set to true', pkg._id) 75 delete pkg.scripts.prepublish 76 } 77 78 hookStat(opts.dir, stage, function (statError) { 79 // makeEnv is a slow operation. This guard clause prevents makeEnv being called 80 // and avoids a ton of unnecessary work, and results in a major perf boost. 81 if (!pkg.scripts[stage] && statError) return resolve() 82 83 validWd(wd || path.resolve(opts.dir, pkg.name), function (er, wd) { 84 if (er) return reject(er) 85 86 if ((wd.indexOf(opts.dir) !== 0 || _incorrectWorkingDirectory(wd, pkg)) && 87 !opts.unsafePerm && pkg.scripts[stage]) { 88 opts.log.warn('lifecycle', logid(pkg, stage), 'cannot run in wd', pkg._id, pkg.scripts[stage], `(wd=${wd})`) 89 return resolve() 90 } 91 92 // set the env variables, then run scripts as a child process. 93 var env = makeEnv(pkg, opts) 94 env.npm_lifecycle_event = stage 95 env.npm_node_execpath = env.NODE = env.NODE || process.execPath 96 env.npm_execpath = require.main.filename 97 env.INIT_CWD = process.cwd() 98 env.npm_config_node_gyp = env.npm_config_node_gyp || DEFAULT_NODE_GYP_PATH 99 100 // 'nobody' typically doesn't have permission to write to /tmp 101 // even if it's never used, sh freaks out. 102 if (!opts.unsafePerm) env.TMPDIR = wd 103 104 lifecycle_(pkg, stage, wd, opts, env, (er) => { 105 if (er) return reject(er) 106 return resolve() 107 }) 108 }) 109 }) 110 }) 111} 112 113function _incorrectWorkingDirectory (wd, pkg) { 114 return wd.lastIndexOf(pkg.name) !== wd.length - pkg.name.length 115} 116 117function lifecycle_ (pkg, stage, wd, opts, env, cb) { 118 var pathArr = [] 119 var p = wd.split(/[\\/]node_modules[\\/]/) 120 var acc = path.resolve(p.shift()) 121 122 p.forEach(function (pp) { 123 pathArr.unshift(path.join(acc, 'node_modules', '.bin')) 124 acc = path.join(acc, 'node_modules', pp) 125 }) 126 pathArr.unshift(path.join(acc, 'node_modules', '.bin')) 127 128 // we also unshift the bundled node-gyp-bin folder so that 129 // the bundled one will be used for installing things. 130 pathArr.unshift(path.join(__dirname, 'node-gyp-bin')) 131 132 if (shouldPrependCurrentNodeDirToPATH(opts)) { 133 // prefer current node interpreter in child scripts 134 pathArr.push(path.dirname(process.execPath)) 135 } 136 137 const existingPath = mergePath(env) 138 if (existingPath) pathArr.push(existingPath) 139 const envPath = pathArr.join(isWindows ? ';' : ':') 140 setPathEnv(env, envPath) 141 142 var packageLifecycle = pkg.scripts && pkg.scripts.hasOwnProperty(stage) 143 144 if (opts.ignoreScripts) { 145 opts.log.info('lifecycle', logid(pkg, stage), 'ignored because ignore-scripts is set to true', pkg._id) 146 packageLifecycle = false 147 } else if (packageLifecycle) { 148 // define this here so it's available to all scripts. 149 env.npm_lifecycle_script = pkg.scripts[stage] 150 } else { 151 opts.log.silly('lifecycle', logid(pkg, stage), 'no script for ' + stage + ', continuing') 152 } 153 154 function done (er) { 155 if (er) { 156 if (opts.force) { 157 opts.log.info('lifecycle', logid(pkg, stage), 'forced, continuing', er) 158 er = null 159 } else if (opts.failOk) { 160 opts.log.warn('lifecycle', logid(pkg, stage), 'continuing anyway', er.message) 161 er = null 162 } 163 } 164 cb(er) 165 } 166 167 chain( 168 [ 169 packageLifecycle && [runPackageLifecycle, pkg, stage, env, wd, opts], 170 [runHookLifecycle, pkg, stage, env, wd, opts] 171 ], 172 done 173 ) 174} 175 176function shouldPrependCurrentNodeDirToPATH (opts) { 177 const cfgsetting = opts.scriptsPrependNodePath 178 if (cfgsetting === false) return false 179 if (cfgsetting === true) return true 180 181 var isDifferentNodeInPath 182 183 var foundExecPath 184 try { 185 foundExecPath = which.sync(path.basename(process.execPath), { pathExt: isWindows ? ';' : ':' }) 186 // Apply `fs.realpath()` here to avoid false positives when `node` is a symlinked executable. 187 isDifferentNodeInPath = fs.realpathSync(process.execPath).toUpperCase() !== 188 fs.realpathSync(foundExecPath).toUpperCase() 189 } catch (e) { 190 isDifferentNodeInPath = true 191 } 192 193 if (cfgsetting === 'warn-only') { 194 if (isDifferentNodeInPath && !shouldPrependCurrentNodeDirToPATH.hasWarned) { 195 if (foundExecPath) { 196 opts.log.warn('lifecycle', 'The node binary used for scripts is', foundExecPath, 'but npm is using', process.execPath, 'itself. Use the `--scripts-prepend-node-path` option to include the path for the node binary npm was executed with.') 197 } else { 198 opts.log.warn('lifecycle', 'npm is using', process.execPath, 'but there is no node binary in the current PATH. Use the `--scripts-prepend-node-path` option to include the path for the node binary npm was executed with.') 199 } 200 shouldPrependCurrentNodeDirToPATH.hasWarned = true 201 } 202 203 return false 204 } 205 206 return isDifferentNodeInPath 207} 208 209function validWd (d, cb) { 210 fs.stat(d, function (er, st) { 211 if (er || !st.isDirectory()) { 212 var p = path.dirname(d) 213 if (p === d) { 214 return cb(new Error('Could not find suitable wd')) 215 } 216 return validWd(p, cb) 217 } 218 return cb(null, d) 219 }) 220} 221 222function runPackageLifecycle (pkg, stage, env, wd, opts, cb) { 223 // run package lifecycle scripts in the package root, or the nearest parent. 224 var cmd = env.npm_lifecycle_script 225 226 var note = '\n> ' + pkg._id + ' ' + stage + ' ' + wd + 227 '\n> ' + cmd + '\n' 228 runCmd(note, cmd, pkg, env, stage, wd, opts, cb) 229} 230 231var running = false 232var queue = [] 233function dequeue () { 234 running = false 235 if (queue.length) { 236 var r = queue.shift() 237 runCmd.apply(null, r) 238 } 239} 240 241function runCmd (note, cmd, pkg, env, stage, wd, opts, cb) { 242 if (running) { 243 queue.push([note, cmd, pkg, env, stage, wd, opts, cb]) 244 return 245 } 246 247 running = true 248 opts.log.pause() 249 var unsafe = opts.unsafePerm 250 var user = unsafe ? null : opts.user 251 var group = unsafe ? null : opts.group 252 253 if (opts.log.level !== 'silent') { 254 opts.log.clearProgress() 255 console.log(note) 256 opts.log.showProgress() 257 } 258 opts.log.verbose('lifecycle', logid(pkg, stage), 'unsafe-perm in lifecycle', unsafe) 259 260 if (isWindows) { 261 unsafe = true 262 } 263 264 if (unsafe) { 265 runCmd_(cmd, pkg, env, wd, opts, stage, unsafe, 0, 0, cb) 266 } else { 267 uidNumber(user, group, function (er, uid, gid) { 268 if (er) { 269 er.code = 'EUIDLOOKUP' 270 opts.log.resume() 271 process.nextTick(dequeue) 272 return cb(er) 273 } 274 runCmd_(cmd, pkg, env, wd, opts, stage, unsafe, uid, gid, cb) 275 }) 276 } 277} 278 279const getSpawnArgs = ({ cmd, wd, opts, uid, gid, unsafe, env }) => { 280 const conf = { 281 cwd: wd, 282 env: env, 283 stdio: opts.stdio || [ 0, 1, 2 ] 284 } 285 286 if (!unsafe) { 287 conf.uid = uid ^ 0 288 conf.gid = gid ^ 0 289 } 290 291 const customShell = opts.scriptShell 292 293 let sh = 'sh' 294 let shFlag = '-c' 295 if (customShell) { 296 sh = customShell 297 } else if (isWindows || opts._TESTING_FAKE_WINDOWS_) { 298 sh = process.env.comspec || 'cmd' 299 // '/d /s /c' is used only for cmd.exe. 300 if (/^(?:.*\\)?cmd(?:\.exe)?$/i.test(sh)) { 301 shFlag = '/d /s /c' 302 conf.windowsVerbatimArguments = true 303 } 304 } 305 306 return [sh, [shFlag, cmd], conf] 307} 308 309exports._getSpawnArgs = getSpawnArgs 310 311function runCmd_ (cmd, pkg, env, wd, opts, stage, unsafe, uid, gid, cb_) { 312 function cb (er) { 313 cb_.apply(null, arguments) 314 opts.log.resume() 315 process.nextTick(dequeue) 316 } 317 318 const [sh, args, conf] = getSpawnArgs({ cmd, wd, opts, uid, gid, unsafe, env }) 319 320 opts.log.verbose('lifecycle', logid(pkg, stage), 'PATH:', env[PATH]) 321 opts.log.verbose('lifecycle', logid(pkg, stage), 'CWD:', wd) 322 opts.log.silly('lifecycle', logid(pkg, stage), 'Args:', args) 323 324 var proc = spawn(sh, args, conf, opts.log) 325 326 proc.on('error', procError) 327 proc.on('close', function (code, signal) { 328 opts.log.silly('lifecycle', logid(pkg, stage), 'Returned: code:', code, ' signal:', signal) 329 if (signal) { 330 process.kill(process.pid, signal) 331 } else if (code) { 332 var er = new Error('Exit status ' + code) 333 er.errno = code 334 } 335 procError(er) 336 }) 337 byline(proc.stdout).on('data', function (data) { 338 opts.log.verbose('lifecycle', logid(pkg, stage), 'stdout', data.toString()) 339 }) 340 byline(proc.stderr).on('data', function (data) { 341 opts.log.verbose('lifecycle', logid(pkg, stage), 'stderr', data.toString()) 342 }) 343 process.once('SIGTERM', procKill) 344 process.once('SIGINT', procInterupt) 345 346 function procError (er) { 347 if (er) { 348 opts.log.info('lifecycle', logid(pkg, stage), 'Failed to exec ' + stage + ' script') 349 er.message = pkg._id + ' ' + stage + ': `' + cmd + '`\n' + 350 er.message 351 if (er.code !== 'EPERM') { 352 er.code = 'ELIFECYCLE' 353 } 354 fs.stat(opts.dir, function (statError, d) { 355 if (statError && statError.code === 'ENOENT' && opts.dir.split(path.sep).slice(-1)[0] === 'node_modules') { 356 opts.log.warn('', 'Local package.json exists, but node_modules missing, did you mean to install?') 357 } 358 }) 359 er.pkgid = pkg._id 360 er.stage = stage 361 er.script = cmd 362 er.pkgname = pkg.name 363 } 364 process.removeListener('SIGTERM', procKill) 365 process.removeListener('SIGTERM', procInterupt) 366 process.removeListener('SIGINT', procKill) 367 process.removeListener('SIGINT', procInterupt) 368 return cb(er) 369 } 370 function procKill () { 371 proc.kill() 372 } 373 function procInterupt () { 374 proc.kill('SIGINT') 375 proc.on('exit', function () { 376 process.exit() 377 }) 378 process.once('SIGINT', procKill) 379 } 380} 381 382function runHookLifecycle (pkg, stage, env, wd, opts, cb) { 383 hookStat(opts.dir, stage, function (er) { 384 if (er) return cb() 385 var cmd = path.join(opts.dir, '.hooks', stage) 386 var note = '\n> ' + pkg._id + ' ' + stage + ' ' + wd + 387 '\n> ' + cmd 388 runCmd(note, cmd, pkg, env, stage, wd, opts, cb) 389 }) 390} 391 392function makeEnv (data, opts, prefix, env) { 393 prefix = prefix || 'npm_package_' 394 if (!env) { 395 env = {} 396 for (var i in process.env) { 397 if (!i.match(/^npm_/)) { 398 env[i] = process.env[i] 399 } 400 } 401 402 // express and others respect the NODE_ENV value. 403 if (opts.production) env.NODE_ENV = 'production' 404 } else if (!data.hasOwnProperty('_lifecycleEnv')) { 405 Object.defineProperty(data, '_lifecycleEnv', 406 { 407 value: env, 408 enumerable: false 409 } 410 ) 411 } 412 413 if (opts.nodeOptions) env.NODE_OPTIONS = opts.nodeOptions 414 415 for (i in data) { 416 if (i.charAt(0) !== '_') { 417 var envKey = (prefix + i).replace(/[^a-zA-Z0-9_]/g, '_') 418 if (i === 'readme') { 419 continue 420 } 421 if (data[i] && typeof data[i] === 'object') { 422 try { 423 // quick and dirty detection for cyclical structures 424 JSON.stringify(data[i]) 425 makeEnv(data[i], opts, envKey + '_', env) 426 } catch (ex) { 427 // usually these are package objects. 428 // just get the path and basic details. 429 var d = data[i] 430 makeEnv( 431 { name: d.name, version: d.version, path: d.path }, 432 opts, 433 envKey + '_', 434 env 435 ) 436 } 437 } else { 438 env[envKey] = String(data[i]) 439 env[envKey] = env[envKey].indexOf('\n') !== -1 440 ? JSON.stringify(env[envKey]) 441 : env[envKey] 442 } 443 } 444 } 445 446 if (prefix !== 'npm_package_') return env 447 448 prefix = 'npm_config_' 449 var pkgConfig = {} 450 var pkgVerConfig = {} 451 var namePref = data.name + ':' 452 var verPref = data.name + '@' + data.version + ':' 453 454 Object.keys(opts.config).forEach(function (i) { 455 // in some rare cases (e.g. working with nerf darts), there are segmented 456 // "private" (underscore-prefixed) config names -- don't export 457 if ((i.charAt(0) === '_' && i.indexOf('_' + namePref) !== 0) || i.match(/:_/)) { 458 return 459 } 460 var value = opts.config[i] 461 if (value instanceof Stream || Array.isArray(value) || typeof value === 'function') return 462 if (i.match(/umask/)) value = umask.toString(value) 463 464 if (!value) value = '' 465 else if (typeof value === 'number') value = '' + value 466 else if (typeof value !== 'string') value = JSON.stringify(value) 467 468 if (typeof value !== 'string') { 469 return 470 } 471 472 value = value.indexOf('\n') !== -1 473 ? JSON.stringify(value) 474 : value 475 i = i.replace(/^_+/, '') 476 var k 477 if (i.indexOf(namePref) === 0) { 478 k = i.substr(namePref.length).replace(/[^a-zA-Z0-9_]/g, '_') 479 pkgConfig[k] = value 480 } else if (i.indexOf(verPref) === 0) { 481 k = i.substr(verPref.length).replace(/[^a-zA-Z0-9_]/g, '_') 482 pkgVerConfig[k] = value 483 } 484 var envKey = (prefix + i).replace(/[^a-zA-Z0-9_]/g, '_') 485 env[envKey] = value 486 }) 487 488 prefix = 'npm_package_config_' 489 ;[pkgConfig, pkgVerConfig].forEach(function (conf) { 490 for (var i in conf) { 491 var envKey = (prefix + i) 492 env[envKey] = conf[i] 493 } 494 }) 495 496 return env 497} 498