teen_process 2.0.2 → 2.0.4

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.
@@ -1,285 +1,290 @@
1
1
  "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
4
-
5
- Object.defineProperty(exports, "__esModule", {
6
- value: true
7
- });
8
- exports.default = exports.SubProcess = void 0;
9
-
10
- require("source-map-support/register");
11
-
12
- var _child_process = require("child_process");
13
-
14
- var _events = _interopRequireDefault(require("events"));
15
-
16
- var _bluebird = _interopRequireDefault(require("bluebird"));
17
-
18
- var _shellQuote = require("shell-quote");
19
-
20
- var _lodash = _interopRequireDefault(require("lodash"));
21
-
22
- var _helpers = require("./helpers");
23
-
24
- const {
25
- EventEmitter
26
- } = _events.default;
2
+ /* eslint-disable promise/prefer-await-to-callbacks */
3
+ var __importDefault = (this && this.__importDefault) || function (mod) {
4
+ return (mod && mod.__esModule) ? mod : { "default": mod };
5
+ };
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.SubProcess = void 0;
8
+ const child_process_1 = require("child_process");
9
+ const events_1 = __importDefault(require("events"));
10
+ const { EventEmitter } = events_1.default;
11
+ const bluebird_1 = __importDefault(require("bluebird"));
12
+ const shell_quote_1 = require("shell-quote");
13
+ const lodash_1 = __importDefault(require("lodash"));
14
+ const helpers_1 = require("./helpers");
15
+ // This is needed to avoid memory leaks
16
+ // when the process output is too long and contains
17
+ // no line breaks
27
18
  const MAX_LINE_PORTION_LENGTH = 0xFFFF;
28
-
29
19
  function cutSuffix(str, suffixLength) {
30
- return str.length > suffixLength ? ` ${str.substr(str.length - suffixLength)}`.substr(1) : str;
20
+ return str.length > suffixLength
21
+ // https://bugs.chromium.org/p/v8/issues/detail?id=2869
22
+ ? ` ${str.substr(str.length - suffixLength)}`.substr(1)
23
+ : str;
31
24
  }
