| 'use strict'; |
| const path = require('path'); |
| const childProcess = require('child_process'); |
| const crossSpawn = require('cross-spawn'); |
| const stripEof = require('strip-eof'); |
| const npmRunPath = require('npm-run-path'); |
| const isStream = require('is-stream'); |
| const _getStream = require('get-stream'); |
| const pFinally = require('p-finally'); |
| const onExit = require('signal-exit'); |
| const errname = require('./lib/errname'); |
| const stdio = require('./lib/stdio'); |
| |
| const TEN_MEGABYTES = 1000 * 1000 * 10; |
| |
| function handleArgs(cmd, args, opts) { |
| let parsed; |
| |
| opts = Object.assign({ |
| extendEnv: true, |
| env: {} |
| }, opts); |
| |
| if (opts.extendEnv) { |
| opts.env = Object.assign({}, process.env, opts.env); |
| } |
| |
| if (opts.__winShell === true) { |
| delete opts.__winShell; |
| parsed = { |
| command: cmd, |
| args, |
| options: opts, |
| file: cmd, |
| original: { |
| cmd, |
| args |
| } |
| }; |
| } else { |
| parsed = crossSpawn._parse(cmd, args, opts); |
| } |
| |
| opts = Object.assign({ |
| maxBuffer: TEN_MEGABYTES, |
| buffer: true, |
| stripEof: true, |
| preferLocal: true, |
| localDir: parsed.options.cwd || process.cwd(), |
| encoding: 'utf8', |
| reject: true, |
| cleanup: true |
| }, parsed.options); |
| |
| opts.stdio = stdio(opts); |
| |
| if (opts.preferLocal) { |
| opts.env = npmRunPath.env(Object.assign({}, opts, {cwd: opts.localDir})); |
| } |
| |
| if (opts.detached) { |
| // #115 |
| opts.cleanup = false; |
| } |
| |
| if (process.platform === 'win32' && path.basename(parsed.command) === 'cmd.exe') { |
| // #116 |
| parsed.args.unshift('/q'); |
| } |
| |
| return { |
| cmd: parsed.command, |
| args: parsed.args, |
| opts, |
| parsed |
| }; |
| } |
| |
| function handleInput(spawned, input) { |
| if (input === null || input === undefined) { |
| return; |
| } |
| |
| if (isStream(input)) { |
| input.pipe(spawned.stdin); |
| } else { |
| spawned.stdin.end(input); |
| } |
| } |
| |
| function handleOutput(opts, val) { |
| if (val && opts.stripEof) { |
| val = stripEof(val); |
| } |
| |
| return val; |
| } |
| |
| function handleShell(fn, cmd, opts) { |
| let file = '/bin/sh'; |
| let args = ['-c', cmd]; |
| |
| opts = Object.assign({}, opts); |
| |
| if (process.platform === 'win32') { |
| opts.__winShell = true; |
| file = process.env.comspec || 'cmd.exe'; |
| args = ['/s', '/c', `"${cmd}"`]; |
| opts.windowsVerbatimArguments = true; |
| } |
| |
| if (opts.shell) { |
| file = opts.shell; |
| delete opts.shell; |
| } |
| |
| return fn(file, args, opts); |
| } |
| |
| function getStream(process, stream, {encoding, buffer, maxBuffer}) { |
| if (!process[stream]) { |
| return null; |
| } |
| |
| let ret; |
| |
| if (!buffer) { |
| // TODO: Use `ret = util.promisify(stream.finished)(process[stream]);` when targeting Node.js 10 |
| ret = new Promise((resolve, reject) => { |
| process[stream] |
| .once('end', resolve) |
| .once('error', reject); |
| }); |
| } else if (encoding) { |
| ret = _getStream(process[stream], { |
| encoding, |
| maxBuffer |
| }); |
| } else { |
| ret = _getStream.buffer(process[stream], {maxBuffer}); |
| } |
| |
| return ret.catch(err => { |
| err.stream = stream; |
| err.message = `${stream} ${err.message}`; |
| throw err; |
| }); |
| } |
| |
| function makeError(result, options) { |
| const {stdout, stderr} = result; |
| |
| let err = result.error; |
| const {code, signal} = result; |
| |
| const {parsed, joinedCmd} = options; |
| const timedOut = options.timedOut || false; |
| |
| if (!err) { |
| let output = ''; |
| |
| if (Array.isArray(parsed.opts.stdio)) { |
| if (parsed.opts.stdio[2] !== 'inherit') { |
| output += output.length > 0 ? stderr : `\n${stderr}`; |
| } |
| |
| if (parsed.opts.stdio[1] !== 'inherit') { |
| output += `\n${stdout}`; |
| } |
| } else if (parsed.opts.stdio !== 'inherit') { |
| output = `\n${stderr}${stdout}`; |
| } |
| |
| err = new Error(`Command failed: ${joinedCmd}${output}`); |
| err.code = code < 0 ? errname(code) : code; |
| } |
| |
| err.stdout = stdout; |
| err.stderr = stderr; |
| err.failed = true; |
| err.signal = signal || null; |
| err.cmd = joinedCmd; |
| err.timedOut = timedOut; |
| |
| return err; |
| } |
| |
| function joinCmd(cmd, args) { |
| let joinedCmd = cmd; |
| |
| if (Array.isArray(args) && args.length > 0) { |
| joinedCmd += ' ' + args.join(' '); |
| } |
| |
| return joinedCmd; |
| } |
| |
| module.exports = (cmd, args, opts) => { |
| const parsed = handleArgs(cmd, args, opts); |
| const {encoding, buffer, maxBuffer} = parsed.opts; |
| const joinedCmd = joinCmd(cmd, args); |
| |
| let spawned; |
| try { |
| spawned = childProcess.spawn(parsed.cmd, parsed.args, parsed.opts); |
| } catch (err) { |
| return Promise.reject(err); |
| } |
| |
| let removeExitHandler; |
| if (parsed.opts.cleanup) { |
| removeExitHandler = onExit(() => { |
| spawned.kill(); |
| }); |
| } |
| |
| let timeoutId = null; |
| let timedOut = false; |
| |
| const cleanup = () => { |
| if (timeoutId) { |
| clearTimeout(timeoutId); |
| timeoutId = null; |
| } |
| |
| if (removeExitHandler) { |
| removeExitHandler(); |
| } |
| }; |
| |
| if (parsed.opts.timeout > 0) { |
| timeoutId = setTimeout(() => { |
| timeoutId = null; |
| timedOut = true; |
| spawned.kill(parsed.opts.killSignal); |
| }, parsed.opts.timeout); |
| } |
| |
| const processDone = new Promise(resolve => { |
| spawned.on('exit', (code, signal) => { |
| cleanup(); |
| resolve({code, signal}); |
| }); |
| |
| spawned.on('error', err => { |
| cleanup(); |
| resolve({error: err}); |
| }); |
| |
| if (spawned.stdin) { |
| spawned.stdin.on('error', err => { |
| cleanup(); |
| resolve({error: err}); |
| }); |
| } |
| }); |
| |
| function destroy() { |
| if (spawned.stdout) { |
| spawned.stdout.destroy(); |
| } |
| |
| if (spawned.stderr) { |
| spawned.stderr.destroy(); |
| } |
| } |
| |
| const handlePromise = () => pFinally(Promise.all([ |
| processDone, |
| getStream(spawned, 'stdout', {encoding, buffer, maxBuffer}), |
| getStream(spawned, 'stderr', {encoding, buffer, maxBuffer}) |
| ]).then(arr => { |
| const result = arr[0]; |
| result.stdout = arr[1]; |
| result.stderr = arr[2]; |
| |
| if (result.error || result.code !== 0 || result.signal !== null) { |
| const err = makeError(result, { |
| joinedCmd, |
| parsed, |
| timedOut |
| }); |
| |
| // TODO: missing some timeout logic for killed |
| // https://github.com/nodejs/node/blob/master/lib/child_process.js#L203 |
| // err.killed = spawned.killed || killed; |
| err.killed = err.killed || spawned.killed; |
| |
| if (!parsed.opts.reject) { |
| return err; |
| } |
| |
| throw err; |
| } |
| |
| return { |
| stdout: handleOutput(parsed.opts, result.stdout), |
| stderr: handleOutput(parsed.opts, result.stderr), |
| code: 0, |
| failed: false, |
| killed: false, |
| signal: null, |
| cmd: joinedCmd, |
| timedOut: false |
| }; |
| }), destroy); |
| |
| crossSpawn._enoent.hookChildProcess(spawned, parsed.parsed); |
| |
| handleInput(spawned, parsed.opts.input); |
| |
| spawned.then = (onfulfilled, onrejected) => handlePromise().then(onfulfilled, onrejected); |
| spawned.catch = onrejected => handlePromise().catch(onrejected); |
| |
| return spawned; |
| }; |
| |
| // TODO: set `stderr: 'ignore'` when that option is implemented |
| module.exports.stdout = (...args) => module.exports(...args).then(x => x.stdout); |
| |
| // TODO: set `stdout: 'ignore'` when that option is implemented |
| module.exports.stderr = (...args) => module.exports(...args).then(x => x.stderr); |
| |
| module.exports.shell = (cmd, opts) => handleShell(module.exports, cmd, opts); |
| |
| module.exports.sync = (cmd, args, opts) => { |
| const parsed = handleArgs(cmd, args, opts); |
| const joinedCmd = joinCmd(cmd, args); |
| |
| if (isStream(parsed.opts.input)) { |
| throw new TypeError('The `input` option cannot be a stream in sync mode'); |
| } |
| |
| const result = childProcess.spawnSync(parsed.cmd, parsed.args, parsed.opts); |
| result.code = result.status; |
| |
| if (result.error || result.status !== 0 || result.signal !== null) { |
| const err = makeError(result, { |
| joinedCmd, |
| parsed |
| }); |
| |
| if (!parsed.opts.reject) { |
| return err; |
| } |
| |
| throw err; |
| } |
| |
| return { |
| stdout: handleOutput(parsed.opts, result.stdout), |
| stderr: handleOutput(parsed.opts, result.stderr), |
| code: 0, |
| failed: false, |
| signal: null, |
| cmd: joinedCmd, |
| timedOut: false |
| }; |
| }; |
| |
| module.exports.shellSync = (cmd, opts) => handleShell(module.exports.sync, cmd, opts); |