teen_process 1.14.1 → 1.16.0

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
@@ -50,6 +48,9 @@ The `exec` function takes some options, with these defaults:
50
48
  stdio: "inherit",
51
49
  isBuffer: false,
52
50
  shell: undefined,
51
+ logger: undefined,
52
+ maxStdoutBufferSize: 100 * 1024 * 1024, // 100 MB
53
+ maxStderrBufferSize: 100 * 1024 * 1024, // 100 MB
53
54
  }
54
55
  ```
55
56
 
@@ -57,6 +58,9 @@ Most of these are self-explanatory. `ignoreOutput` is useful if you have a very
57
58
  chatty process whose output you don't care about and don't want to add it to
58
59
  the memory consumed by your program.
59
60
 
61
+ Both buffer size limits are needed to avoid memory overflow while collecting
62
+ process output. If the overall size of output chunks for different stream types exceeds the the given one then the oldest chunks will be pulled out in order to keep the memory load within the acceptable ranges.
63
+
60
64
  If you're on Windows, you'll want to pass `shell: true`, because `exec`
61
65
  actually uses `spawn` under the hood, and is therefore subject to the issues
62
66
  noted about Windows + `spawn` in [the Node
@@ -84,6 +88,10 @@ let {stdout, stderr} = await exec('cat', [filename], {isBuffer: true});
84
88
  Buffer.isBuffer(stdout); // true
85
89
  ```
86
90
 
91
+ The `logger` option allows stdout and stderr to be sent to a particular logger,
92
+ as it it received. This is overridden by the `ignoreOutput` option.
93
+
94
+
87
95
  ## teen_process.SubProcess
88
96
 
89
97
  `spawn` is already pretty great but for some uses there's a fair amount of
package/build/lib/exec.js CHANGED
@@ -16,7 +16,13 @@ var _shellQuote = require("shell-quote");
16
16
 
17
17
  var _bluebird = _interopRequireDefault(require("bluebird"));
18
18
 
