Run a lifecycle script for a package (descendant of npm-lifecycle)
npm install @npmcli/run-scriptRun a lifecycle script for a package (descendant of npm-lifecycle)
``js
const runScript = require('@npmcli/run-script')
runScript({
// required, the script to run
event: 'install',
// extra args to pass to the command, defaults to []
args: [],
// required, the folder where the package lives
path: '/path/to/package/folder',
// optional, these paths will be put at the beginning of $PATH, evenprocess.cwd()
// after run-script adds the node_modules/.bin folder(s) from
// . This is for commands like npm init, npm exec,npx
// and to make sure manually installed packages come beforeprocess.cwd()
// anything that happens to be in the tree in .
binPaths: [
'/path/to/npx/node_modules/.bin',
'/path/to/npm/prefix/node_modules/.bin',
],
// optional, defaults to /bin/sh on unix, or cmd.exe on windows
scriptShell: '/bin/bash',
// optional, passed directly to @npmcli/promise-spawn which defaults it to true
// return stdout and stderr as strings rather than buffers
stdioString: false,
// optional, additional environment variables to add
// note that process.env IS inherited by default
// Always set:
// - npm_package_json The package.json file in the folder
// - npm_lifecycle_event The event that this is being run for
// - npm_lifecycle_script The script being run
// The fields described in https://github.com/npm/rfcs/pull/183
env: {
npm_package_from: 'foo@bar',
npm_package_resolved: 'https://registry.npmjs.org/foo/-/foo-1.2.3.tgz',
npm_package_integrity: 'sha512-foobarbaz',
},
// defaults to 'pipe'. Can also pass an array like you would to node's
// exec or spawn functions. Note that if it's anything other than
// 'pipe' then the stdout/stderr values on the result will be missing.
// npm cli sets this to 'inherit' for explicit run-scripts (test, etc.)
// but leaves it as 'pipe' for install scripts that run in parallel.
stdio: 'inherit',
// print the package id and script, and the command to be run, like:
// > somepackage@1.2.3 postinstall
// > make all-the-things
})
.then(({ code, signal, stdout, stderr, pkgid, path, event, script }) => {
// do something with the results
})
.catch(er => {
// command did not work.
// er is decorated with:
// - code
// - signal
// - stdout
// - stderr
// - path
// - pkgid (name@version string)
// - event
// - script
})
`
Call the exported runScript function with an options object.
Returns a promise that resolves to the result of the execution. Promise
rejects if the execution fails (exits non-zero) or has any other error.
Rejected errors are decorated with the same values as the result object.
If the stdio options mean that it'll have a piped stdin, then the stdin is
ended immediately on the child process. If stdin is shared with the parent
terminal, then it is up to the user to end it, of course.
- code Process exit codesignal
- Process exit signalstdout
- stdout data (Buffer, or String when stdioString set to true)stderr
- stderr data (Buffer, or String when stdioString set to true)path
- Path to the package executing its scriptevent
- Lifecycle event being runscript
- Command being run
If stdio is inherit this package will emit a banner with the packageproc-log.output.standard
name and version, event name, and script command to be run, and send it
to . Consuming
libraries can decide whether or not to display this.
- path Required. The path to the package having its script run.event
- Required. The event being executed.args
- Optional, default []. Extra arguments to pass to the script.env
- Optional, object of fields to add to the environment of thenpm_package_json
subprocess. Note that process.env IS inherited by default These are
always set:
- The package.json file in the foldernpm_lifecycle_event
- The event that this is being run fornpm_lifecycle_script
- The script being runpackage.json
- The fields described inscriptShell
RFC183.
- Optional, defaults to /bin/sh on Unix, defaults toenv.ComSpec
or cmd on Windows. Custom script to use to execute thestdio
command.
- Optional, defaults to 'pipe'. The same as the stdio argumentchild_process
passed to functions in Node.js. Note that if a stdiopipe
output is set to anything other than , it will not be present incmd
the result/error object.
- Optional. Override the script from the package.json withstdioString
something else, which will be run in an otherwise matching environment.
- Optional, passed directly to @npmcli/promise-spawn whichtrue
defaults it to . Return string values for stderr and stdout rather
than Buffers.
Note that this does _not_ run pre-event and post-event scripts. The
caller has to manage that process themselves.
This is an implementation to satisfy RFC
90, RFC
77, and RFC
73.
Apart from those behavior changes in npm v7, this is also just refresh of
the codebase, with modern coding techniques and better test coverage.
Functionally, this means:
- Output is not dumped to the top level process's stdio by default.
- Less stuff is put into the environment.
- It is not opinionated about logging. (So, at least with the logging
framework in npm v7.0 and before, the caller has to call
log.disableProgress() and log.enableProgress() at the appropriatenode
times, if necessary.)
- The directory containing the executable is _never_ added to thePATH
environment variable. (Ie, --scripts-prepend-node-path isfalse
effectively always set to .) Doing so causes more unintended sidecmd
effects than it ever prevented.
- Hook scripts are not run by this module. If the caller wishes to run
hook scripts, then they can override the default package script with an
explicit option pointing to the node_modules/.hook/${event}
script.
In order to ensure that arguments are handled consistently, this module
writes a temporary script file containing the command as it exists in
the package.json, followed by the user supplied arguments having been
escaped to ensure they are processed as literal strings. We then instruct
the shell to execute the script file, and when the process exits we remove
the temporary file.
In Windows, when the shell is cmd, and when the initial command in the script
is a known batch file (i.e. something.cmd) we double escape additional
arguments so that the shim scripts npm installs work correctly.
The actual implementation of the escaping is in lib/escape.js`.