32
-
33
25
  class SubProcess extends EventEmitter {
34
- lastLinePortion;
35
- proc;
36
- args;
37
- cmd;
38
- opts;
39
- expectingExit;
40
- rep;
41
-
42
- constructor(cmd, args = [], opts = {}) {
43
- super();
44
- if (!cmd) throw new Error('Command is required');
45
- if (!_lodash.default.isString(cmd)) throw new Error('Command must be a string');
46
- if (!_lodash.default.isArray(args)) throw new Error('Args must be an array');
47
- this.cmd = cmd;
48
- this.args = args;
49
- this.proc = null;
50
- this.opts = opts;
51
- this.expectingExit = false;
52
- this.rep = (0, _shellQuote.quote)([cmd, ...args]);
53
- this.lastLinePortion = {
54
- stdout: '',
55
- stderr: ''
56
- };
57
- }
58
-
59
- get isRunning() {
60
- return !!this.proc;
61
- }
62
-
63
- emitLines(stream, lines) {
64
- for (let line of lines) {
65
- this.emit('stream-line', `[${stream.toUpperCase()}] ${line}`);
66
- }
67
- }
68
-
69
- async start(startDetector = null, timeoutMs = null, detach = false) {
70
- let startDelay = 10;
71
-
72
- const genericStartDetector = function genericStartDetector(stdout, stderr) {
73
- return stdout || stderr;
74
- };
75
-
76
- if (startDetector === null) {
77
- startDetector = genericStartDetector;
26
+ /**
27
+ * @param {string} cmd
28
+ * @param {string[]} [args]
29
+ * @param {any} [opts]
30
+ */
31
+ constructor(cmd, args = [], opts = {}) {
32
+ super();
33
+ if (!cmd)
34
+ throw new Error('Command is required'); // eslint-disable-line curly
35
+ if (!lodash_1.default.isString(cmd))
36
+ throw new Error('Command must be a string'); // eslint-disable-line curly
37
+ if (!lodash_1.default.isArray(args))
38
+ throw new Error('Args must be an array'); // eslint-disable-line curly
39
+ this.cmd = cmd;
40
+ this.args = args;
41
+ this.proc = null;
42
+ this.opts = opts;
43
+ this.expectingExit = false;
44
+ // get a quoted representation of the command for error strings
45
+ this.rep = (0, shell_quote_1.quote)([cmd, ...args]);
46
+ this.lastLinePortion = { stdout: '', stderr: '' };
78
47
  }
79
-
80
- if (_lodash.default.isNumber(startDetector)) {
81
- startDelay = startDetector;
82
- startDetector = null;
48
+ get isRunning() {
49
+ // presence of `proc` means we have connected and started
50
+ return !!this.proc;
83
51
  }
84
-
85
- if (_lodash.default.isBoolean(startDetector) && startDetector) {
86
- if (!this.opts.detached) {
87
- throw new Error(`Unable to detach process that is not started with 'detached' option`);
88
- }
89
-
90
- detach = true;
91
- startDetector = genericStartDetector;
92
- } else if (_lodash.default.isBoolean(timeoutMs) && timeoutMs) {
93
- if (!this.opts.detached) {
94
- throw new Error(`Unable to detach process that is not started with 'detached' option`);
95
- }
96
-
97
- detach = true;
98
- timeoutMs = null;
52
+ /**
53
+ *
54
+ * @param {string} stream
55
+ * @param {Iterable<string>} lines
56
+ */
57
+ emitLines(stream, lines) {
58
+ for (let line of lines) {
59
+ this.emit('stream-line', `[${stream.toUpperCase()}] ${line}`);
60
+ }
99
61
  }
100
-
101
- return await new _bluebird.default((resolve, reject) => {
102
- this.proc = (0, _child_process.spawn)(this.cmd, this.args, this.opts);
103
-
104
- if (this.proc.stdout) {
105
- this.proc.stdout.setEncoding(this.opts.encoding || 'utf8');
106
- }
107
-
108
- if (this.proc.stderr) {
109
- this.proc.stderr.setEncoding(this.opts.encoding || 'utf8');
110
- }
111
-
112
- this.lastLinePortion = {
113
- stdout: '',
114
- stderr: ''
115
- };
116
-
117
- const handleOutput = streams => {
118
- const {
119
- stdout,
120
- stderr
121
- } = streams;
122
-
123
- try {
124
- if (_lodash.default.isFunction(startDetector) && startDetector(stdout, stderr)) {
62
+ // spawn the subprocess and return control whenever we deem that it has fully
63
+ // "started"
64
+ /**
65
+ *
66
+ * @param {StartDetector|number?} startDetector
67
+ * @param {number?} timeoutMs
68
+ * @param {boolean} detach
69
+ * @returns {Promise<void>}
70
+ */
71
+ async start(startDetector = null, timeoutMs = null, detach = false) {
72
+ let startDelay = 10;
73
+ const genericStartDetector = /** @type {StartDetector} */ (function genericStartDetector(stdout, stderr) {
74
+ return stdout || stderr;
75
+ });
76
+ // the default start detector simply returns true when we get any output
77
+ if (startDetector === null) {
78
+ startDetector = genericStartDetector;
79
+ }
80
+ // if the user passes a number, then we simply delay a certain amount of
81
+ // time before returning control, rather than waiting for a condition
82
+ if (lodash_1.default.isNumber(startDetector)) {
83
+ startDelay = startDetector;
125
84
  startDetector = null;
126
- resolve();
127
- }
128
- } catch (e) {
129
- reject(e);
130
85
  }
131
-
132
- this.emit('output', stdout, stderr);
133
-
134
- for (const [streamName, streamData] of _lodash.default.toPairs(streams)) {
135
- if (!streamData) continue;
136
- const lines = streamData.split('\n').map(x => ` ${x}`.substr(1));
137
-
138
- if (lines.length > 1) {
139
- lines[0] = this.lastLinePortion[streamName] + lines[0];
140
- this.lastLinePortion[streamName] = cutSuffix(_lodash.default.last(lines), MAX_LINE_PORTION_LENGTH);
141
- const resultLines = lines.slice(0, -1);
142
- this.emit(`lines-${streamName}`, resultLines);
143
- this.emitLines(streamName, resultLines);
144
- } else {
145
- const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);
146
-
147
- if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {
148
- this.lastLinePortion[streamName] = currentPortion;
149
- } else {
150
- this.lastLinePortion[streamName] += currentPortion;
86
+ // if the user passes in a boolean as one of the arguments, use it for `detach`
87
+ if (lodash_1.default.isBoolean(startDetector) && startDetector) {
88
+ if (!this.opts.detached) {
89
+ throw new Error(`Unable to detach process that is not started with 'detached' option`);
151
90
  }
152
- }
91
+ detach = true;
92
+ startDetector = genericStartDetector;
153
93
  }
154
- };
155
-
156
- this.proc.on('error', err => {
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;
164
-
165
- err = (0, _helpers.formatEnoent)(err, this.cmd, (_this$opts = this.opts) === null || _this$opts === void 0 ? void 0 : _this$opts.cwd);
94
+ else if (lodash_1.default.isBoolean(timeoutMs) && timeoutMs) {
95
+ if (!this.opts.detached) {
96
+ throw new Error(`Unable to detach process that is not started with 'detached' option`);
97
+ }
98
+ detach = true;
99
+ timeoutMs = null;
166
100
  }
167
-
168
- reject(err);
169
- (_this$proc3 = this.proc) === null || _this$proc3 === void 0 ? void 0 : _this$proc3.unref();
170
- this.proc = null;
171
- });
172
-
173
- if (this.proc.stdout) {
174
- this.proc.stdout.on('data', chunk => handleOutput({
175
- stdout: chunk.toString(),
176
- stderr: ''
177
- }));
178
- }
179
-
180
- if (this.proc.stderr) {
181
- this.proc.stderr.on('data', chunk => handleOutput({
182
- stdout: '',
183
- stderr: chunk.toString()
184
- }));
185
- }
186
-
187
- this.proc.on('exit', (code, signal) => {
188
- this.handleLastLines();
189
- this.emit('exit', code, signal);
190
- let event = this.expectingExit ? 'stop' : 'die';
191
-
192
- if (!this.expectingExit && code === 0) {
193
- event = 'end';
101
+ // return a promise so we can wrap the async behavior
102
+ return await new bluebird_1.default((resolve, reject) => {
103
+ // actually spawn the subproc
104
+ this.proc = (0, child_process_1.spawn)(this.cmd, this.args, this.opts);
105
+ if (this.proc.stdout) {
106
+ this.proc.stdout.setEncoding(this.opts.encoding || 'utf8');
107
+ }
108
+ if (this.proc.stderr) {
109
+ this.proc.stderr.setEncoding(this.opts.encoding || 'utf8');
110
+ }
111
+ this.lastLinePortion = { stdout: '', stderr: '' };
112
+ // this function handles output that we collect from the subproc
113
+ /**
114
+ *
115
+ * @param { {stdout: string, stderr: string} } streams
116
+ */
117
+ const handleOutput = (streams) => {
118
+ const { stdout, stderr } = streams;
119
+ // if we have a startDetector, run it on the output so we can resolve/
120
+ // reject and move on from start
121
+ try {
122
+ if (lodash_1.default.isFunction(startDetector) && startDetector(stdout, stderr)) {
123
+ startDetector = null;
124
+ resolve();
125
+ }
126
+ }
127
+ catch (e) {
128
+ reject(e);
129
+ }
130
+ // emit the actual output for whomever's listening
131
+ this.emit('output', stdout, stderr);
132
+ // we also want to emit lines, but it's more complex since output
133
+ // comes in chunks and a line could come in two different chunks, so
134
+ // we have logic to handle that case (using this.lastLinePortion to
135
+ // remember a line that started but did not finish in the last chunk)
136
+ for (const [streamName, streamData] of /** @type {[['stdout', string], ['stderr', string]]} */ (lodash_1.default.toPairs(streams))) {
137
+ if (!streamData)
138
+ continue; // eslint-disable-line curly
139
+ const lines = streamData.split('\n')
140
+ // https://bugs.chromium.org/p/v8/issues/detail?id=2869
141
+ .map((x) => ` ${x}`.substr(1));
142
+ if (lines.length > 1) {
143
+ lines[0] = this.lastLinePortion[streamName] + lines[0];
144
+ this.lastLinePortion[streamName] = cutSuffix(lodash_1.default.last(lines), MAX_LINE_PORTION_LENGTH);
145
+ const resultLines = lines.slice(0, -1);
146
+ this.emit(`lines-${streamName}`, resultLines);
147
+ this.emitLines(streamName, resultLines);
148
+ }
149
+ else {
150
+ const currentPortion = cutSuffix(lines[0], MAX_LINE_PORTION_LENGTH);
151
+ if (this.lastLinePortion[streamName].length + currentPortion.length > MAX_LINE_PORTION_LENGTH) {
152
+ this.lastLinePortion[streamName] = currentPortion;
153
+ }
154
+ else {
155
+ this.lastLinePortion[streamName] += currentPortion;
156
+ }
157
+ }
158
+ }
159
+ };
160
+ // if we get an error spawning the proc, reject and clean up the proc
161
+ this.proc.on('error', /** @param {NodeJS.ErrnoException} err */ async (err) => {
162
+ this.proc?.removeAllListeners('exit');
163
+ this.proc?.kill('SIGINT');
164
+ if (err.code === 'ENOENT') {
165
+ err = await (0, helpers_1.formatEnoent)(err, this.cmd, this.opts?.cwd);
166
+ }
167
+ reject(err);
168
+ this.proc?.unref();
169
+ this.proc = null;
170
+ });
171
+ if (this.proc.stdout) {
172
+ this.proc.stdout.on('data', (chunk) => handleOutput({ stdout: chunk.toString(), stderr: '' }));
173
+ }
174
+ if (this.proc.stderr) {
175
+ this.proc.stderr.on('data', (chunk) => handleOutput({ stdout: '', stderr: chunk.toString() }));
176
+ }
177
+ // when the proc exits, we might still have a buffer of lines we were
178
+ // waiting on more chunks to complete. Go ahead and emit those, then
179
+ // re-emit the exit so a listener can handle the possibly-unexpected exit
180
+ this.proc.on('exit', (code, signal) => {
181
+ this.handleLastLines();
182
+ this.emit('exit', code, signal);
183
+ // in addition to the bare exit event, also emit one of three other
184
+ // events that contain more helpful information:
185
+ // 'stop': we stopped this
186
+ // 'die': the process ended out of our control with a non-zero exit
187
+ // 'end': the process ended out of our control with a zero exit
188
+ let event = this.expectingExit ? 'stop' : 'die';
189
+ if (!this.expectingExit && code === 0) {
190
+ event = 'end';
191
+ }
192
+ this.emit(event, code, signal);
193
+ // finally clean up the proc and make sure to reset our exit
194
+ // expectations
195
+ this.proc = null;
196
+ this.expectingExit = false;
197
+ });
198
+ // if the user hasn't given us a startDetector, instead just resolve
199
+ // when startDelay ms have passed
200
+ if (!startDetector) {
201
+ setTimeout(() => { resolve(); }, startDelay);
202
+ }
203
+ // if the user has given us a timeout, start the clock for rejecting
204
+ // the promise if we take too long to start
205
+ if (lodash_1.default.isNumber(timeoutMs)) {
206
+ setTimeout(() => {
207
+ reject(new Error(`The process did not start within ${timeoutMs}ms ` +
208
+ `(cmd: '${this.rep}')`));
209
+ }, timeoutMs);
210
+ }
211
+ }).finally(() => {
212
+ if (detach && this.proc) {
213
+ this.proc.unref();
214
+ }
215
+ });
216
+ }
217
+ handleLastLines() {
218
+ for (let stream of ['stdout', 'stderr']) {
219
+ if (this.lastLinePortion[stream]) {
220
+ const lastLines = [this.lastLinePortion[stream]];
221
+ this.emit(`lines-${stream}`, lastLines);
222
+ this.emitLines(stream, lastLines);
223
+ this.lastLinePortion[stream] = '';
224
+ }
194
225
  }
195
-
196
- this.emit(event, code, signal);
197
- this.proc = null;
198
- this.expectingExit = false;
199
- });
200
-
201
- if (!startDetector) {
202
- setTimeout(() => {
203
- resolve();
204
- }, startDelay);
205
- }
206
-
207
- if (_lodash.default.isNumber(timeoutMs)) {
208
- setTimeout(() => {
209
- reject(new Error(`The process did not start within ${timeoutMs}ms ` + `(cmd: '${this.rep}')`));
210
- }, timeoutMs);
211
- }
212
- }).finally(() => {
213
- if (detach && this.proc) {
214
- this.proc.unref();
215
- }
216
- });
217
- }
218
-
219
- handleLastLines() {
220
- for (let stream of ['stdout', 'stderr']) {
221
- if (this.lastLinePortion[stream]) {
222
- const lastLines = [this.lastLinePortion[stream]];
223
- this.emit(`lines-${stream}`, lastLines);
224
- this.emitLines(stream, lastLines);
225
- this.lastLinePortion[stream] = '';
226
- }
227
226
  }
228
- }
229
-
230
- async stop(signal = 'SIGTERM', timeout = 10000) {
231
- if (!this.isRunning) {
232
- throw new Error(`Can't stop process; it's not currently running (cmd: '${this.rep}')`);
227
+ /**
228
+ *
229
+ * @param {NodeJS.Signals} signal
230
+ * @param {number} timeout
231
+ * @returns {Promise<void>}
232
+ */
233
+ async stop(signal = 'SIGTERM', timeout = 10000) {
234
+ if (!this.isRunning) {
235
+ throw new Error(`Can't stop process; it's not currently running (cmd: '${this.rep}')`);
236
+ }
237
+ // make sure to emit any data in our lines buffer whenever we're done with
238
+ // the proc
239
+ this.handleLastLines();
240
+ return await new bluebird_1.default((resolve, reject) => {
241
+ this.proc?.on('close', resolve);
242
+ this.expectingExit = true;
243
+ this.proc?.kill(signal);
244
+ // this timeout needs unref() or node will wait for the timeout to fire before
245
+ // exiting the process.
246
+ setTimeout(() => {
247
+ reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));
248
+ }, timeout).unref();
249
+ });
233
250
  }
234
-
235
- this.handleLastLines();
236
- return await new _bluebird.default((resolve, reject) => {
237
- var _this$proc4, _this$proc5;
238
-
239
- (_this$proc4 = this.proc) === null || _this$proc4 === void 0 ? void 0 : _this$proc4.on('close', resolve);
240
- this.expectingExit = true;
241
- (_this$proc5 = this.proc) === null || _this$proc5 === void 0 ? void 0 : _this$proc5.kill(signal);
242
- setTimeout(() => {
243
- reject(new Error(`Process didn't end after ${timeout}ms (cmd: '${this.rep}')`));
244
- }, timeout).unref();
245
- });
246
- }
247
-
248
- async join(allowedExitCodes = [0]) {
249
- if (!this.isRunning) {
250
- throw new Error(`Cannot join process; it is not currently running (cmd: '${this.rep}')`);
251
+ async join(allowedExitCodes = [0]) {
252
+ if (!this.isRunning) {
253
+ throw new Error(`Cannot join process; it is not currently running (cmd: '${this.rep}')`);
254
+ }
255
+ return await new bluebird_1.default((resolve, reject) => {
256
+ this.proc?.on('exit', (code) => {
257
+ if (code !== null && allowedExitCodes.indexOf(code) === -1) {
258
+ reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));
259
+ }
260
+ else {
261
+ resolve(code);
262
+ }
263
+ });
264
+ });
251
265
  }
252
-
253
- return await new _bluebird.default((resolve, reject) => {
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) {
258
- reject(new Error(`Process ended with exitcode ${code} (cmd: '${this.rep}')`));
259
- } else {
260
- resolve(code);
266
+ /*
267
+ * This will only work if the process is created with the `detached` option
268
+ */
269
+ detachProcess() {
270
+ if (!this.opts.detached) {
271
+ // this means that there is a misconfiguration in the calling code
272
+ throw new Error(`Unable to detach process that is not started with 'detached' option`);
273
+ }
274
+ if (this.proc) {
275
+ this.proc.unref();
261
276
  }
262
- });
263
- });
264
- }
265
-
266
- detachProcess() {
267
- if (!this.opts.detached) {
268
- throw new Error(`Unable to detach process that is not started with 'detached' option`);
269
277
  }
270
-
271
- if (this.proc) {
272
- this.proc.unref();
278
+ get pid() {
279
+ return this.proc ? this.proc.pid : null;
273
280
  }
274
- }
275
-
276
- get pid() {
277
- return this.proc ? this.proc.pid : null;
278
- }
279
-
280
281
  }
281
-
282
282
  exports.SubProcess = SubProcess;
283
- var _default = SubProcess;
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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"}
283
+ exports.default = SubProcess;
284
+ /**
285
+ * @callback StartDetector
286
+ * @param {string} stdout
287
+ * @param {string} [stderr]
288
+ * @returns {any}
289
+ */
290
+ //# sourceMappingURL=subprocess.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"subprocess.js","sourceRoot":"","sources":["../../lib/subprocess.js"],"names":[],"mappings":"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"}
package/index.js CHANGED
@@ -1 +1 @@
1
- module.exports = require('./build/lib/index.js');
1
+ module.exports = require('./build/lib');
package/lib/exec.js CHANGED
@@ -47,10 +47,9 @@ async function exec (cmd, args = [], opts = /** @type {T} */({})) {
47
47
  let stdoutArr = [], stderrArr = [], timer = null;
48
48
 
49
49
  // if the process errors out, reject the promise
50
- proc.on('error', /** @param {NodeJS.ErrnoException} err */(err) => {
51
- // @ts-ignore
52
- if (err.errno === 'ENOENT') {
53
- err = formatEnoent(err, cmd, opts.cwd);
50
+ proc.on('error', /** @param {NodeJS.ErrnoException} err */ async (err) => {
51
+ if (err.code === 'ENOENT') {
52
+ err = await formatEnoent(err, cmd, opts.cwd?.toString());
54
53
  }
55
54
  reject(err);
56
55
  });