19
- function exec(cmd, args = [], opts = {}) {
19
+ var _lodash = _interopRequireDefault(require("lodash"));
20
+
21
+ var _helpers = require("./helpers");
22
+
23
+ const MAX_BUFFER_SIZE = 100 * 1024 * 1024;
24
+
25
+ async function exec(cmd, args = [], opts = {}) {
20
26
  const rep = (0, _shellQuote.quote)([cmd, ...args]);
21
27
  opts = Object.assign({
22
28
  timeout: null,
@@ -25,11 +31,14 @@ function exec(cmd, args = [], opts = {}) {
25
31
  cwd: undefined,
26
32
  env: process.env,
27
33
  ignoreOutput: false,
28
- stdio: "inherit",
34
+ stdio: 'inherit',
29
35
  isBuffer: false,
30
- shell: undefined
36
+ shell: undefined,
37
+ logger: undefined,
38
+ maxStdoutBufferSize: MAX_BUFFER_SIZE,
39
+ maxStderrBufferSize: MAX_BUFFER_SIZE
31
40
  }, opts);
32
- return new _bluebird.default((resolve, reject) => {
41
+ return await new _bluebird.default((resolve, reject) => {
33
42
  let proc = (0, _child_process.spawn)(cmd, args, {
34
43
  cwd: opts.cwd,
35
44
  env: opts.env,
@@ -39,13 +48,11 @@ function exec(cmd, args = [], opts = {}) {
39
48
  stderrArr = [],
40
49
  timer = null;
41
50
  proc.on('error', err => {
42
- let msg = `Command '${rep}' errored out: ${err.stack}`;
43
-
44
51
  if (err.errno === 'ENOENT') {
45
- msg = `Command '${cmd}' not found. Is it installed?`;
52
+ err = (0, _helpers.formatEnoent)(err, cmd, opts.cwd);
46
53
  }
47
54
 
48
- reject(new Error(msg));
55
+ reject(err);
49
56
  });
50
57
 
51
58
  if (proc.stdin) {
@@ -54,31 +61,48 @@ function exec(cmd, args = [], opts = {}) {
54
61
  });
55
62
  }
56
63
 
57
- if (proc.stdout) {
58
- proc.stdout.on('error', err => {
59
- reject(new Error(`Standard output '${err.syscall}' error: ${err.stack}`));
60
- });
61
- }
64
+ const handleStream = (streamType, streamProps) => {
65
+ if (!proc[streamType]) {
66
+ return;
67
+ }
62
68
 
63
- if (proc.stderr) {
64
- proc.stderr.on('error', err => {
65
- reject(new Error(`Standard error '${err.syscall}' error: ${err.stack}`));
69
+ proc[streamType].on('error', err => {
70
+ reject(new Error(`${_lodash.default.capitalize(streamType)} '${err.syscall}' error: ${err.stack}`));
66
71
  });
67
- }
68
72
 
69
- if (!opts.ignoreOutput) {
70
- if (proc.stdout) {
71
- proc.stdout.on('data', data => {
72
- stdoutArr.push(data);
73
- });
73
+ if (opts.ignoreOutput) {
74
+ proc[streamType].on('data', () => {});
75
+ return;
74
76
  }
75
77
 
76
- if (proc.stderr) {
77
- proc.stderr.on('data', data => {
78
- stderrArr.push(data);
79
- });
80
- }
81
- }
78
+ const {
79
+ chunks,
80
+ maxSize
81
+ } = streamProps;
82
+ let size = 0;
83
+ proc[streamType].on('data', chunk => {
84
+ chunks.push(chunk);
85
+ size += chunk.length;
86
+
87
+ while (chunks.length > 1 && size >= maxSize) {
88
+ size -= chunks[0].length;
89
+ chunks.shift();
90
+ }
91
+
92
+ if (opts.logger && _lodash.default.isFunction(opts.logger.debug)) {
93
+ opts.logger.debug(chunk.toString());
94
+ }
95
+ });
96
+ };
97
+
98
+ handleStream('stdout', {
99
+ maxSize: opts.maxStdoutBufferSize,
100
+ chunks: stdoutArr
101
+ });
102
+ handleStream('stderr', {
103
+ maxSize: opts.maxStderrBufferSize,
104
+ chunks: stderrArr
105
+ });
82
106
 
83
107
  function getStdio(isBuffer) {
84
108
  let stdout, stderr;
@@ -147,4 +171,4 @@ var _default = exec;
147
171
  exports.default = _default;require('source-map-support').install();
148
172
 
149
173
 
150
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/exec.js"],"names":["exec","cmd","args","opts","rep","Object","assign","timeout","encoding","killSignal","cwd","undefined","env","process","ignoreOutput","stdio","isBuffer","shell","B","resolve","reject","proc","stdoutArr","stderrArr","timer","on","err","msg","stack","errno","Error","stdin","syscall","stdout","stderr","data","push","getStdio","Buffer","concat","toString","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';\n\n\nfunction 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  }, opts);\n\n  // this is an async function, so return a promise\n  return 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    if (proc.stdout) {\n      proc.stdout.on('error', (err) => {\n        reject(new Error(`Standard output '${err.syscall}' error: ${err.stack}`));\n      });\n    }\n    if (proc.stderr) {\n      proc.stderr.on('error', (err) => {\n        reject(new Error(`Standard error '${err.syscall}' error: ${err.stack}`));\n      });\n    }\n\n    // keep track of stdout/stderr if we haven't said not to\n    if (!opts.ignoreOutput) {\n      if (proc.stdout) {\n        proc.stdout.on('data', (data) => {\n          stdoutArr.push(data);\n        });\n      }\n      if (proc.stderr) {\n        proc.stderr.on('data', (data) => {\n          stderrArr.push(data);\n        });\n      }\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":"../.."}
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","errno","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","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';\nimport { formatEnoent } from './helpers';\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      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', (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":"../.."}
@@ -0,0 +1,35 @@
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.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
+ }require('source-map-support').install();
33
+
34
+
35
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9oZWxwZXJzLmpzIl0sIm5hbWVzIjpbImZvcm1hdEVub2VudCIsImVycm9yIiwiY21kIiwiY3dkIiwid2hpY2giLCJzeW5jIiwiZnMiLCJhY2Nlc3NTeW5jIiwiUl9PSyIsImlnbiIsIm1lc3NhZ2UiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBQ0E7O0FBYUEsU0FBU0EsWUFBVCxDQUF1QkMsS0FBdkIsRUFBOEJDLEdBQTlCLEVBQW1DQyxHQUFHLEdBQUcsSUFBekMsRUFBK0M7QUFDN0MsTUFBSTtBQUNGQyxtQkFBTUMsSUFBTixDQUFXSCxHQUFYOztBQUNBLFFBQUlDLEdBQUosRUFBUztBQUNQLFVBQUk7QUFDRkcsb0JBQUdDLFVBQUgsQ0FBY0osR0FBZCxFQUFtQkcsWUFBR0UsSUFBdEI7QUFDRCxPQUZELENBRUUsT0FBT0MsR0FBUCxFQUFZO0FBQ1pSLFFBQUFBLEtBQUssQ0FBQ1MsT0FBTixHQUFpQixrQ0FBaUNQLEdBQUksVUFBU0QsR0FBSSxZQUFuRCxHQUNiLDRDQURIO0FBRUQ7QUFDRjtBQUNGLEdBVkQsQ0FVRSxPQUFPTyxHQUFQLEVBQVk7QUFDWlIsSUFBQUEsS0FBSyxDQUFDUyxPQUFOLEdBQWlCLFlBQVdSLEdBQUksK0JBQWhDO0FBQ0Q7O0FBQ0QsU0FBT0QsS0FBUDtBQUNEIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHdoaWNoIGZyb20gJ3doaWNoJztcbmltcG9ydCBmcyBmcm9tICdmcyc7XG5cbi8qKlxuICogRGVjb3JhdGVzIEVOT0VOVCBlcnJvciByZWNlaXZlZCBmcm9tIGEgc3Bhd24gc3lzdGVtIGNhbGxcbiAqIHdpdGggYSBtb3JlIGRlc2NyaXB0aXZlIG1lc3NhZ2UsIHNvIGl0IGNvdWxkIGJlIHByb3Blcmx5IGhhbmRsZWQgYnkgYSB1c2VyLlxuICpcbiAqIEBwYXJhbSB7IUVycm9yfSBlcnJvciBPcmlnaW5hbCBlcnJvciBpbnN0YW5jZS4gISEhIFRoZSBpbnN0YW5jZSBpcyBtdXRhdGVkIGFmdGVyXG4gKiB0aGlzIGhlbHBlciBmdW5jdGlvbiBpbnZvY2F0aW9uXG4gKiBAcGFyYW0geyFzdHJpbmd9IGNtZCBPcmlnaW5hbCBjb21tYW5kIHRvIGV4ZWN1dGVcbiAqIEBwYXJhbSB7P3N0cmluZ30gY3dkIE9wdGlvbmFsIHBhdGggdG8gdGhlIGN1cnJlbnQgd29ya2luZyBkaXJcbiAqIEByZXR1cm4ge0Vycm9yfSBNdXRhdGVkIGVycm9yIGluc3RhbmNlIHdpdGggYW4gaW1wcm92ZWQgZGVzY3JpcHRpb24gb3IgYW5cbiAqIHVuY2hhbmdlZCBlcnJvciBpbnN0YW5jZVxuICovXG5mdW5jdGlvbiBmb3JtYXRFbm9lbnQgKGVycm9yLCBjbWQsIGN3ZCA9IG51bGwpIHtcbiAgdHJ5IHtcbiAgICB3aGljaC5zeW5jKGNtZCk7XG4gICAgaWYgKGN3ZCkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgZnMuYWNjZXNzU3luYyhjd2QsIGZzLlJfT0spO1xuICAgICAgfSBjYXRjaCAoaWduKSB7XG4gICAgICAgIGVycm9yLm1lc3NhZ2UgPSBgVGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkgJyR7Y3dkfScgZm9yICcke2NtZH0nIGNvbW1hbmQgYCArXG4gICAgICAgICAgYGVpdGhlciBkb2VzIG5vdCBleGlzdCBvciBpcyBub3QgYWNjZXNzaWJsZWA7XG4gICAgICB9XG4gICAgfVxuICB9IGNhdGNoIChpZ24pIHtcbiAgICBlcnJvci5tZXNzYWdlID0gYENvbW1hbmQgJyR7Y21kfScgbm90IGZvdW5kLiBJcyBpdCBpbnN0YWxsZWQ/YDtcbiAgfVxuICByZXR1cm4gZXJyb3I7XG59XG5cbmV4cG9ydCB7IGZvcm1hdEVub2VudCB9O1xuIl0sImZpbGUiOiJsaWIvaGVscGVycy5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLiJ9
@@ -13,17 +13,22 @@ var _child_process = require("child_process");
13
13
 
14
14
  var _events = _interopRequireDefault(require("events"));
15
15
 
16
- var _through = _interopRequireDefault(require("through"));
17
-
18
16
  var _bluebird = _interopRequireDefault(require("bluebird"));
19
17
 
20
18
  var _shellQuote = require("shell-quote");
21
19
 
22
20
  var _lodash = _interopRequireDefault(require("lodash"));
23
21
 
22
+ var _helpers = require("./helpers");
23
+
24
24
  const {
25
25
  EventEmitter
26
26
  } = _events.default;
27
+ const MAX_LINE_PORTION_LENGTH = 0xFFFF;
28
+
29
+ function cutSuffix(str, suffixLength) {
30
+ return str.length > suffixLength ? ` ${str.substr(str.length - suffixLength)}`.substr(1) : str;
31
+ }
27
32
 
28
33
  class SubProcess extends EventEmitter {
29
34
  constructor(cmd, args = [], opts = {}) {
@@ -97,9 +102,14 @@ class SubProcess extends EventEmitter {
97
102
  stderr: ''
98
103
  };
99
104
 
100
- const handleOutput = data => {
105
+ const handleOutput = streams => {
106
+ const {
107
+ stdout,
108
+ stderr
109
+ } = streams;
110
+
101
111
  try {
102
- if (startDetector && startDetector(data.stdout, data.stderr)) {
112
+ if (startDetector && startDetector(stdout, stderr)) {
103
113
  startDetector = null;
104
114
  resolve();
105
115
  }
@@ -107,20 +117,26 @@ class SubProcess extends EventEmitter {
107
117
  reject(e);
108
118
  }
109
119
 
110
- this.emit('output', data.stdout, data.stderr);
120
+ this.emit('output', stdout, stderr);
111
121
 
112
- for (const stream of ['stdout', 'stderr']) {
113
- if (!data[stream]) continue;
114
- let lines = data[stream].split("\n");
122
+ for (const [streamName, streamData] of _lodash.default.toPairs(streams)) {
123
+ if (!streamData) continue;
124
+ const lines = streamData.split('\n').map(x => ` ${x}`.substr(1));
115
125
 
116
126
  if (lines.length > 1) {
117
- let retLines = lines.slice(0, -1);
118
- retLines[0] = this.lastLinePortion[stream] + retLines[0];
119
- this.lastLinePortion[stream] = lines[lines.length - 1];
120
- this.emit(`lines-${stream}`, retLines);
121
- this.emitLines(stream, retLines);
127
+ lines[0] = this.lastLinePortion[streamName] + lines[0];
128
+ this.lastLinePortion[streamName] = cutSuffix(_lodash.default.last(lines), MAX_LINE_PORTION_LENGTH);
129
+ const resultLines = lines.slice(0, -1);
130
+ this.emit(`lines-${streamName}`, resultLines);
131
+ this.emitLines(streamName, resultLines);
122
132
  } else {
123
- this.lastLinePortion[stream] += lines[0];
133
+ const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);
134
+
135
+ if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {
136
+ this.lastLinePortion[streamName] = currentPortion;
137
+ } else {
138
+ this.lastLinePortion[streamName] += currentPortion;
139
+ }
124
140
  }
125
141
  }
126
142
  };
@@ -130,27 +146,25 @@ class SubProcess extends EventEmitter {
130
146
  this.proc.kill('SIGINT');
131
147
 
132
148
  if (err.errno === 'ENOENT') {
133
- err = new Error(`Command '${this.cmd}' not found. Is it installed?`);
149
+ var _this$opts;
150
+
151
+ err = (0, _helpers.formatEnoent)(err, this.cmd, (_this$opts = this.opts) === null || _this$opts === void 0 ? void 0 : _this$opts.cwd);
134
152
  }
135
153
 
136
154
  reject(err);
137
155
  });
138
156
 
139
157
  if (this.proc.stdout) {
140
- this.proc.stdout.pipe((0, _through.default)(stdout => {
141
- handleOutput({
142
- stdout,
143
- stderr: ''
144
- });
158
+ this.proc.stdout.on('data', chunk => handleOutput({
159
+ stdout: chunk.toString(),
160
+ stderr: ''
145
161
  }));
146
162
  }
147
163
 
148
164
  if (this.proc.stderr) {
149
- this.proc.stderr.pipe((0, _through.default)(stderr => {
150
- handleOutput({
151
- stdout: '',
152
- stderr
153
- });
165
+ this.proc.stderr.on('data', chunk => handleOutput({
166
+ stdout: '',
167
+ stderr: chunk.toString()
154
168
  }));
155
169
  }
156
170
 
@@ -250,4 +264,4 @@ var _default = SubProcess;
250
264
  exports.default = _default;require('source-map-support').install();
251
265
 
252
266
 
253
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/subprocess.js"],"names":["EventEmitter","events","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","data","e","split","length","retLines","slice","on","err","removeAllListeners","kill","errno","pipe","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';\nimport through from 'through';\nconst { EventEmitter } = events;\nimport B from 'bluebird';\nimport { quote } from 'shell-quote';\nimport _ from 'lodash';\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 = (data) => {\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(data.stdout, data.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', data.stdout, data.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 stream of ['stdout', 'stderr']) {\n          if (!data[stream]) continue; // eslint-disable-line curly\n          let lines = data[stream].split(\"\\n\");\n          if (lines.length > 1) {\n            let retLines = lines.slice(0, -1);\n            retLines[0] = this.lastLinePortion[stream] + retLines[0];\n            this.lastLinePortion[stream] = lines[lines.length - 1];\n            this.emit(`lines-${stream}`, retLines);\n            this.emitLines(stream, retLines);\n          } else {\n            this.lastLinePortion[stream] += lines[0];\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.pipe(through(stdout => {\n          handleOutput({stdout, stderr: ''});\n        }));\n      }\n\n      if (this.proc.stderr) {\n        this.proc.stderr.pipe(through(stderr => {\n          handleOutput({stdout: '', stderr});\n        }));\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":"../.."}
267
+ //# 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","cwd","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';\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  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 = formatEnoent(err, this.cmd, this.opts?.cwd);\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":"../.."}
package/lib/exec.js CHANGED
@@ -3,9 +3,12 @@
3
3
  import { spawn } from 'child_process';
4
4
  import { quote } from 'shell-quote';
5
5
  import B from 'bluebird';
6
+ import _ from 'lodash';
7
+ import { formatEnoent } from './helpers';
6
8
 
9
+ const MAX_BUFFER_SIZE = 100 * 1024 * 1024;
7
10
 
8
- function exec (cmd, args = [], opts = {}) {
11
+ async function exec (cmd, args = [], opts = {}) {
9
12
  // get a quoted representation of the command for error strings
10
13
  const rep = quote([cmd, ...args]);
11
14
 
@@ -18,13 +21,16 @@ function exec (cmd, args = [], opts = {}) {
18
21
  cwd: undefined,
19
22
  env: process.env,
20
23
  ignoreOutput: false,
21
- stdio: "inherit",
24
+ stdio: 'inherit',
22
25
  isBuffer: false,
23
26
  shell: undefined,
27
+ logger: undefined,
28
+ maxStdoutBufferSize: MAX_BUFFER_SIZE,
29
+ maxStderrBufferSize: MAX_BUFFER_SIZE,
24
30
  }, opts);
25
31
 
26
32
  // this is an async function, so return a promise
27
- return new B((resolve, reject) => {
33
+ return await new B((resolve, reject) => {
28
34
  // spawn the child process with options; we don't currently expose any of
29
35
  // the other 'spawn' options through the API
30
36
  let proc = spawn(cmd, args, {cwd: opts.cwd, env: opts.env, shell: opts.shell});
@@ -32,41 +38,54 @@ function exec (cmd, args = [], opts = {}) {
32
38
 
33
39
  // if the process errors out, reject the promise
34
40
  proc.on('error', (err) => {
35
- let msg = `Command '${rep}' errored out: ${err.stack}`;
36
41
  if (err.errno === 'ENOENT') {
37
- msg = `Command '${cmd}' not found. Is it installed?`;
42
+ err = formatEnoent(err, cmd, opts.cwd);
38
43
  }
39
- reject(new Error(msg));
44
+ reject(err);
40
45
  });
41
46
  if (proc.stdin) {
42
47
  proc.stdin.on('error', (err) => {
43
48
  reject(new Error(`Standard input '${err.syscall}' error: ${err.stack}`));
44
49
  });
45
50
  }
46
- if (proc.stdout) {
47
- proc.stdout.on('error', (err) => {
48
- reject(new Error(`Standard output '${err.syscall}' error: ${err.stack}`));
49
- });
50
- }
51
- if (proc.stderr) {
52
- proc.stderr.on('error', (err) => {
53
- reject(new Error(`Standard error '${err.syscall}' error: ${err.stack}`));
51
+ const handleStream = (streamType, streamProps) => {
52
+ if (!proc[streamType]) {
53
+ return;
54
+ }
55
+
56
+ proc[streamType].on('error', (err) => {
57
+ reject(new Error(`${_.capitalize(streamType)} '${err.syscall}' error: ${err.stack}`));
54
58
  });
55
- }
56
59
 
57
- // keep track of stdout/stderr if we haven't said not to
58
- if (!opts.ignoreOutput) {
59
- if (proc.stdout) {
60
- proc.stdout.on('data', (data) => {
61
- stdoutArr.push(data);
62
- });
63
- }
64
- if (proc.stderr) {
65
- proc.stderr.on('data', (data) => {
66
- stderrArr.push(data);
67
- });
60
+ if (opts.ignoreOutput) {
61
+ // https://github.com/nodejs/node/issues/4236
62
+ proc[streamType].on('data', () => {});
63
+ return;
68
64
  }
69
- }
65
+
66
+ // keep track of the stream if we don't want to ignore it
67
+ const {chunks, maxSize} = streamProps;
68
+ let size = 0;
69
+ proc[streamType].on('data', (chunk) => {
70
+ chunks.push(chunk);
71
+ size += chunk.length;
72
+ while (chunks.length > 1 && size >= maxSize) {
73
+ size -= chunks[0].length;
74
+ chunks.shift();
75
+ }
76
+ if (opts.logger && _.isFunction(opts.logger.debug)) {
77
+ opts.logger.debug(chunk.toString());
78
+ }
79
+ });
80
+ };
81
+ handleStream('stdout', {
82
+ maxSize: opts.maxStdoutBufferSize,
83
+ chunks: stdoutArr,
84
+ });
85
+ handleStream('stderr', {
86
+ maxSize: opts.maxStderrBufferSize,
87
+ chunks: stderrArr,
88
+ });
70
89
 
71
90
  function getStdio (isBuffer) {
72
91
  let stdout, stderr;
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 {!Error} 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} cwd Optional path to the current working dir
12
+ * @return {Error} 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.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/subprocess.js CHANGED
@@ -2,11 +2,24 @@
2
2
 
3
3
  import { spawn } from 'child_process';
4
4
  import events from 'events';
5
- import through from 'through';
6
5
  const { EventEmitter } = events;
7
6
  import B from 'bluebird';
8
7
  import { quote } from 'shell-quote';
9
8
  import _ from 'lodash';
9
+ import { formatEnoent } from './helpers';
10
+
11
+
12
+ // This is needed to avoid memory leaks
13
+ // when the process output is too long and contains
14
+ // no line breaks
15
+ const MAX_LINE_PORTION_LENGTH = 0xFFFF;
16
+
17
+ function cutSuffix (str, suffixLength) {
18
+ return str.length > suffixLength
19
+ // https://bugs.chromium.org/p/v8/issues/detail?id=2869
20
+ ? ` ${str.substr(str.length - suffixLength)}`.substr(1)
21
+ : str;
22
+ }
10
23
 
11
24
 
12
25
  class SubProcess extends EventEmitter {
@@ -87,11 +100,12 @@ class SubProcess extends EventEmitter {
87
100
  this.lastLinePortion = {stdout: '', stderr: ''};
88
101
 
89
102
  // this function handles output that we collect from the subproc
90
- const handleOutput = (data) => {
103
+ const handleOutput = (streams) => {
104
+ const {stdout, stderr} = streams;
91
105
  // if we have a startDetector, run it on the output so we can resolve/
92
106
  // reject and move on from start
93
107
  try {
94
- if (startDetector && startDetector(data.stdout, data.stderr)) {
108
+ if (startDetector && startDetector(stdout, stderr)) {
95
109
  startDetector = null;
96
110
  resolve();
97
111
  }
@@ -100,48 +114,51 @@ class SubProcess extends EventEmitter {
100
114
  }
101
115
 
102
116
  // emit the actual output for whomever's listening
103
- this.emit('output', data.stdout, data.stderr);
117
+ this.emit('output', stdout, stderr);
104
118
 
105
119
  // we also want to emit lines, but it's more complex since output
106
120
  // comes in chunks and a line could come in two different chunks, so
107
121
  // we have logic to handle that case (using this.lastLinePortion to
108
122
  // remember a line that started but did not finish in the last chunk)
109
- for (const stream of ['stdout', 'stderr']) {
110
- if (!data[stream]) continue; // eslint-disable-line curly
111
- let lines = data[stream].split("\n");
123
+ for (const [streamName, streamData] of _.toPairs(streams)) {
124
+ if (!streamData) continue; // eslint-disable-line curly
125
+ const lines = streamData.split('\n')
126
+ // https://bugs.chromium.org/p/v8/issues/detail?id=2869
127
+ .map((x) => ` ${x}`.substr(1));
112
128
  if (lines.length > 1) {
113
- let retLines = lines.slice(0, -1);
114
- retLines[0] = this.lastLinePortion[stream] + retLines[0];
115
- this.lastLinePortion[stream] = lines[lines.length - 1];
116
- this.emit(`lines-${stream}`, retLines);
117
- this.emitLines(stream, retLines);
129
+ lines[0] = this.lastLinePortion[streamName] + lines[0];
130
+ this.lastLinePortion[streamName] = cutSuffix(_.last(lines), MAX_LINE_PORTION_LENGTH);
131
+ const resultLines = lines.slice(0, -1);
132
+ this.emit(`lines-${streamName}`, resultLines);
133
+ this.emitLines(streamName, resultLines);
118
134
  } else {
119
- this.lastLinePortion[stream] += lines[0];
135
+ const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);
136
+ if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {
137
+ this.lastLinePortion[streamName] = currentPortion;
138
+ } else {
139
+ this.lastLinePortion[streamName] += currentPortion;
140
+ }
120
141
  }
121
142
  }
122
143
  };
123
144
 
124
145
  // if we get an error spawning the proc, reject and clean up the proc
125
- this.proc.on('error', err => {
146
+ this.proc.on('error', (err) => {
126
147
  this.proc.removeAllListeners('exit');
127
148
  this.proc.kill('SIGINT');
128
149
 
129
150
  if (err.errno === 'ENOENT') {
130
- err = new Error(`Command '${this.cmd}' not found. Is it installed?`);
151
+ err = formatEnoent(err, this.cmd, this.opts?.cwd);
131
152
  }
132
153
  reject(err);
133
154
  });
134
155
 
135
156
  if (this.proc.stdout) {
136
- this.proc.stdout.pipe(through(stdout => {
137
- handleOutput({stdout, stderr: ''});
138
- }));
157
+ this.proc.stdout.on('data', (chunk) => handleOutput({stdout: chunk.toString(), stderr: ''}));
139
158
  }
140
159
 
141
160
  if (this.proc.stderr) {
142
- this.proc.stderr.pipe(through(stderr => {
143
- handleOutput({stdout: '', stderr});
144
- }));
161
+ this.proc.stderr.on('data', (chunk) => handleOutput({stdout: '', stderr: chunk.toString()}));
145
162
  }
146
163
 
147
164
  // when the proc exits, we might still have a buffer of lines we were
package/package.json CHANGED
@@ -5,7 +5,7 @@
5
5
  "child_process",
6
6
  "process management"
7
7
  ],
8
- "version": "1.14.1",
8
+ "version": "1.16.0",
9
9
  "author": "appium",
10
10
  "license": "Apache-2.0",
11
11
  "repository": {
@@ -31,12 +31,11 @@
31
31
  ],
32
32
  "dependencies": {
33
33
  "@babel/runtime": "^7.0.0",
34
- "appium-support": "^2.0.10",
35
34
  "bluebird": "^3.5.1",
36
35
  "lodash": "^4.17.4",
37
36
  "shell-quote": "^1.4.3",
38
37
  "source-map-support": "^0.5.3",
39
- "through": "^2.3.8"
38
+ "which": "^2.0.2"
40
39
  },
41
40
  "scripts": {
42
41
  "clean": "rm -rf node_modules && rm -f package-lock.json && npm install",
@@ -54,20 +53,12 @@
54
53
  "precommit-test"
55
54
  ],
56
55
  "devDependencies": {
57
- "ajv": "^6.5.3",
58
- "appium-gulp-plugins": "^3.1.0",
59
- "babel-eslint": "^10.0.0",
56
+ "appium-gulp-plugins": "^5.4.1",
57
+ "appium-support": "^2.0.10",
60
58
  "chai": "^4.1.2",
61
59
  "chai-as-promised": "^7.1.1",
62
- "eslint": "^5.2.0",
63
- "eslint-config-appium": "^3.1.0",
64
- "eslint-plugin-import": "^2.2.0",
65
- "eslint-plugin-mocha": "^5.0.0",
66
- "eslint-plugin-promise": "^4.0.0",
60
+ "eslint-config-appium": "^4.0.0",
67
61
  "gulp": "^4.0.0",
68
62
  "pre-commit": "^1.2.2"
69
- },
70
- "greenkeeper": {
71
- "ignore": []
72
63
  }
73
64
  }