Skip to content

Commit ed91703

Browse files
committedNov 22, 2019
fix: ubuntu loop waiting for sub processes
Fixes #1633 Multiple parts but all localised to run.js: - Refactor the kill logic to be simpler - Consistently use pstree to determine sub process list - Pipe stderr through nodemon to squash `sh -c` error warning Bug was caused by waiting on multiple sub processes and if they all ended the logic would only subtract one from the count list (rather than the total number). I've refactored the code so that it doesn't use the `kill -0 <pid>` as this was a little confusing to read (it's effectively a no op) and switched to using pstree to test if any sub processes are still running. The logic for killing the processes has also been refactored to simplify. Before it would fork the logic based on whether `ps` existed on the system. Now it uses the same logic with the exception of the kill signal sent - when `ps` isn't on the system, we have to send numeric signals (I can't remember how I found that out, but I do remember it was a painful process!). The last part required due to a side effect of the refactor on kill: when a kill signial is sent to `sh -c` the shell prints a warning. Details on how to replicate: https://git.io/Je6d0 To squash this, I track if the process is about to be killed (by flagging the sub process right before the kill function call) and if there's an error whilst shutdown is in effect, the error is only printed to nodemon's detailed output (using nodemon -V).
1 parent 9a67f36 commit ed91703

File tree

2 files changed

+51
-46
lines changed

2 files changed

+51
-46
lines changed
 

‎LICENSE

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
MIT License
22

3-
Copyright (c) 2019 Remy Sharp, https://remysharp.com <remy@remysharp.com>
3+
Copyright (c) 2010 - present, Remy Sharp, https://remysharp.com <remy@remysharp.com>
44

55
Permission is hereby granted, free of charge, to any person obtaining a copy
66
of this software and associated documentation files (the "Software"), to deal

‎lib/monitor/run.js

+50-45
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
var debug = require('debug')('nodemon');
1+
var debug = require('debug')('nodemon:run');
22
const statSync = require('fs').statSync;
33
var utils = require('../utils');
44
var bus = utils.bus;
@@ -10,34 +10,12 @@ var watch = require('./watch').watch;
1010
var config = require('../config');
1111
var child = null; // the actual child process we spawn
1212
var killedAfterChange = false;
13-
var noop = function () { };
13+
var noop = () => {};
1414
var restart = null;
1515
var psTree = require('pstree.remy');
1616
var path = require('path');
1717
var signals = require('./signals');
1818

19-
function waitForSubprocesses(subprocesses, callback) {
20-
if (Array.isArray(subprocesses) && subprocesses.length > 0) {
21-
// check if all old subprocesses have been terminated
22-
exec('kill -0 ' + subprocesses.join(' '), (error) => {
23-
const returnCode = error ? error.code : 0;
24-
if (returnCode < 126) { // ignore command not found error
25-
const stillRunning = subprocesses.length - returnCode;
26-
if (stillRunning > 0) {
27-
utils.log.status('still waiting for ' + stillRunning +
28-
' subprocess(es) to finish...');
29-
setTimeout(waitForSubprocesses.bind(this, subprocesses, callback),
30-
100);
31-
return;
32-
}
33-
}
34-
callback();
35-
});
36-
} else {
37-
callback();
38-
}
39-
}
40-
4119
function run(options) {
4220
var cmd = config.command.raw;
4321

@@ -55,11 +33,11 @@ function run(options) {
5533
var stdio = ['pipe', 'pipe', 'pipe'];
5634

5735
if (config.options.stdout) {
58-
stdio = ['pipe', process.stdout, process.stderr];
36+
stdio = ['pipe', process.stdout, 'pipe'];
5937
}
6038

6139
if (config.options.stdin === false) {
62-
stdio = [process.stdin, process.stdout, process.stderr];
40+
stdio = [process.stdin, process.stdout, 'pipe'];
6341
}
6442

6543
var sh = 'sh';
@@ -161,6 +139,20 @@ function run(options) {
161139
bus.emit('message', message, sendHandle);
162140
});
163141
}
142+
} else { // else if not required…
143+
144+
// this swallows a shell message that happens because we kill the sh -c
145+
// more details here: https://git.io/Je6d0
146+
child.stderr.on('data', s => {
147+
s = s.toString();
148+
149+
if (child.__nodemonRestart) { // this flag is set right before the kill
150+
utils.log.detail('stderr: ' + s);
151+
return;
152+
}
153+
154+
process.stderr.write(s);
155+
});
164156
}
165157

166158
bus.emit('start');
@@ -284,6 +276,7 @@ function run(options) {
284276
/* Now kill the entire subtree of processes belonging to nodemon */
285277
var oldPid = child.pid;
286278
if (child) {
279+
child.__nodemonRestart = true;
287280
kill(child, config.signal, function () {
288281
// this seems to fix the 0.11.x issue with the "rs" restart command,
289282
// though I'm unsure why. it seems like more data is streamed in to
@@ -340,6 +333,19 @@ function run(options) {
340333
}
341334
}
342335

336+
function waitForSubProcesses(pid, callback) {
337+
debug('checking ps tree for pids of ' + pid);
338+
psTree(pid, (err, pids) => {
339+
if (!pids.length) {
340+
return callback();
341+
}
342+
343+
utils.log.status(`still waiting for ${pids.length} sub-process${
344+
pids.length > 2 ? 'es' : ''} to finish...`);
345+
setTimeout(() => waitForSubProcesses(pid, callback), 1000);
346+
});
347+
}
348+
343349
function kill(child, signal, callback) {
344350
if (!callback) {
345351
callback = function () { };
@@ -362,26 +368,25 @@ function kill(child, signal, callback) {
362368
// an array of PIDs that have spawned under nodemon, and we send each the
363369
// configured signal (default: SIGUSR2) signal, which fixes #335
364370
// note that psTree also works if `ps` is missing by looking in /proc
365-
const sig = signal.replace('SIG', '');
366-
psTree(child.pid, function (err, subprocesses) {
367-
if (psTree.hasPS) {
368-
spawn('kill', ['-s', sig].concat(subprocesses))
369-
.on('close', waitForSubprocesses.bind(this, subprocesses,
370-
function () {
371-
spawn('kill', ['-s', sig, child.pid])
372-
.on('close', callback);
373-
}));
374-
} else {
375-
// make sure we kill from smallest to largest
376-
const pids = subprocesses.slice().sort();
377-
pids.forEach(pid => {
378-
exec('kill -' + signals[signal] + ' ' + pid, () => { });
379-
});
380-
waitForSubprocesses(subprocesses, function () {
381-
exec('kill -' + signals[signal] + ' ' + child.pid, () => { });
382-
callback();
383-
})
371+
let sig = signal.replace('SIG', '');
372+
373+
psTree(child.pid, function (err, pids) {
374+
// if ps isn't native to the OS, then we need to send the numeric value
375+
// for the signal during the kill, `signals` is a lookup table for that.
376+
if (!psTree.hasPS) {
377+
sig = signals[signal];
384378
}
379+
380+
// the sub processes need to be killed from smallest to largest
381+
debug('sending kill signal to ' + pids.join(', '));
382+
383+
pids.sort().forEach(pid => exec(`kill -${sig} ${pid}`, noop));
384+
385+
waitForSubProcesses(child.pid, () => {
386+
// finally kill the main user process
387+
exec(`kill -${sig} ${child.pid}`, callback);
388+
});
389+
385390
});
386391

387392
}

0 commit comments

Comments
 (0)
Please sign in to comment.