teen_process 1.15.0 → 2.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,8 +1,6 @@
1
1
  node-teen_process
2
2
  =================
3
3
 
4
- [![Greenkeeper badge](https://badges.greenkeeper.io/appium/node-teen_process.svg)](https://greenkeeper.io/)
5
-
6
4
  A grown-up version of Node's child_process. `exec` is really useful, but it
7
5
  suffers many limitations. This is an es7 (`async`/`await`) implementation of
8
6
  `exec` that uses `spawn` under the hood. It takes care of wrapping commands and
package/build/lib/exec.js CHANGED
@@ -5,8 +5,8 @@ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefau
5
5
  Object.defineProperty(exports, "__esModule", {
6
6
  value: true
7
7
  });
8
- exports.exec = exec;
9
8
  exports.default = void 0;
9
+ exports.exec = exec;
10
10
 
11
11
  require("source-map-support/register");
12
12
 
@@ -18,11 +18,13 @@ var _bluebird = _interopRequireDefault(require("bluebird"));
18
18
 
19
19
  var _lodash = _interopRequireDefault(require("lodash"));
20
20
 
21
+ var _helpers = require("./helpers");
22
+
21
23
  const MAX_BUFFER_SIZE = 100 * 1024 * 1024;
22
24
 
23
25
  async function exec(cmd, args = [], opts = {}) {
24
26
  const rep = (0, _shellQuote.quote)([cmd, ...args]);
25
- opts = Object.assign({
27
+ opts = _lodash.default.defaults(opts, {
26
28
  timeout: null,
27
29
  encoding: 'utf8',
28
30
  killSignal: 'SIGTERM',
@@ -35,7 +37,8 @@ async function exec(cmd, args = [], opts = {}) {
35
37
  logger: undefined,
36
38
  maxStdoutBufferSize: MAX_BUFFER_SIZE,
37
39
  maxStderrBufferSize: MAX_BUFFER_SIZE
38
- }, opts);
40
+ });
41
+ const isBuffer = Boolean(opts.isBuffer);
39
42
  return await new _bluebird.default((resolve, reject) => {
40
43
  let proc = (0, _child_process.spawn)(cmd, args, {
41
44
  cwd: opts.cwd,
@@ -46,13 +49,11 @@ async function exec(cmd, args = [], opts = {}) {
46
49
  stderrArr = [],
47
50
  timer = null;
48
51
  proc.on('error', err => {
49
- let msg = `Command '${rep}' errored out: ${err.stack}`;
50
-
51
52
  if (err.errno === 'ENOENT') {
52
- msg = `Command '${cmd}' not found. Is it installed?`;
53
+ err = (0, _helpers.formatEnoent)(err, cmd, opts.cwd);
53
54
  }
54
55
 
55
- reject(new Error(msg));
56
+ reject(err);
56
57
  });
57
58
 
58
59
  if (proc.stdin) {
@@ -129,7 +130,7 @@ async function exec(cmd, args = [], opts = {}) {
129
130
  let {
130
131
  stdout,
131
132
  stderr
132
- } = getStdio(opts.isBuffer);
133
+ } = getStdio(isBuffer);
133
134
 
134
135
  if (code === 0) {
135
136
  resolve({
@@ -153,7 +154,7 @@ async function exec(cmd, args = [], opts = {}) {
153
154
  let {
154
155
  stdout,
155
156
  stderr
156
- } = getStdio(opts.isBuffer);
157
+ } = getStdio(isBuffer);
157
158
  let err = new Error(`Command '${rep}' timed out after ${opts.timeout}ms`);
158
159
  err = Object.assign(err, {
159
160
  stdout,
@@ -168,7 +169,5 @@ async function exec(cmd, args = [], opts = {}) {
168
169
  }
169
170
 
170
171
  var _default = exec;
171
- exports.default = _default;require('source-map-support').install();
172
-
173
-
174
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/exec.js"],"names":["MAX_BUFFER_SIZE","exec","cmd","args","opts","rep","Object","assign","timeout","encoding","killSignal","cwd","undefined","env","process","ignoreOutput","stdio","isBuffer","shell","logger","maxStdoutBufferSize","maxStderrBufferSize","B","resolve","reject","proc","stdoutArr","stderrArr","timer","on","err","msg","stack","errno","Error","stdin","syscall","handleStream","streamType","streamProps","_","capitalize","chunks","maxSize","size","chunk","push","length","shift","isFunction","debug","toString","getStdio","stdout","stderr","Buffer","concat","code","clearTimeout","setTimeout","kill"],"mappings":"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","sourcesContent":["/* eslint-disable promise/prefer-await-to-callbacks */\n\nimport { spawn } from 'child_process';\nimport { quote } from 'shell-quote';\nimport B from 'bluebird';\nimport _ from 'lodash';\n\nconst MAX_BUFFER_SIZE = 100 * 1024 * 1024;\n\nasync function exec (cmd, args = [], opts = {}) {\n  // get a quoted representation of the command for error strings\n  const rep = quote([cmd, ...args]);\n\n  // extend default options; we're basically re-implementing exec's options\n  // for use here with spawn under the hood\n  opts = Object.assign({\n    timeout: null,\n    encoding: 'utf8',\n    killSignal: 'SIGTERM',\n    cwd: undefined,\n    env: process.env,\n    ignoreOutput: false,\n    stdio: 'inherit',\n    isBuffer: false,\n    shell: undefined,\n    logger: undefined,\n    maxStdoutBufferSize: MAX_BUFFER_SIZE,\n    maxStderrBufferSize: MAX_BUFFER_SIZE,\n  }, opts);\n\n  // this is an async function, so return a promise\n  return await new B((resolve, reject) => {\n    // spawn the child process with options; we don't currently expose any of\n    // the other 'spawn' options through the API\n    let proc = spawn(cmd, args, {cwd: opts.cwd, env: opts.env, shell: opts.shell});\n    let stdoutArr = [], stderrArr = [], timer = null;\n\n    // if the process errors out, reject the promise\n    proc.on('error', (err) => {\n      let msg = `Command '${rep}' errored out: ${err.stack}`;\n      if (err.errno === 'ENOENT') {\n        msg = `Command '${cmd}' not found. Is it installed?`;\n      }\n      reject(new Error(msg));\n    });\n    if (proc.stdin) {\n      proc.stdin.on('error', (err) => {\n        reject(new Error(`Standard input '${err.syscall}' error: ${err.stack}`));\n      });\n    }\n    const handleStream = (streamType, streamProps) => {\n      if (!proc[streamType]) {\n        return;\n      }\n\n      proc[streamType].on('error', (err) => {\n        reject(new Error(`${_.capitalize(streamType)} '${err.syscall}' error: ${err.stack}`));\n      });\n\n      if (opts.ignoreOutput) {\n        // https://github.com/nodejs/node/issues/4236\n        proc[streamType].on('data', () => {});\n        return;\n      }\n\n      // keep track of the stream if we don't want to ignore it\n      const {chunks, maxSize} = streamProps;\n      let size = 0;\n      proc[streamType].on('data', (chunk) => {\n        chunks.push(chunk);\n        size += chunk.length;\n        while (chunks.length > 1 && size >= maxSize) {\n          size -= chunks[0].length;\n          chunks.shift();\n        }\n        if (opts.logger && _.isFunction(opts.logger.debug)) {\n          opts.logger.debug(chunk.toString());\n        }\n      });\n    };\n    handleStream('stdout', {\n      maxSize: opts.maxStdoutBufferSize,\n      chunks: stdoutArr,\n    });\n    handleStream('stderr', {\n      maxSize: opts.maxStderrBufferSize,\n      chunks: stderrArr,\n    });\n\n    function getStdio (isBuffer) {\n      let stdout, stderr;\n      if (isBuffer) {\n        stdout = Buffer.concat(stdoutArr);\n        stderr = Buffer.concat(stderrArr);\n      } else {\n        stdout = Buffer.concat(stdoutArr).toString(opts.encoding);\n        stderr = Buffer.concat(stderrArr).toString(opts.encoding);\n      }\n      return {stdout, stderr};\n    }\n\n    // if the process ends, either resolve or reject the promise based on the\n    // exit code of the process. either way, attach stdout, stderr, and code.\n    // Also clean up the timer if it exists\n    proc.on('close', (code) => {\n      if (timer) {\n        clearTimeout(timer);\n      }\n      let {stdout, stderr} = getStdio(opts.isBuffer);\n      if (code === 0) {\n        resolve({stdout, stderr, code});\n      } else {\n        let err = new Error(`Command '${rep}' exited with code ${code}`);\n        err = Object.assign(err, {stdout, stderr, code});\n        reject(err);\n      }\n    });\n\n    // if we set a timeout on the child process, cut into the execution and\n    // reject if the timeout is reached. Attach the stdout/stderr we currently\n    // have in case it's helpful in debugging\n    if (opts.timeout) {\n      timer = setTimeout(() => {\n        let {stdout, stderr} = getStdio(opts.isBuffer);\n        let err = new Error(`Command '${rep}' timed out after ${opts.timeout}ms`);\n        err = Object.assign(err, {stdout, stderr, code: null});\n        reject(err);\n        // reject and THEN kill to avoid race conditions with the handlers\n        // above\n        proc.kill(opts.killSignal);\n      }, opts.timeout);\n    }\n  });\n}\n\nexport { exec };\nexport default exec;\n"],"file":"lib/exec.js","sourceRoot":"../.."}
172
+ exports.default = _default;
173
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["MAX_BUFFER_SIZE","exec","cmd","args","opts","rep","quote","_","defaults","timeout","encoding","killSignal","cwd","undefined","env","process","ignoreOutput","stdio","isBuffer","shell","logger","maxStdoutBufferSize","maxStderrBufferSize","Boolean","B","resolve","reject","proc","spawn","stdoutArr","stderrArr","timer","on","err","errno","formatEnoent","stdin","Error","syscall","stack","handleStream","streamType","streamProps","capitalize","chunks","maxSize","size","chunk","push","length","shift","isFunction","debug","toString","getStdio","stdout","stderr","Buffer","concat","code","clearTimeout","Object","assign","setTimeout","kill"],"sources":["../../lib/exec.js"],"sourcesContent":["/* eslint-disable promise/prefer-await-to-callbacks */\n\nimport { spawn } from 'child_process';\nimport { quote } from 'shell-quote';\nimport B from 'bluebird';\nimport _ from 'lodash';\nimport { formatEnoent } from './helpers';\n\nconst MAX_BUFFER_SIZE = 100 * 1024 * 1024;\n\n/**\n * Spawns a process\n * @template {TeenProcessExecOptions} T\n * @param {string} cmd - Program to execute\n * @param {string[]} [args] - Arguments to pass to the program\n * @param {T} [opts] - Options\n * @returns {Promise<BufferProp<T> extends true ? TeenProcessExecBufferResult : TeenProcessExecStringResult>}\n */\nasync function exec (cmd, args = [], opts = /** @type {T} */({})) {\n  // get a quoted representation of the command for error strings\n  const rep = quote([cmd, ...args]);\n\n  // extend default options; we're basically re-implementing exec's options\n  // for use here with spawn under the hood\n  opts = /** @type {T} */(_.defaults(opts, {\n    timeout: null,\n    encoding: 'utf8',\n    killSignal: 'SIGTERM',\n    cwd: undefined,\n    env: process.env,\n    ignoreOutput: false,\n    stdio: 'inherit',\n    isBuffer: false,\n    shell: undefined,\n    logger: undefined,\n    maxStdoutBufferSize: MAX_BUFFER_SIZE,\n    maxStderrBufferSize: MAX_BUFFER_SIZE,\n  }));\n\n  const isBuffer = Boolean(opts.isBuffer);\n\n  // this is an async function, so return a promise\n  return await new B((resolve, reject) => {\n    // spawn the child process with options; we don't currently expose any of\n    // the other 'spawn' options through the API\n    let proc = spawn(cmd, args, {cwd: opts.cwd, env: opts.env, shell: opts.shell});\n    let stdoutArr = [], stderrArr = [], timer = null;\n\n    // if the process errors out, reject the promise\n    proc.on('error', /** @param {NodeJS.ErrnoException} err */(err) => {\n      // @ts-ignore\n      if (err.errno === 'ENOENT') {\n        err = formatEnoent(err, cmd, opts.cwd);\n      }\n      reject(err);\n    });\n    if (proc.stdin) {\n      proc.stdin.on('error', /** @param {NodeJS.ErrnoException} err */(err) => {\n        reject(new Error(`Standard input '${err.syscall}' error: ${err.stack}`));\n      });\n    }\n    const handleStream = (streamType, streamProps) => {\n      if (!proc[streamType]) {\n        return;\n      }\n\n      proc[streamType].on('error', (err) => {\n        reject(new Error(`${_.capitalize(streamType)} '${err.syscall}' error: ${err.stack}`));\n      });\n\n      if (opts.ignoreOutput) {\n        // https://github.com/nodejs/node/issues/4236\n        proc[streamType].on('data', () => {});\n        return;\n      }\n\n      // keep track of the stream if we don't want to ignore it\n      const {chunks, maxSize} = streamProps;\n      let size = 0;\n      proc[streamType].on('data', (chunk) => {\n        chunks.push(chunk);\n        size += chunk.length;\n        while (chunks.length > 1 && size >= maxSize) {\n          size -= chunks[0].length;\n          chunks.shift();\n        }\n        if (opts.logger && _.isFunction(opts.logger.debug)) {\n          opts.logger.debug(chunk.toString());\n        }\n      });\n    };\n    handleStream('stdout', {\n      maxSize: opts.maxStdoutBufferSize,\n      chunks: stdoutArr,\n    });\n    handleStream('stderr', {\n      maxSize: opts.maxStderrBufferSize,\n      chunks: stderrArr,\n    });\n\n    /**\n     * @template {boolean} U\n     * @param {U} isBuffer\n     * @returns {U extends true ? {stdout: Buffer, stderr: Buffer} : {stdout: string, stderr: string}}\n     */\n    function getStdio (isBuffer) {\n      let stdout, stderr;\n      if (isBuffer) {\n        stdout = Buffer.concat(stdoutArr);\n        stderr = Buffer.concat(stderrArr);\n      } else {\n        stdout = Buffer.concat(stdoutArr).toString(opts.encoding);\n        stderr = Buffer.concat(stderrArr).toString(opts.encoding);\n      }\n      return /** @type {U extends true ? {stdout: Buffer, stderr: Buffer} : {stdout: string, stderr: string}} */({stdout, stderr});\n    }\n\n    // if the process ends, either resolve or reject the promise based on the\n    // exit code of the process. either way, attach stdout, stderr, and code.\n    // Also clean up the timer if it exists\n    proc.on('close', (code) => {\n      if (timer) {\n        clearTimeout(timer);\n      }\n      let {stdout, stderr} = getStdio(isBuffer);\n      if (code === 0) {\n        resolve(/** @type {BufferProp<T> extends true ? TeenProcessExecBufferResult : TeenProcessExecStringResult} */({stdout, stderr, code}));\n      } else {\n        let err = new Error(`Command '${rep}' exited with code ${code}`);\n        err = Object.assign(err, {stdout, stderr, code});\n        reject(err);\n      }\n    });\n\n    // if we set a timeout on the child process, cut into the execution and\n    // reject if the timeout is reached. Attach the stdout/stderr we currently\n    // have in case it's helpful in debugging\n    if (opts.timeout) {\n      timer = setTimeout(() => {\n        let {stdout, stderr} = getStdio(isBuffer);\n        let err = new Error(`Command '${rep}' timed out after ${opts.timeout}ms`);\n        err = Object.assign(err, {stdout, stderr, code: null});\n        reject(err);\n        // reject and THEN kill to avoid race conditions with the handlers\n        // above\n        proc.kill(opts.killSignal);\n      }, opts.timeout);\n    }\n  });\n}\n\nexport { exec };\nexport default exec;\n\n/**\n * Options on top of `SpawnOptions`, unique to `teen_process.`\n * @typedef {Object} TeenProcessProps\n * @property {boolean} [ignoreOutput] - Ignore & discard all output\n * @property {boolean} [isBuffer] - Return output as a Buffer\n * @property {TeenProcessLogger} [logger] - Logger to use for debugging\n * @property {number} [maxStdoutBufferSize] - Maximum size of `stdout` buffer\n * @property {number} [maxStderrBufferSize] - Maximum size of `stderr` buffer\n * @property {BufferEncoding} [encoding='utf8'] - Encoding to use for output\n */\n\n/**\n * A logger object understood by {@link exec teen_process.exec}.\n * @typedef {Object} TeenProcessLogger\n * @property {(...args: any[]) => void} debug\n */\n\n/**\n * Options for {@link exec teen_process.exec}.\n * @typedef {import('child_process').SpawnOptions & TeenProcessProps} TeenProcessExecOptions\n */\n\n/**\n * The value {@link exec teen_process.exec} resolves to when `isBuffer` is `false`\n * @typedef {Object} TeenProcessExecStringResult\n * @property {string} stdout - Stdout\n * @property {string} stderr - Stderr\n * @property {number?} code - Exit code\n */\n\n/**\n * The value {@link exec teen_process.exec} resolves to when `isBuffer` is `true`\n * @typedef {Object} TeenProcessExecBufferResult\n * @property {Buffer} stdout - Stdout\n * @property {Buffer} stderr - Stderr\n * @property {number?} code - Exit code\n */\n\n/**\n * Extra props {@link exec teen_process.exec} adds to its error objects\n * @typedef {Object} TeenProcessExecErrorProps\n * @property {string} stdout - STDOUT\n * @property {string} stderr - STDERR\n * @property {number?} code - Exit code\n */\n\n/**\n * Error thrown by {@link exec teen_process.exec}\n * @typedef {Error & TeenProcessExecErrorProps} TeenProcessExecError\n */\n\n/**\n * @template {{isBuffer?: boolean}} MaybeBuffer\n * @typedef {MaybeBuffer['isBuffer']} BufferProp\n * @private\n */\n"],"mappings":";;;;;;;;;;;;AAEA;;AACA;;AACA;;AACA;;AACA;;AAEA,MAAMA,eAAe,GAAG,MAAM,IAAN,GAAa,IAArC;;AAUA,eAAeC,IAAf,CAAqBC,GAArB,EAA0BC,IAAI,GAAG,EAAjC,EAAqCC,IAAI,GAAoB,EAA7D,EAAkE;EAEhE,MAAMC,GAAG,GAAG,IAAAC,iBAAA,EAAM,CAACJ,GAAD,EAAM,GAAGC,IAAT,CAAN,CAAZ;EAIAC,IAAI,GAAoBG,eAAA,CAAEC,QAAF,CAAWJ,IAAX,EAAiB;IACvCK,OAAO,EAAE,IAD8B;IAEvCC,QAAQ,EAAE,MAF6B;IAGvCC,UAAU,EAAE,SAH2B;IAIvCC,GAAG,EAAEC,SAJkC;IAKvCC,GAAG,EAAEC,OAAO,CAACD,GAL0B;IAMvCE,YAAY,EAAE,KANyB;IAOvCC,KAAK,EAAE,SAPgC;IAQvCC,QAAQ,EAAE,KAR6B;IASvCC,KAAK,EAAEN,SATgC;IAUvCO,MAAM,EAAEP,SAV+B;IAWvCQ,mBAAmB,EAAErB,eAXkB;IAYvCsB,mBAAmB,EAAEtB;EAZkB,CAAjB,CAAxB;EAeA,MAAMkB,QAAQ,GAAGK,OAAO,CAACnB,IAAI,CAACc,QAAN,CAAxB;EAGA,OAAO,MAAM,IAAIM,iBAAJ,CAAM,CAACC,OAAD,EAAUC,MAAV,KAAqB;IAGtC,IAAIC,IAAI,GAAG,IAAAC,oBAAA,EAAM1B,GAAN,EAAWC,IAAX,EAAiB;MAACS,GAAG,EAAER,IAAI,CAACQ,GAAX;MAAgBE,GAAG,EAAEV,IAAI,CAACU,GAA1B;MAA+BK,KAAK,EAAEf,IAAI,CAACe;IAA3C,CAAjB,CAAX;IACA,IAAIU,SAAS,GAAG,EAAhB;IAAA,IAAoBC,SAAS,GAAG,EAAhC;IAAA,IAAoCC,KAAK,GAAG,IAA5C;IAGAJ,IAAI,CAACK,EAAL,CAAQ,OAAR,EAA2DC,GAAD,IAAS;MAEjE,IAAIA,GAAG,CAACC,KAAJ,KAAc,QAAlB,EAA4B;QAC1BD,GAAG,GAAG,IAAAE,qBAAA,EAAaF,GAAb,EAAkB/B,GAAlB,EAAuBE,IAAI,CAACQ,GAA5B,CAAN;MACD;;MACDc,MAAM,CAACO,GAAD,CAAN;IACD,CAND;;IAOA,IAAIN,IAAI,CAACS,KAAT,EAAgB;MACdT,IAAI,CAACS,KAAL,CAAWJ,EAAX,CAAc,OAAd,EAAiEC,GAAD,IAAS;QACvEP,MAAM,CAAC,IAAIW,KAAJ,CAAW,mBAAkBJ,GAAG,CAACK,OAAQ,YAAWL,GAAG,CAACM,KAAM,EAA9D,CAAD,CAAN;MACD,CAFD;IAGD;;IACD,MAAMC,YAAY,GAAG,CAACC,UAAD,EAAaC,WAAb,KAA6B;MAChD,IAAI,CAACf,IAAI,CAACc,UAAD,CAAT,EAAuB;QACrB;MACD;;MAEDd,IAAI,CAACc,UAAD,CAAJ,CAAiBT,EAAjB,CAAoB,OAApB,EAA8BC,GAAD,IAAS;QACpCP,MAAM,CAAC,IAAIW,KAAJ,CAAW,GAAE9B,eAAA,CAAEoC,UAAF,CAAaF,UAAb,CAAyB,KAAIR,GAAG,CAACK,OAAQ,YAAWL,GAAG,CAACM,KAAM,EAA3E,CAAD,CAAN;MACD,CAFD;;MAIA,IAAInC,IAAI,CAACY,YAAT,EAAuB;QAErBW,IAAI,CAACc,UAAD,CAAJ,CAAiBT,EAAjB,CAAoB,MAApB,EAA4B,MAAM,CAAE,CAApC;QACA;MACD;;MAGD,MAAM;QAACY,MAAD;QAASC;MAAT,IAAoBH,WAA1B;MACA,IAAII,IAAI,GAAG,CAAX;MACAnB,IAAI,CAACc,UAAD,CAAJ,CAAiBT,EAAjB,CAAoB,MAApB,EAA6Be,KAAD,IAAW;QACrCH,MAAM,CAACI,IAAP,CAAYD,KAAZ;QACAD,IAAI,IAAIC,KAAK,CAACE,MAAd;;QACA,OAAOL,MAAM,CAACK,MAAP,GAAgB,CAAhB,IAAqBH,IAAI,IAAID,OAApC,EAA6C;UAC3CC,IAAI,IAAIF,MAAM,CAAC,CAAD,CAAN,CAAUK,MAAlB;UACAL,MAAM,CAACM,KAAP;QACD;;QACD,IAAI9C,IAAI,CAACgB,MAAL,IAAeb,eAAA,CAAE4C,UAAF,CAAa/C,IAAI,CAACgB,MAAL,CAAYgC,KAAzB,CAAnB,EAAoD;UAClDhD,IAAI,CAACgB,MAAL,CAAYgC,KAAZ,CAAkBL,KAAK,CAACM,QAAN,EAAlB;QACD;MACF,CAVD;IAWD,CA7BD;;IA8BAb,YAAY,CAAC,QAAD,EAAW;MACrBK,OAAO,EAAEzC,IAAI,CAACiB,mBADO;MAErBuB,MAAM,EAAEf;IAFa,CAAX,CAAZ;IAIAW,YAAY,CAAC,QAAD,EAAW;MACrBK,OAAO,EAAEzC,IAAI,CAACkB,mBADO;MAErBsB,MAAM,EAAEd;IAFa,CAAX,CAAZ;;IAUA,SAASwB,QAAT,CAAmBpC,QAAnB,EAA6B;MAC3B,IAAIqC,MAAJ,EAAYC,MAAZ;;MACA,IAAItC,QAAJ,EAAc;QACZqC,MAAM,GAAGE,MAAM,CAACC,MAAP,CAAc7B,SAAd,CAAT;QACA2B,MAAM,GAAGC,MAAM,CAACC,MAAP,CAAc5B,SAAd,CAAT;MACD,CAHD,MAGO;QACLyB,MAAM,GAAGE,MAAM,CAACC,MAAP,CAAc7B,SAAd,EAAyBwB,QAAzB,CAAkCjD,IAAI,CAACM,QAAvC,CAAT;QACA8C,MAAM,GAAGC,MAAM,CAACC,MAAP,CAAc5B,SAAd,EAAyBuB,QAAzB,CAAkCjD,IAAI,CAACM,QAAvC,CAAT;MACD;;MACD,OAA2G;QAAC6C,MAAD;QAASC;MAAT,CAA3G;IACD;;IAKD7B,IAAI,CAACK,EAAL,CAAQ,OAAR,EAAkB2B,IAAD,IAAU;MACzB,IAAI5B,KAAJ,EAAW;QACT6B,YAAY,CAAC7B,KAAD,CAAZ;MACD;;MACD,IAAI;QAACwB,MAAD;QAASC;MAAT,IAAmBF,QAAQ,CAACpC,QAAD,CAA/B;;MACA,IAAIyC,IAAI,KAAK,CAAb,EAAgB;QACdlC,OAAO,CAAuG;UAAC8B,MAAD;UAASC,MAAT;UAAiBG;QAAjB,CAAvG,CAAP;MACD,CAFD,MAEO;QACL,IAAI1B,GAAG,GAAG,IAAII,KAAJ,CAAW,YAAWhC,GAAI,sBAAqBsD,IAAK,EAApD,CAAV;QACA1B,GAAG,GAAG4B,MAAM,CAACC,MAAP,CAAc7B,GAAd,EAAmB;UAACsB,MAAD;UAASC,MAAT;UAAiBG;QAAjB,CAAnB,CAAN;QACAjC,MAAM,CAACO,GAAD,CAAN;MACD;IACF,CAZD;;IAiBA,IAAI7B,IAAI,CAACK,OAAT,EAAkB;MAChBsB,KAAK,GAAGgC,UAAU,CAAC,MAAM;QACvB,IAAI;UAACR,MAAD;UAASC;QAAT,IAAmBF,QAAQ,CAACpC,QAAD,CAA/B;QACA,IAAIe,GAAG,GAAG,IAAII,KAAJ,CAAW,YAAWhC,GAAI,qBAAoBD,IAAI,CAACK,OAAQ,IAA3D,CAAV;QACAwB,GAAG,GAAG4B,MAAM,CAACC,MAAP,CAAc7B,GAAd,EAAmB;UAACsB,MAAD;UAASC,MAAT;UAAiBG,IAAI,EAAE;QAAvB,CAAnB,CAAN;QACAjC,MAAM,CAACO,GAAD,CAAN;QAGAN,IAAI,CAACqC,IAAL,CAAU5D,IAAI,CAACO,UAAf;MACD,CARiB,EAQfP,IAAI,CAACK,OARU,CAAlB;IASD;EACF,CA1GY,CAAb;AA2GD;;eAGcR,I"}
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+
3
+ var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
+
5
+ Object.defineProperty(exports, "__esModule", {
6
+ value: true
7
+ });
8
+ exports.formatEnoent = formatEnoent;
9
+
10
+ require("source-map-support/register");
11
+
12
+ var _which = _interopRequireDefault(require("which"));
13
+
14
+ var _fs = _interopRequireDefault(require("fs"));
15
+
16
+ function formatEnoent(error, cmd, cwd = null) {
17
+ try {
18
+ _which.default.sync(cmd);
19
+
20
+ if (cwd) {
21
+ try {
22
+ _fs.default.accessSync(cwd, _fs.default.constants.R_OK);
23
+ } catch (ign) {
24
+ error.message = `The current working directory '${cwd}' for '${cmd}' command ` + `either does not exist or is not accessible`;
25
+ }
26
+ }
27
+ } catch (ign) {
28
+ error.message = `Command '${cmd}' not found. Is it installed?`;
29
+ }
30
+
31
+ return error;
32
+ }
33
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmb3JtYXRFbm9lbnQiLCJlcnJvciIsImNtZCIsImN3ZCIsIndoaWNoIiwic3luYyIsImZzIiwiYWNjZXNzU3luYyIsImNvbnN0YW50cyIsIlJfT0siLCJpZ24iLCJtZXNzYWdlIl0sInNvdXJjZXMiOlsiLi4vLi4vbGliL2hlbHBlcnMuanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHdoaWNoIGZyb20gJ3doaWNoJztcbmltcG9ydCBmcyBmcm9tICdmcyc7XG5cbi8qKlxuICogRGVjb3JhdGVzIEVOT0VOVCBlcnJvciByZWNlaXZlZCBmcm9tIGEgc3Bhd24gc3lzdGVtIGNhbGxcbiAqIHdpdGggYSBtb3JlIGRlc2NyaXB0aXZlIG1lc3NhZ2UsIHNvIGl0IGNvdWxkIGJlIHByb3Blcmx5IGhhbmRsZWQgYnkgYSB1c2VyLlxuICpcbiAqIEBwYXJhbSB7Tm9kZUpTLkVycm5vRXhjZXB0aW9ufSBlcnJvciBPcmlnaW5hbCBlcnJvciBpbnN0YW5jZS4gISEhIFRoZSBpbnN0YW5jZSBpcyBtdXRhdGVkIGFmdGVyXG4gKiB0aGlzIGhlbHBlciBmdW5jdGlvbiBpbnZvY2F0aW9uXG4gKiBAcGFyYW0ge3N0cmluZ30gY21kIE9yaWdpbmFsIGNvbW1hbmQgdG8gZXhlY3V0ZVxuICogQHBhcmFtIHtzdHJpbmd8VVJMP30gW2N3ZF0gT3B0aW9uYWwgcGF0aCB0byB0aGUgY3VycmVudCB3b3JraW5nIGRpclxuICogQHJldHVybnMge05vZGVKUy5FcnJub0V4Y2VwdGlvbn0gTXV0YXRlZCBlcnJvciBpbnN0YW5jZSB3aXRoIGFuIGltcHJvdmVkIGRlc2NyaXB0aW9uIG9yIGFuXG4gKiB1bmNoYW5nZWQgZXJyb3IgaW5zdGFuY2VcbiAqL1xuZnVuY3Rpb24gZm9ybWF0RW5vZW50IChlcnJvciwgY21kLCBjd2QgPSBudWxsKSB7XG4gIHRyeSB7XG4gICAgd2hpY2guc3luYyhjbWQpO1xuICAgIGlmIChjd2QpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIGZzLmFjY2Vzc1N5bmMoY3dkLCBmcy5jb25zdGFudHMuUl9PSyk7XG4gICAgICB9IGNhdGNoIChpZ24pIHtcbiAgICAgICAgZXJyb3IubWVzc2FnZSA9IGBUaGUgY3VycmVudCB3b3JraW5nIGRpcmVjdG9yeSAnJHtjd2R9JyBmb3IgJyR7Y21kfScgY29tbWFuZCBgICtcbiAgICAgICAgICBgZWl0aGVyIGRvZXMgbm90IGV4aXN0IG9yIGlzIG5vdCBhY2Nlc3NpYmxlYDtcbiAgICAgIH1cbiAgICB9XG4gIH0gY2F0Y2ggKGlnbikge1xuICAgIGVycm9yLm1lc3NhZ2UgPSBgQ29tbWFuZCAnJHtjbWR9JyBub3QgZm91bmQuIElzIGl0IGluc3RhbGxlZD9gO1xuICB9XG4gIHJldHVybiBlcnJvcjtcbn1cblxuZXhwb3J0IHsgZm9ybWF0RW5vZW50IH07XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBYUEsU0FBU0EsWUFBVCxDQUF1QkMsS0FBdkIsRUFBOEJDLEdBQTlCLEVBQW1DQyxHQUFHLEdBQUcsSUFBekMsRUFBK0M7RUFDN0MsSUFBSTtJQUNGQyxjQUFBLENBQU1DLElBQU4sQ0FBV0gsR0FBWDs7SUFDQSxJQUFJQyxHQUFKLEVBQVM7TUFDUCxJQUFJO1FBQ0ZHLFdBQUEsQ0FBR0MsVUFBSCxDQUFjSixHQUFkLEVBQW1CRyxXQUFBLENBQUdFLFNBQUgsQ0FBYUMsSUFBaEM7TUFDRCxDQUZELENBRUUsT0FBT0MsR0FBUCxFQUFZO1FBQ1pULEtBQUssQ0FBQ1UsT0FBTixHQUFpQixrQ0FBaUNSLEdBQUksVUFBU0QsR0FBSSxZQUFuRCxHQUNiLDRDQURIO01BRUQ7SUFDRjtFQUNGLENBVkQsQ0FVRSxPQUFPUSxHQUFQLEVBQVk7SUFDWlQsS0FBSyxDQUFDVSxPQUFOLEdBQWlCLFlBQVdULEdBQUksK0JBQWhDO0VBQ0Q7O0VBQ0QsT0FBT0QsS0FBUDtBQUNEIn0=
@@ -0,0 +1,32 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.spawn = exports.exec = exports.SubProcess = void 0;
7
+
8
+ require("source-map-support/register");
9
+
10
+ var cp = _interopRequireWildcard(require("child_process"));
11
+
12
+ var spIndex = _interopRequireWildcard(require("./subprocess"));
13
+
14
+ var execIndex = _interopRequireWildcard(require("./exec"));
15
+
16
+ function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
17
+
18
+ function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
19
+
20
+ const {
21
+ spawn
22
+ } = cp;
23
+ exports.spawn = spawn;
24
+ const {
25
+ SubProcess
26
+ } = spIndex;
27
+ exports.SubProcess = SubProcess;
28
+ const {
29
+ exec
30
+ } = execIndex;
31
+ exports.exec = exec;
32
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJzcGF3biIsImNwIiwiU3ViUHJvY2VzcyIsInNwSW5kZXgiLCJleGVjIiwiZXhlY0luZGV4Il0sInNvdXJjZXMiOlsiLi4vLi4vbGliL2luZGV4LmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIHRyYW5zcGlsZTptYWluXG5pbXBvcnQgKiBhcyBjcCBmcm9tICdjaGlsZF9wcm9jZXNzJztcbmltcG9ydCAqIGFzIHNwSW5kZXggZnJvbSAnLi9zdWJwcm9jZXNzJztcbmltcG9ydCAqIGFzIGV4ZWNJbmRleCBmcm9tICcuL2V4ZWMnO1xuXG5cbmNvbnN0IHsgc3Bhd24gfSA9IGNwO1xuY29uc3QgeyBTdWJQcm9jZXNzIH0gPSBzcEluZGV4O1xuY29uc3QgeyBleGVjIH0gPSBleGVjSW5kZXg7XG5cbmV4cG9ydCB7IGV4ZWMsIHNwYXduLCBTdWJQcm9jZXNzIH07XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBOztBQUNBOztBQUNBOzs7Ozs7QUFHQSxNQUFNO0VBQUVBO0FBQUYsSUFBWUMsRUFBbEI7O0FBQ0EsTUFBTTtFQUFFQztBQUFGLElBQWlCQyxPQUF2Qjs7QUFDQSxNQUFNO0VBQUVDO0FBQUYsSUFBV0MsU0FBakIifQ==
@@ -19,6 +19,8 @@ var _shellQuote = require("shell-quote");
19
19
 
20
20
  var _lodash = _interopRequireDefault(require("lodash"));
21
21
 
22
+ var _helpers = require("./helpers");
23
+
22
24
  const {
23
25
  EventEmitter
24
26
  } = _events.default;
@@ -29,6 +31,14 @@ function cutSuffix(str, suffixLength) {
29
31
  }
30
32
 
31
33
  class SubProcess extends EventEmitter {
34
+ lastLinePortion;
35
+ proc;
36
+ args;
37
+ cmd;
38
+ opts;
39
+ expectingExit;
40
+ rep;
41
+
32
42
  constructor(cmd, args = [], opts = {}) {
33
43
  super();
34
44
  if (!cmd) throw new Error('Command is required');
@@ -40,6 +50,10 @@ class SubProcess extends EventEmitter {
40
50
  this.opts = opts;
41
51
  this.expectingExit = false;
42
52
  this.rep = (0, _shellQuote.quote)([cmd, ...args]);
53
+ this.lastLinePortion = {
54
+ stdout: '',
55
+ stderr: ''
56
+ };
43
57
  }
44
58
 
45
59
  get isRunning() {
@@ -107,7 +121,7 @@ class SubProcess extends EventEmitter {
107
121
  } = streams;
108
122
 
109
123
  try {
110
- if (startDetector && startDetector(stdout, stderr)) {
124
+ if (_lodash.default.isFunction(startDetector) && startDetector(stdout, stderr)) {
111
125
  startDetector = null;
112
126
  resolve();
113
127
  }
@@ -140,14 +154,20 @@ class SubProcess extends EventEmitter {
140
154
  };
141
155
 
142
156
  this.proc.on('error', err => {
143
- this.proc.removeAllListeners('exit');
144
- this.proc.kill('SIGINT');
157
+ var _this$proc, _this$proc2, _this$proc3;
158
+
159
+ (_this$proc = this.proc) === null || _this$proc === void 0 ? void 0 : _this$proc.removeAllListeners('exit');
160
+ (_this$proc2 = this.proc) === null || _this$proc2 === void 0 ? void 0 : _this$proc2.kill('SIGINT');
161
+
162
+ if (err.code === 'ENOENT') {
163
+ var _this$opts;
145
164
 
146
- if (err.errno === 'ENOENT') {
147
- err = new Error(`Command '${this.cmd}' not found. Is it installed?`);
165
+ err = (0, _helpers.formatEnoent)(err, this.cmd, (_this$opts = this.opts) === null || _this$opts === void 0 ? void 0 : _this$opts.cwd);
148
166
  }
149
167
 
150
168
  reject(err);
169
+ (_this$proc3 = this.proc) === null || _this$proc3 === void 0 ? void 0 : _this$proc3.unref();
170
+ this.proc = null;
151
171
  });
152
172
 
153
173
  if (this.proc.stdout) {
@@ -214,12 +234,14 @@ class SubProcess extends EventEmitter {
214
234
 
215
235
  this.handleLastLines();
216
236
  return await new _bluebird.default((resolve, reject) => {
217
- this.proc.on('close', resolve);
237
+ var _this$proc4, _this$proc5;
238
+
239
+ (_this$proc4 = this.proc) === null || _this$proc4 === void 0 ? void 0 : _this$proc4.on('close', resolve);
218
240
  this.expectingExit = true;
219
- this.proc.kill(signal);
241
+ (_this$proc5 = this.proc) === null || _this$proc5 === void 0 ? void 0 : _this$proc5.kill(signal);
220
242
  setTimeout(() => {
221
243
  reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));
222
- }, timeout);
244
+ }, timeout).unref();
223
245
  });
224
246
  }
225
247
 
@@ -229,8 +251,10 @@ class SubProcess extends EventEmitter {
229
251
  }
230
252
 
231
253
  return await new _bluebird.default((resolve, reject) => {
232
- this.proc.on('exit', code => {
233
- if (allowedExitCodes.indexOf(code) === -1) {
254
+ var _this$proc6;
255
+
256
+ (_this$proc6 = this.proc) === null || _this$proc6 === void 0 ? void 0 : _this$proc6.on('exit', code => {
257
+ if (code !== null && allowedExitCodes.indexOf(code) === -1) {
234
258
  reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));
235
259
  } else {
236
260
  resolve(code);
@@ -257,7 +281,5 @@ class SubProcess extends EventEmitter {
257
281
 
258
282
  exports.SubProcess = SubProcess;
259
283
  var _default = SubProcess;
260
- exports.default = _default;require('source-map-support').install();
261
-
262
-
263
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/subprocess.js"],"names":["EventEmitter","events","MAX_LINE_PORTION_LENGTH","cutSuffix","str","suffixLength","length","substr","SubProcess","constructor","cmd","args","opts","Error","_","isString","isArray","proc","expectingExit","rep","isRunning","emitLines","stream","lines","line","emit","toUpperCase","start","startDetector","timeoutMs","detach","startDelay","genericStartDetector","stdout","stderr","isNumber","isBoolean","detached","B","resolve","reject","setEncoding","encoding","lastLinePortion","handleOutput","streams","e","streamName","streamData","toPairs","split","map","x","last","resultLines","slice","currentPortion","on","err","removeAllListeners","kill","errno","chunk","toString","code","signal","handleLastLines","event","setTimeout","finally","unref","lastLines","stop","timeout","join","allowedExitCodes","indexOf","detachProcess","pid"],"mappings":"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","sourcesContent":["/* eslint-disable promise/prefer-await-to-callbacks */\n\nimport { spawn } from 'child_process';\nimport events from 'events';\nconst { EventEmitter } = events;\nimport B from 'bluebird';\nimport { quote } from 'shell-quote';\nimport _ from 'lodash';\n\n// This is needed to avoid memory leaks\n// when the process output is too long and contains\n// no line breaks\nconst MAX_LINE_PORTION_LENGTH = 0xFFFF;\n\nfunction cutSuffix (str, suffixLength) {\n  return str.length > suffixLength\n    // https://bugs.chromium.org/p/v8/issues/detail?id=2869\n    ? ` ${str.substr(str.length - suffixLength)}`.substr(1)\n    : str;\n}\n\n\nclass SubProcess extends EventEmitter {\n  constructor (cmd, args = [], opts = {}) {\n    super();\n    if (!cmd) throw new Error('Command is required'); // eslint-disable-line curly\n    if (!_.isString(cmd)) throw new Error('Command must be a string'); // eslint-disable-line curly\n    if (!_.isArray(args)) throw new Error('Args must be an array'); // eslint-disable-line curly\n\n    this.cmd = cmd;\n    this.args = args;\n    this.proc = null;\n    this.opts = opts;\n    this.expectingExit = false;\n\n    // get a quoted representation of the command for error strings\n    this.rep = quote([cmd, ...args]);\n  }\n\n  get isRunning () {\n    // presence of `proc` means we have connected and started\n    return !!this.proc;\n  }\n\n  emitLines (stream, lines) {\n    for (let line of lines) {\n      this.emit('stream-line', `[${stream.toUpperCase()}] ${line}`);\n    }\n  }\n\n  // spawn the subprocess and return control whenever we deem that it has fully\n  // \"started\"\n  async start (startDetector = null, timeoutMs = null, detach = false) {\n    let startDelay = 10;\n\n    const genericStartDetector = function genericStartDetector (stdout, stderr) {\n      return stdout || stderr;\n    };\n\n    // the default start detector simply returns true when we get any output\n    if (startDetector === null) {\n      startDetector = genericStartDetector;\n    }\n\n    // if the user passes a number, then we simply delay a certain amount of\n    // time before returning control, rather than waiting for a condition\n    if (_.isNumber(startDetector)) {\n      startDelay = startDetector;\n      startDetector = null;\n    }\n\n    // if the user passes in a boolean as one of the arguments, use it for `detach`\n    if (_.isBoolean(startDetector) && startDetector) {\n      if (!this.opts.detached) {\n        throw new Error(`Unable to detach process that is not started with 'detached' option`);\n      }\n      detach = true;\n      startDetector = genericStartDetector;\n    } else if (_.isBoolean(timeoutMs) && timeoutMs) {\n      if (!this.opts.detached) {\n        throw new Error(`Unable to detach process that is not started with 'detached' option`);\n      }\n      detach = true;\n      timeoutMs = null;\n    }\n\n    // return a promise so we can wrap the async behavior\n    return await new B((resolve, reject) => {\n      // actually spawn the subproc\n      this.proc = spawn(this.cmd, this.args, this.opts);\n\n      if (this.proc.stdout) {\n        this.proc.stdout.setEncoding(this.opts.encoding || 'utf8');\n      }\n      if (this.proc.stderr) {\n        this.proc.stderr.setEncoding(this.opts.encoding || 'utf8');\n      }\n      this.lastLinePortion = {stdout: '', stderr: ''};\n\n      // this function handles output that we collect from the subproc\n      const handleOutput = (streams) => {\n        const {stdout, stderr} = streams;\n        // if we have a startDetector, run it on the output so we can resolve/\n        // reject and move on from start\n        try {\n          if (startDetector && startDetector(stdout, stderr)) {\n            startDetector = null;\n            resolve();\n          }\n        } catch (e) {\n          reject(e);\n        }\n\n        // emit the actual output for whomever's listening\n        this.emit('output', stdout, stderr);\n\n        // we also want to emit lines, but it's more complex since output\n        // comes in chunks and a line could come in two different chunks, so\n        // we have logic to handle that case (using this.lastLinePortion to\n        // remember a line that started but did not finish in the last chunk)\n        for (const [streamName, streamData] of _.toPairs(streams)) {\n          if (!streamData) continue; // eslint-disable-line curly\n          const lines = streamData.split('\\n')\n            // https://bugs.chromium.org/p/v8/issues/detail?id=2869\n            .map((x) => ` ${x}`.substr(1));\n          if (lines.length > 1) {\n            lines[0] = this.lastLinePortion[streamName] + lines[0];\n            this.lastLinePortion[streamName] = cutSuffix(_.last(lines), MAX_LINE_PORTION_LENGTH);\n            const resultLines = lines.slice(0, -1);\n            this.emit(`lines-${streamName}`, resultLines);\n            this.emitLines(streamName, resultLines);\n          } else {\n            const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);\n            if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {\n              this.lastLinePortion[streamName] = currentPortion;\n            } else {\n              this.lastLinePortion[streamName] += currentPortion;\n            }\n          }\n        }\n      };\n\n      // if we get an error spawning the proc, reject and clean up the proc\n      this.proc.on('error', err => {\n        this.proc.removeAllListeners('exit');\n        this.proc.kill('SIGINT');\n\n        if (err.errno === 'ENOENT') {\n          err = new Error(`Command '${this.cmd}' not found. Is it installed?`);\n        }\n        reject(err);\n      });\n\n      if (this.proc.stdout) {\n        this.proc.stdout.on('data', (chunk) => handleOutput({stdout: chunk.toString(), stderr: ''}));\n      }\n\n      if (this.proc.stderr) {\n        this.proc.stderr.on('data', (chunk) => handleOutput({stdout: '', stderr: chunk.toString()}));\n      }\n\n      // when the proc exits, we might still have a buffer of lines we were\n      // waiting on more chunks to complete. Go ahead and emit those, then\n      // re-emit the exit so a listener can handle the possibly-unexpected exit\n      this.proc.on('exit', (code, signal) => {\n        this.handleLastLines();\n\n        this.emit('exit', code, signal);\n\n        // in addition to the bare exit event, also emit one of three other\n        // events that contain more helpful information:\n        // 'stop': we stopped this\n        // 'die': the process ended out of our control with a non-zero exit\n        // 'end': the process ended out of our control with a zero exit\n        let event = this.expectingExit ? 'stop' : 'die';\n        if (!this.expectingExit && code === 0) {\n          event = 'end';\n        }\n        this.emit(event, code, signal);\n\n        // finally clean up the proc and make sure to reset our exit\n        // expectations\n        this.proc = null;\n        this.expectingExit = false;\n      });\n\n      // if the user hasn't given us a startDetector, instead just resolve\n      // when startDelay ms have passed\n      if (!startDetector) {\n        setTimeout(() => { resolve(); }, startDelay);\n      }\n\n      // if the user has given us a timeout, start the clock for rejecting\n      // the promise if we take too long to start\n      if (_.isNumber(timeoutMs)) {\n        setTimeout(() => {\n          reject(new Error(`The process did not start within ${timeoutMs}ms ` +\n            `(cmd: '${this.rep}')`));\n        }, timeoutMs);\n      }\n    }).finally(() => {\n      if (detach && this.proc) {\n        this.proc.unref();\n      }\n    });\n  }\n\n  handleLastLines () {\n    for (let stream of ['stdout', 'stderr']) {\n      if (this.lastLinePortion[stream]) {\n        const lastLines = [this.lastLinePortion[stream]];\n        this.emit(`lines-${stream}`, lastLines);\n        this.emitLines(stream, lastLines);\n        this.lastLinePortion[stream] = '';\n      }\n    }\n  }\n\n  async stop (signal = 'SIGTERM', timeout = 10000) {\n    if (!this.isRunning) {\n      throw new Error(`Can't stop process; it's not currently running (cmd: '${this.rep}')`);\n    }\n    // make sure to emit any data in our lines buffer whenever we're done with\n    // the proc\n    this.handleLastLines();\n    return await new B((resolve, reject) => {\n      this.proc.on('close', resolve);\n      this.expectingExit = true;\n      this.proc.kill(signal);\n      setTimeout(() => {\n        reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));\n      }, timeout);\n    });\n  }\n\n  async join (allowedExitCodes = [0]) {\n    if (!this.isRunning) {\n      throw new Error(`Cannot join process; it is not currently running (cmd: '${this.rep}')`);\n    }\n\n    return await new B((resolve, reject) => {\n      this.proc.on('exit', (code) => {\n        if (allowedExitCodes.indexOf(code) === -1) {\n          reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));\n        } else {\n          resolve(code);\n        }\n      });\n    });\n  }\n\n  /*\n   * This will only work if the process is created with the `detached` option\n   */\n  detachProcess () {\n    if (!this.opts.detached) {\n      // this means that there is a misconfiguration in the calling code\n      throw new Error(`Unable to detach process that is not started with 'detached' option`);\n    }\n    if (this.proc) {\n      this.proc.unref();\n    }\n  }\n\n  get pid () {\n    return this.proc ? this.proc.pid : null;\n  }\n}\n\nexport { SubProcess };\nexport default SubProcess;\n"],"file":"lib/subprocess.js","sourceRoot":"../.."}
284
+ exports.default = _default;
285
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["EventEmitter","events","MAX_LINE_PORTION_LENGTH","cutSuffix","str","suffixLength","length","substr","SubProcess","lastLinePortion","proc","args","cmd","opts","expectingExit","rep","constructor","Error","_","isString","isArray","quote","stdout","stderr","isRunning","emitLines","stream","lines","line","emit","toUpperCase","start","startDetector","timeoutMs","detach","startDelay","genericStartDetector","isNumber","isBoolean","detached","B","resolve","reject","spawn","setEncoding","encoding","handleOutput","streams","isFunction","e","streamName","streamData","toPairs","split","map","x","last","resultLines","slice","currentPortion","on","err","removeAllListeners","kill","code","formatEnoent","cwd","unref","chunk","toString","signal","handleLastLines","event","setTimeout","finally","lastLines","stop","timeout","join","allowedExitCodes","indexOf","detachProcess","pid"],"sources":["../../lib/subprocess.js"],"sourcesContent":["/* eslint-disable promise/prefer-await-to-callbacks */\n\nimport { spawn } from 'child_process';\nimport events from 'events';\nconst { EventEmitter } = events;\nimport B from 'bluebird';\nimport { quote } from 'shell-quote';\nimport _ from 'lodash';\nimport { formatEnoent } from './helpers';\n\n\n// This is needed to avoid memory leaks\n// when the process output is too long and contains\n// no line breaks\nconst MAX_LINE_PORTION_LENGTH = 0xFFFF;\n\nfunction cutSuffix (str, suffixLength) {\n  return str.length > suffixLength\n    // https://bugs.chromium.org/p/v8/issues/detail?id=2869\n    ? ` ${str.substr(str.length - suffixLength)}`.substr(1)\n    : str;\n}\n\n\nclass SubProcess extends EventEmitter {\n\n  /**\n   * @type { {stdout: string, stderr: string} }\n   */\n  lastLinePortion;\n\n  /** @type {import('child_process').ChildProcess?} */\n  proc;\n\n  /** @type {string[]} */\n  args;\n\n  /**\n   * @type {string}\n   */\n  cmd;\n\n  /**\n   * @type {any}\n  */\n  opts;\n\n  /**\n   * @type {boolean}\n   */\n  expectingExit;\n\n  /**\n   * @type {string}\n   */\n  rep;\n\n  /**\n   * @param {string} cmd\n   * @param {string[]} [args]\n   * @param {any} [opts]\n   */\n  constructor (cmd, args = [], opts = {}) {\n    super();\n    if (!cmd) throw new Error('Command is required'); // eslint-disable-line curly\n    if (!_.isString(cmd)) throw new Error('Command must be a string'); // eslint-disable-line curly\n    if (!_.isArray(args)) throw new Error('Args must be an array'); // eslint-disable-line curly\n\n    this.cmd = cmd;\n    this.args = args;\n    this.proc = null;\n    this.opts = opts;\n    this.expectingExit = false;\n\n    // get a quoted representation of the command for error strings\n    this.rep = quote([cmd, ...args]);\n\n    this.lastLinePortion = {stdout: '', stderr: ''};\n  }\n\n  get isRunning () {\n    // presence of `proc` means we have connected and started\n    return !!this.proc;\n  }\n\n  /**\n   *\n   * @param {string} stream\n   * @param {Iterable<string>} lines\n   */\n  emitLines (stream, lines) {\n    for (let line of lines) {\n      this.emit('stream-line', `[${stream.toUpperCase()}] ${line}`);\n    }\n  }\n\n  // spawn the subprocess and return control whenever we deem that it has fully\n  // \"started\"\n  /**\n   *\n   * @param {StartDetector|number?} startDetector\n   * @param {number?} timeoutMs\n   * @param {boolean} detach\n   * @returns {Promise<void>}\n   */\n  async start (startDetector = null, timeoutMs = null, detach = false) {\n    let startDelay = 10;\n\n    const genericStartDetector = /** @type {StartDetector} */(function genericStartDetector (stdout, stderr) {\n      return stdout || stderr;\n    });\n\n    // the default start detector simply returns true when we get any output\n    if (startDetector === null) {\n      startDetector = genericStartDetector;\n    }\n\n    // if the user passes a number, then we simply delay a certain amount of\n    // time before returning control, rather than waiting for a condition\n    if (_.isNumber(startDetector)) {\n      startDelay = startDetector;\n      startDetector = null;\n    }\n\n    // if the user passes in a boolean as one of the arguments, use it for `detach`\n    if (_.isBoolean(startDetector) && startDetector) {\n      if (!this.opts.detached) {\n        throw new Error(`Unable to detach process that is not started with 'detached' option`);\n      }\n      detach = true;\n      startDetector = genericStartDetector;\n    } else if (_.isBoolean(timeoutMs) && timeoutMs) {\n      if (!this.opts.detached) {\n        throw new Error(`Unable to detach process that is not started with 'detached' option`);\n      }\n      detach = true;\n      timeoutMs = null;\n    }\n\n    // return a promise so we can wrap the async behavior\n    return await new B((resolve, reject) => {\n      // actually spawn the subproc\n      this.proc = spawn(this.cmd, this.args, this.opts);\n\n      if (this.proc.stdout) {\n        this.proc.stdout.setEncoding(this.opts.encoding || 'utf8');\n      }\n      if (this.proc.stderr) {\n        this.proc.stderr.setEncoding(this.opts.encoding || 'utf8');\n      }\n      this.lastLinePortion = {stdout: '', stderr: ''};\n\n      // this function handles output that we collect from the subproc\n      /**\n       *\n       * @param { {stdout: string, stderr: string} } streams\n       */\n      const handleOutput = (streams) => {\n        const {stdout, stderr} = streams;\n        // if we have a startDetector, run it on the output so we can resolve/\n        // reject and move on from start\n        try {\n          if (_.isFunction(startDetector) && startDetector(stdout, stderr)) {\n            startDetector = null;\n            resolve();\n          }\n        } catch (e) {\n          reject(e);\n        }\n\n        // emit the actual output for whomever's listening\n        this.emit('output', stdout, stderr);\n\n        // we also want to emit lines, but it's more complex since output\n        // comes in chunks and a line could come in two different chunks, so\n        // we have logic to handle that case (using this.lastLinePortion to\n        // remember a line that started but did not finish in the last chunk)\n        for (const [streamName, streamData] of /** @type {[['stdout', string], ['stderr', string]]} */(_.toPairs(streams))) {\n          if (!streamData) continue; // eslint-disable-line curly\n          const lines = streamData.split('\\n')\n            // https://bugs.chromium.org/p/v8/issues/detail?id=2869\n            .map((x) => ` ${x}`.substr(1));\n          if (lines.length > 1) {\n            lines[0] = this.lastLinePortion[streamName] + lines[0];\n            this.lastLinePortion[streamName] = cutSuffix(_.last(lines), MAX_LINE_PORTION_LENGTH);\n            const resultLines = lines.slice(0, -1);\n            this.emit(`lines-${streamName}`, resultLines);\n            this.emitLines(streamName, resultLines);\n          } else {\n            const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);\n            if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {\n              this.lastLinePortion[streamName] = currentPortion;\n            } else {\n              this.lastLinePortion[streamName] += currentPortion;\n            }\n          }\n        }\n      };\n\n      // if we get an error spawning the proc, reject and clean up the proc\n      this.proc.on('error', /** @param {NodeJS.ErrnoException} err */ (err) => {\n        this.proc?.removeAllListeners('exit');\n        this.proc?.kill('SIGINT');\n\n        if (err.code === 'ENOENT') {\n          err = formatEnoent(err, this.cmd, this.opts?.cwd);\n        }\n        reject(err);\n\n        this.proc?.unref();\n        this.proc = null;\n      });\n\n      if (this.proc.stdout) {\n        this.proc.stdout.on('data', (chunk) => handleOutput({stdout: chunk.toString(), stderr: ''}));\n      }\n\n      if (this.proc.stderr) {\n        this.proc.stderr.on('data', (chunk) => handleOutput({stdout: '', stderr: chunk.toString()}));\n      }\n\n      // when the proc exits, we might still have a buffer of lines we were\n      // waiting on more chunks to complete. Go ahead and emit those, then\n      // re-emit the exit so a listener can handle the possibly-unexpected exit\n      this.proc.on('exit', (code, signal) => {\n        this.handleLastLines();\n\n        this.emit('exit', code, signal);\n\n        // in addition to the bare exit event, also emit one of three other\n        // events that contain more helpful information:\n        // 'stop': we stopped this\n        // 'die': the process ended out of our control with a non-zero exit\n        // 'end': the process ended out of our control with a zero exit\n        let event = this.expectingExit ? 'stop' : 'die';\n        if (!this.expectingExit && code === 0) {\n          event = 'end';\n        }\n        this.emit(event, code, signal);\n\n        // finally clean up the proc and make sure to reset our exit\n        // expectations\n        this.proc = null;\n        this.expectingExit = false;\n      });\n\n      // if the user hasn't given us a startDetector, instead just resolve\n      // when startDelay ms have passed\n      if (!startDetector) {\n        setTimeout(() => { resolve(); }, startDelay);\n      }\n\n      // if the user has given us a timeout, start the clock for rejecting\n      // the promise if we take too long to start\n      if (_.isNumber(timeoutMs)) {\n        setTimeout(() => {\n          reject(new Error(`The process did not start within ${timeoutMs}ms ` +\n            `(cmd: '${this.rep}')`));\n        }, timeoutMs);\n      }\n    }).finally(() => {\n      if (detach && this.proc) {\n        this.proc.unref();\n      }\n    });\n  }\n\n  handleLastLines () {\n    for (let stream of ['stdout', 'stderr']) {\n      if (this.lastLinePortion[stream]) {\n        const lastLines = [this.lastLinePortion[stream]];\n        this.emit(`lines-${stream}`, lastLines);\n        this.emitLines(stream, lastLines);\n        this.lastLinePortion[stream] = '';\n      }\n    }\n  }\n\n  /**\n   *\n   * @param {NodeJS.Signals} signal\n   * @param {number} timeout\n   * @returns {Promise<void>}\n   */\n  async stop (signal = 'SIGTERM', timeout = 10000) {\n    if (!this.isRunning) {\n      throw new Error(`Can't stop process; it's not currently running (cmd: '${this.rep}')`);\n    }\n    // make sure to emit any data in our lines buffer whenever we're done with\n    // the proc\n    this.handleLastLines();\n    return await new B((resolve, reject) => {\n      this.proc?.on('close', resolve);\n      this.expectingExit = true;\n      this.proc?.kill(signal);\n      // this timeout needs unref() or node will wait for the timeout to fire before\n      // exiting the process.\n      setTimeout(() => {\n        reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));\n      }, timeout).unref();\n    });\n  }\n\n  async join (allowedExitCodes = [0]) {\n    if (!this.isRunning) {\n      throw new Error(`Cannot join process; it is not currently running (cmd: '${this.rep}')`);\n    }\n\n    return await new B((resolve, reject) => {\n      this.proc?.on('exit', (code) => {\n        if (code !== null && allowedExitCodes.indexOf(code) === -1) {\n          reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));\n        } else {\n          resolve(code);\n        }\n      });\n    });\n  }\n\n  /*\n   * This will only work if the process is created with the `detached` option\n   */\n  detachProcess () {\n    if (!this.opts.detached) {\n      // this means that there is a misconfiguration in the calling code\n      throw new Error(`Unable to detach process that is not started with 'detached' option`);\n    }\n    if (this.proc) {\n      this.proc.unref();\n    }\n  }\n\n  get pid () {\n    return this.proc ? this.proc.pid : null;\n  }\n}\n\nexport { SubProcess };\nexport default SubProcess;\n\n/**\n * @callback StartDetector\n * @param {string} stdout\n * @param {string} [stderr]\n * @returns {any}\n */\n"],"mappings":"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"}
package/index.js CHANGED
@@ -1,11 +1 @@
1
- // transpile:main
2
- import * as cp from 'child_process';
3
- import * as spIndex from './lib/subprocess';
4
- import * as execIndex from './lib/exec';
5
-
6
-
7
- const { spawn } = cp;
8
- const { SubProcess } = spIndex;
9
- const { exec } = execIndex;
10
-
11
- export { exec, spawn, SubProcess };
1
+ module.exports = require('./build/lib/index.js');
package/lib/exec.js CHANGED
@@ -4,16 +4,25 @@ import { spawn } from 'child_process';
4
4
  import { quote } from 'shell-quote';
5
5
  import B from 'bluebird';
6
6
  import _ from 'lodash';
7
+ import { formatEnoent } from './helpers';
7
8
 
8
9
  const MAX_BUFFER_SIZE = 100 * 1024 * 1024;
9
10
 
10
- async function exec (cmd, args = [], opts = {}) {
11
+ /**
12
+ * Spawns a process
13
+ * @template {TeenProcessExecOptions} T
14
+ * @param {string} cmd - Program to execute
15
+ * @param {string[]} [args] - Arguments to pass to the program
16
+ * @param {T} [opts] - Options
17
+ * @returns {Promise<BufferProp<T> extends true ? TeenProcessExecBufferResult : TeenProcessExecStringResult>}
18
+ */
19
+ async function exec (cmd, args = [], opts = /** @type {T} */({})) {
11
20
  // get a quoted representation of the command for error strings
12
21
  const rep = quote([cmd, ...args]);
13
22
 
14
23
  // extend default options; we're basically re-implementing exec's options
15
24
  // for use here with spawn under the hood
16
- opts = Object.assign({
25
+ opts = /** @type {T} */(_.defaults(opts, {
17
26
  timeout: null,
18
27
  encoding: 'utf8',
19
28
  killSignal: 'SIGTERM',
@@ -26,7 +35,9 @@ async function exec (cmd, args = [], opts = {}) {
26
35
  logger: undefined,
27
36
  maxStdoutBufferSize: MAX_BUFFER_SIZE,
28
37
  maxStderrBufferSize: MAX_BUFFER_SIZE,
29
- }, opts);
38
+ }));
39
+
40
+ const isBuffer = Boolean(opts.isBuffer);
30
41
 
31
42
  // this is an async function, so return a promise
32
43
  return await new B((resolve, reject) => {
@@ -36,15 +47,15 @@ async function exec (cmd, args = [], opts = {}) {
36
47
  let stdoutArr = [], stderrArr = [], timer = null;
37
48
 
38
49
  // if the process errors out, reject the promise
39
- proc.on('error', (err) => {
40
- let msg = `Command '${rep}' errored out: ${err.stack}`;
50
+ proc.on('error', /** @param {NodeJS.ErrnoException} err */(err) => {
51
+ // @ts-ignore
41
52
  if (err.errno === 'ENOENT') {
42
- msg = `Command '${cmd}' not found. Is it installed?`;
53
+ err = formatEnoent(err, cmd, opts.cwd);
43
54
  }
44
- reject(new Error(msg));
55
+ reject(err);
45
56
  });
46
57
  if (proc.stdin) {
47
- proc.stdin.on('error', (err) => {
58
+ proc.stdin.on('error', /** @param {NodeJS.ErrnoException} err */(err) => {
48
59
  reject(new Error(`Standard input '${err.syscall}' error: ${err.stack}`));
49
60
  });
50
61
  }
@@ -87,6 +98,11 @@ async function exec (cmd, args = [], opts = {}) {
87
98
  chunks: stderrArr,
88
99
  });
89
100
 
101
+ /**
102
+ * @template {boolean} U
103
+ * @param {U} isBuffer
104
+ * @returns {U extends true ? {stdout: Buffer, stderr: Buffer} : {stdout: string, stderr: string}}
105
+ */
90
106
  function getStdio (isBuffer) {
91
107
  let stdout, stderr;
92
108
  if (isBuffer) {
@@ -96,7 +112,7 @@ async function exec (cmd, args = [], opts = {}) {
96
112
  stdout = Buffer.concat(stdoutArr).toString(opts.encoding);
97
113
  stderr = Buffer.concat(stderrArr).toString(opts.encoding);
98
114
  }
99
- return {stdout, stderr};
115
+ return /** @type {U extends true ? {stdout: Buffer, stderr: Buffer} : {stdout: string, stderr: string}} */({stdout, stderr});
100
116
  }
101
117
 
102
118
  // if the process ends, either resolve or reject the promise based on the
@@ -106,9 +122,9 @@ async function exec (cmd, args = [], opts = {}) {
106
122
  if (timer) {
107
123
  clearTimeout(timer);
108
124
  }
109
- let {stdout, stderr} = getStdio(opts.isBuffer);
125
+ let {stdout, stderr} = getStdio(isBuffer);
110
126
  if (code === 0) {
111
- resolve({stdout, stderr, code});
127
+ resolve(/** @type {BufferProp<T> extends true ? TeenProcessExecBufferResult : TeenProcessExecStringResult} */({stdout, stderr, code}));
112
128
  } else {
113
129
  let err = new Error(`Command '${rep}' exited with code ${code}`);
114
130
  err = Object.assign(err, {stdout, stderr, code});
@@ -121,7 +137,7 @@ async function exec (cmd, args = [], opts = {}) {
121
137
  // have in case it's helpful in debugging
122
138
  if (opts.timeout) {
123
139
  timer = setTimeout(() => {
124
- let {stdout, stderr} = getStdio(opts.isBuffer);
140
+ let {stdout, stderr} = getStdio(isBuffer);
125
141
  let err = new Error(`Command '${rep}' timed out after ${opts.timeout}ms`);
126
142
  err = Object.assign(err, {stdout, stderr, code: null});
127
143
  reject(err);
@@ -135,3 +151,60 @@ async function exec (cmd, args = [], opts = {}) {
135
151
 
136
152
  export { exec };
137
153
  export default exec;
154
+
155
+ /**
156
+ * Options on top of `SpawnOptions`, unique to `teen_process.`
157
+ * @typedef {Object} TeenProcessProps
158
+ * @property {boolean} [ignoreOutput] - Ignore & discard all output
159
+ * @property {boolean} [isBuffer] - Return output as a Buffer
160
+ * @property {TeenProcessLogger} [logger] - Logger to use for debugging
161
+ * @property {number} [maxStdoutBufferSize] - Maximum size of `stdout` buffer
162
+ * @property {number} [maxStderrBufferSize] - Maximum size of `stderr` buffer
163
+ * @property {BufferEncoding} [encoding='utf8'] - Encoding to use for output
164
+ */
165
+
166
+ /**
167
+ * A logger object understood by {@link exec teen_process.exec}.
168
+ * @typedef {Object} TeenProcessLogger
169
+ * @property {(...args: any[]) => void} debug
170
+ */
171
+
172
+ /**
173
+ * Options for {@link exec teen_process.exec}.
174
+ * @typedef {import('child_process').SpawnOptions & TeenProcessProps} TeenProcessExecOptions
175
+ */
176
+
177
+ /**
178
+ * The value {@link exec teen_process.exec} resolves to when `isBuffer` is `false`
179
+ * @typedef {Object} TeenProcessExecStringResult
180
+ * @property {string} stdout - Stdout
181
+ * @property {string} stderr - Stderr
182
+ * @property {number?} code - Exit code
183
+ */
184
+
185
+ /**
186
+ * The value {@link exec teen_process.exec} resolves to when `isBuffer` is `true`
187
+ * @typedef {Object} TeenProcessExecBufferResult
188
+ * @property {Buffer} stdout - Stdout
189
+ * @property {Buffer} stderr - Stderr
190
+ * @property {number?} code - Exit code
191
+ */
192
+
193
+ /**
194
+ * Extra props {@link exec teen_process.exec} adds to its error objects
195
+ * @typedef {Object} TeenProcessExecErrorProps
196
+ * @property {string} stdout - STDOUT
197
+ * @property {string} stderr - STDERR
198
+ * @property {number?} code - Exit code
199
+ */
200
+
201
+ /**
202
+ * Error thrown by {@link exec teen_process.exec}
203
+ * @typedef {Error & TeenProcessExecErrorProps} TeenProcessExecError
204
+ */
205
+
206
+ /**
207
+ * @template {{isBuffer?: boolean}} MaybeBuffer
208
+ * @typedef {MaybeBuffer['isBuffer']} BufferProp
209
+ * @private
210
+ */
package/lib/helpers.js ADDED
@@ -0,0 +1,32 @@
1
+ import which from 'which';
2
+ import fs from 'fs';
3
+
4
+ /**
5
+ * Decorates ENOENT error received from a spawn system call
6
+ * with a more descriptive message, so it could be properly handled by a user.
7
+ *
8
+ * @param {NodeJS.ErrnoException} error Original error instance. !!! The instance is mutated after
9
+ * this helper function invocation
10
+ * @param {string} cmd Original command to execute
11
+ * @param {string|URL?} [cwd] Optional path to the current working dir
12
+ * @returns {NodeJS.ErrnoException} Mutated error instance with an improved description or an
13
+ * unchanged error instance
14
+ */
15
+ function formatEnoent (error, cmd, cwd = null) {
16
+ try {
17
+ which.sync(cmd);
18
+ if (cwd) {
19
+ try {
20
+ fs.accessSync(cwd, fs.constants.R_OK);
21
+ } catch (ign) {
22
+ error.message = `The current working directory '${cwd}' for '${cmd}' command ` +
23
+ `either does not exist or is not accessible`;
24
+ }
25
+ }
26
+ } catch (ign) {
27
+ error.message = `Command '${cmd}' not found. Is it installed?`;
28
+ }
29
+ return error;
30
+ }
31
+
32
+ export { formatEnoent };
package/lib/index.js ADDED
@@ -0,0 +1,11 @@
1
+ // transpile:main
2
+ import * as cp from 'child_process';
3
+ import * as spIndex from './subprocess';
4
+ import * as execIndex from './exec';
5
+
6
+
7
+ const { spawn } = cp;
8
+ const { SubProcess } = spIndex;
9
+ const { exec } = execIndex;
10
+
11
+ export { exec, spawn, SubProcess };
package/lib/subprocess.js CHANGED
@@ -6,6 +6,8 @@ const { EventEmitter } = events;
6
6
  import B from 'bluebird';
7
7
  import { quote } from 'shell-quote';
8
8
  import _ from 'lodash';
9
+ import { formatEnoent } from './helpers';
10
+
9
11
 
10
12
  // This is needed to avoid memory leaks
11
13
  // when the process output is too long and contains
@@ -21,6 +23,43 @@ function cutSuffix (str, suffixLength) {
21
23
 
22
24
 
23
25
  class SubProcess extends EventEmitter {
26
+
27
+ /**
28
+ * @type { {stdout: string, stderr: string} }
29
+ */
30
+ lastLinePortion;
31
+
32
+ /** @type {import('child_process').ChildProcess?} */
33
+ proc;
34
+
35
+ /** @type {string[]} */
36
+ args;
37
+
38
+ /**
39
+ * @type {string}
40
+ */
41
+ cmd;
42
+
43
+ /**
44
+ * @type {any}
45
+ */
46
+ opts;
47
+
48
+ /**
49
+ * @type {boolean}
50
+ */
51
+ expectingExit;
52
+
53
+ /**
54
+ * @type {string}
55
+ */
56
+ rep;
57
+
58
+ /**
59
+ * @param {string} cmd
60
+ * @param {string[]} [args]
61
+ * @param {any} [opts]
62
+ */
24
63
  constructor (cmd, args = [], opts = {}) {
25
64
  super();
26
65
  if (!cmd) throw new Error('Command is required'); // eslint-disable-line curly
@@ -35,6 +74,8 @@ class SubProcess extends EventEmitter {
35
74
 
36
75
  // get a quoted representation of the command for error strings
37
76
  this.rep = quote([cmd, ...args]);
77
+
78
+ this.lastLinePortion = {stdout: '', stderr: ''};
38
79
  }
39
80
 
40
81
  get isRunning () {
@@ -42,6 +83,11 @@ class SubProcess extends EventEmitter {
42
83
  return !!this.proc;
43
84
  }
44
85
 
86
+ /**
87
+ *
88
+ * @param {string} stream
89
+ * @param {Iterable<string>} lines
90
+ */
45
91
  emitLines (stream, lines) {
46
92
  for (let line of lines) {
47
93
  this.emit('stream-line', `[${stream.toUpperCase()}] ${line}`);
@@ -50,12 +96,19 @@ class SubProcess extends EventEmitter {
50
96
 
51
97
  // spawn the subprocess and return control whenever we deem that it has fully
52
98
  // "started"
99
+ /**
100
+ *
101
+ * @param {StartDetector|number?} startDetector
102
+ * @param {number?} timeoutMs
103
+ * @param {boolean} detach
104
+ * @returns {Promise<void>}
105
+ */
53
106
  async start (startDetector = null, timeoutMs = null, detach = false) {
54
107
  let startDelay = 10;
55
108
 
56
- const genericStartDetector = function genericStartDetector (stdout, stderr) {
109
+ const genericStartDetector = /** @type {StartDetector} */(function genericStartDetector (stdout, stderr) {
57
110
  return stdout || stderr;
58
- };
111
+ });
59
112
 
60
113
  // the default start detector simply returns true when we get any output
61
114
  if (startDetector === null) {
@@ -98,12 +151,16 @@ class SubProcess extends EventEmitter {
98
151
  this.lastLinePortion = {stdout: '', stderr: ''};
99
152
 
100
153
  // this function handles output that we collect from the subproc
154
+ /**
155
+ *
156
+ * @param { {stdout: string, stderr: string} } streams
157
+ */
101
158
  const handleOutput = (streams) => {
102
159
  const {stdout, stderr} = streams;
103
160
  // if we have a startDetector, run it on the output so we can resolve/
104
161
  // reject and move on from start
105
162
  try {
106
- if (startDetector && startDetector(stdout, stderr)) {
163
+ if (_.isFunction(startDetector) && startDetector(stdout, stderr)) {
107
164
  startDetector = null;
108
165
  resolve();
109
166
  }
@@ -118,7 +175,7 @@ class SubProcess extends EventEmitter {
118
175
  // comes in chunks and a line could come in two different chunks, so
119
176
  // we have logic to handle that case (using this.lastLinePortion to
120
177
  // remember a line that started but did not finish in the last chunk)
121
- for (const [streamName, streamData] of _.toPairs(streams)) {
178
+ for (const [streamName, streamData] of /** @type {[['stdout', string], ['stderr', string]]} */(_.toPairs(streams))) {
122
179
  if (!streamData) continue; // eslint-disable-line curly
123
180
  const lines = streamData.split('\n')
124
181
  // https://bugs.chromium.org/p/v8/issues/detail?id=2869
@@ -141,14 +198,17 @@ class SubProcess extends EventEmitter {
141
198
  };
142
199
 
143
200
  // if we get an error spawning the proc, reject and clean up the proc
144
- this.proc.on('error', err => {
145
- this.proc.removeAllListeners('exit');
146
- this.proc.kill('SIGINT');
201
+ this.proc.on('error', /** @param {NodeJS.ErrnoException} err */ (err) => {
202
+ this.proc?.removeAllListeners('exit');
203
+ this.proc?.kill('SIGINT');
147
204
 
148
- if (err.errno === 'ENOENT') {
149
- err = new Error(`Command '${this.cmd}' not found. Is it installed?`);
205
+ if (err.code === 'ENOENT') {
206
+ err = formatEnoent(err, this.cmd, this.opts?.cwd);
150
207
  }
151
208
  reject(err);
209
+
210
+ this.proc?.unref();
211
+ this.proc = null;
152
212
  });
153
213
 
154
214
  if (this.proc.stdout) {
@@ -216,6 +276,12 @@ class SubProcess extends EventEmitter {
216
276
  }
217
277
  }
218
278
 
279
+ /**
280
+ *
281
+ * @param {NodeJS.Signals} signal
282
+ * @param {number} timeout
283
+ * @returns {Promise<void>}
284
+ */
219
285
  async stop (signal = 'SIGTERM', timeout = 10000) {
220
286
  if (!this.isRunning) {
221
287
  throw new Error(`Can't stop process; it's not currently running (cmd: '${this.rep}')`);
@@ -224,12 +290,14 @@ class SubProcess extends EventEmitter {
224
290
  // the proc
225
291
  this.handleLastLines();
226
292
  return await new B((resolve, reject) => {
227
- this.proc.on('close', resolve);
293
+ this.proc?.on('close', resolve);
228
294
  this.expectingExit = true;
229
- this.proc.kill(signal);
295
+ this.proc?.kill(signal);
296
+ // this timeout needs unref() or node will wait for the timeout to fire before
297
+ // exiting the process.
230
298
  setTimeout(() => {
231
299
  reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));
232
- }, timeout);
300
+ }, timeout).unref();
233
301
  });
234
302
  }
235
303
 
@@ -239,8 +307,8 @@ class SubProcess extends EventEmitter {
239
307
  }
240
308
 
241
309
  return await new B((resolve, reject) => {
242
- this.proc.on('exit', (code) => {
243
- if (allowedExitCodes.indexOf(code) === -1) {
310
+ this.proc?.on('exit', (code) => {
311
+ if (code !== null && allowedExitCodes.indexOf(code) === -1) {
244
312
  reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));
245
313
  } else {
246
314
  resolve(code);
@@ -269,3 +337,10 @@ class SubProcess extends EventEmitter {
269
337
 
270
338
  export { SubProcess };
271
339
  export default SubProcess;
340
+
341
+ /**
342
+ * @callback StartDetector
343
+ * @param {string} stdout
344
+ * @param {string} [stderr]
345
+ * @returns {any}
346
+ */
package/package.json CHANGED
@@ -1,24 +1,26 @@
1
1
  {
2
2
  "name": "teen_process",
3
+ "version": "2.0.1",
3
4
  "description": "A grown up version of Node's spawn/exec",
4
5
  "keywords": [
5
6
  "child_process",
6
- "process management"
7
+ "process management",
8
+ "subprocess",
9
+ "spawn",
10
+ "exec",
11
+ "process",
12
+ "child"
7
13
  ],
8
- "version": "1.15.0",
9
- "author": "appium",
10
- "license": "Apache-2.0",
14
+ "bugs": {
15
+ "url": "https://github.com/appium/node-teen_process/issues"
16
+ },
11
17
  "repository": {
12
18
  "type": "git",
13
19
  "url": "https://github.com/appium/node-teen_process.git"
14
20
  },
15
- "bugs": {
16
- "url": "https://github.com/appium/node-teen_process/issues"
17
- },
18
- "engines": [
19
- "node"
20
- ],
21
- "main": "./build/index.js",
21
+ "license": "Apache-2.0",
22
+ "author": "Appium Contributors",
23
+ "main": "./index.js",
22
24
  "bin": {},
23
25
  "directories": {
24
26
  "lib": "lib"
@@ -26,42 +28,74 @@
26
28
  "files": [
27
29
  "index.js",
28
30
  "lib",
29
- "build/index.js",
30
31
  "build/lib"
31
32
  ],
32
- "dependencies": {
33
- "@babel/runtime": "^7.0.0",
34
- "bluebird": "^3.5.1",
35
- "lodash": "^4.17.4",
36
- "shell-quote": "^1.4.3",
37
- "source-map-support": "^0.5.3"
38
- },
39
33
  "scripts": {
40
- "clean": "rm -rf node_modules && rm -f package-lock.json && npm install",
41
- "build": "gulp transpile",
42
- "prepare": "gulp prepublish",
43
- "test": "gulp once",
44
- "watch": "gulp",
45
- "lint": "gulp eslint",
46
- "lint:fix": "gulp eslint --fix",
34
+ "build": "babel --out-dir=build/lib lib",
35
+ "dev": "npm run build -- --watch",
36
+ "lint": "eslint .",
37
+ "lint:fix": "npm run lint -- --fix",
38
+ "lint:types": "tsc",
39
+ "precommit-lint": "lint-staged",
47
40
  "precommit-msg": "echo 'Pre-commit checks...' && exit 0",
48
- "precommit-test": "REPORTER=dot gulp once"
41
+ "prepare": "npm run build",
42
+ "test": "mocha"
49
43
  },
50
44
  "pre-commit": [
51
45
  "precommit-msg",
52
- "precommit-test"
46
+ "precommit-lint"
53
47
  ],
48
+ "lint-staged": {
49
+ "*.js": [
50
+ "eslint --fix"
51
+ ]
52
+ },
53
+ "prettier": {
54
+ "bracketSpacing": false,
55
+ "printWidth": 100,
56
+ "singleQuote": true
57
+ },
58
+ "dependencies": {
59
+ "@babel/runtime": "7.18.9",
60
+ "bluebird": "3.7.2",
61
+ "lodash": "4.17.21",
62
+ "shell-quote": "1.7.3",
63
+ "source-map-support": "0.5.21",
64
+ "which": "2.0.2"
65
+ },
54
66
  "devDependencies": {
55
- "ajv": "^6.5.3",
56
- "appium-gulp-plugins": "^4.0.0",
57
- "appium-support": "^2.0.10",
58
- "chai": "^4.1.2",
59
- "chai-as-promised": "^7.1.1",
60
- "eslint-config-appium": "^4.0.0",
61
- "gulp": "^4.0.0",
62
- "pre-commit": "^1.2.2"
67
+ "@appium/eslint-config-appium": "6.0.4",
68
+ "@appium/support": "2.59.5",
69
+ "@babel/cli": "7.18.10",
70
+ "@babel/core": "7.18.10",
71
+ "@babel/eslint-parser": "7.18.9",
72
+ "@babel/plugin-transform-runtime": "7.18.10",
73
+ "@babel/preset-env": "7.18.10",
74
+ "@babel/register": "7.18.9",
75
+ "@types/bluebird": "3.5.36",
76
+ "@types/chai": "4.3.3",
77
+ "@types/chai-as-promised": "7.1.5",
78
+ "@types/lodash": "4.14.182",
79
+ "@types/mocha": "9.1.1",
80
+ "@types/node": "18.6.5",
81
+ "@types/shell-quote": "1.7.1",
82
+ "@types/which": "2.0.1",
83
+ "babel-plugin-source-map-support": "2.2.0",
84
+ "chai": "4.3.6",
85
+ "chai-as-promised": "7.1.1",
86
+ "eslint": "7.32.0",
87
+ "eslint-config-prettier": "8.5.0",
88
+ "eslint-plugin-import": "2.26.0",
89
+ "eslint-plugin-mocha": "9.0.0",
90
+ "eslint-plugin-promise": "6.0.0",
91
+ "lint-staged": "13.0.3",
92
+ "mocha": "10.0.0",
93
+ "pre-commit": "1.2.2",
94
+ "prettier": "2.7.1",
95
+ "typescript": "4.7.4"
63
96
  },
64
- "greenkeeper": {
65
- "ignore": []
97
+ "engines": {
98
+ "node": ">=14",
99
+ "npm": ">=6"
66
100
  }
67
101
  }
package/build/index.js DELETED
@@ -1,32 +0,0 @@
1
- "use strict";
2
-
3
- var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
4
-
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
8
- exports.SubProcess = exports.spawn = exports.exec = void 0;
9
-
10
- require("source-map-support/register");
11
-
12
- var cp = _interopRequireWildcard(require("child_process"));
13
-
14
- var spIndex = _interopRequireWildcard(require("./lib/subprocess"));
15
-
16
- var execIndex = _interopRequireWildcard(require("./lib/exec"));
17
-
18
- const {
19
- spawn
20
- } = cp;
21
- exports.spawn = spawn;
22
- const {
23
- SubProcess
24
- } = spIndex;
25
- exports.SubProcess = SubProcess;
26
- const {
27
- exec
28
- } = execIndex;
29
- exports.exec = exec;require('source-map-support').install();
30
-
31
-
32
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LmpzIl0sIm5hbWVzIjpbInNwYXduIiwiY3AiLCJTdWJQcm9jZXNzIiwic3BJbmRleCIsImV4ZWMiLCJleGVjSW5kZXgiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBR0EsTUFBTTtBQUFFQSxFQUFBQTtBQUFGLElBQVlDLEVBQWxCOztBQUNBLE1BQU07QUFBRUMsRUFBQUE7QUFBRixJQUFpQkMsT0FBdkI7O0FBQ0EsTUFBTTtBQUFFQyxFQUFBQTtBQUFGLElBQVdDLFNBQWpCIiwic291cmNlc0NvbnRlbnQiOlsiLy8gdHJhbnNwaWxlOm1haW5cbmltcG9ydCAqIGFzIGNwIGZyb20gJ2NoaWxkX3Byb2Nlc3MnO1xuaW1wb3J0ICogYXMgc3BJbmRleCBmcm9tICcuL2xpYi9zdWJwcm9jZXNzJztcbmltcG9ydCAqIGFzIGV4ZWNJbmRleCBmcm9tICcuL2xpYi9leGVjJztcblxuXG5jb25zdCB7IHNwYXduIH0gPSBjcDtcbmNvbnN0IHsgU3ViUHJvY2VzcyB9ID0gc3BJbmRleDtcbmNvbnN0IHsgZXhlYyB9ID0gZXhlY0luZGV4O1xuXG5leHBvcnQgeyBleGVjLCBzcGF3biwgU3ViUHJvY2VzcyB9O1xuIl0sImZpbGUiOiJpbmRleC5qcyIsInNvdXJjZVJvb3QiOiIuLiJ9