diff --git a/doc/api/child_process.md b/doc/api/child_process.md index e79b411e91eacf..7c01e34cecb7b2 100644 --- a/doc/api/child_process.md +++ b/doc/api/child_process.md @@ -253,6 +253,8 @@ added: v0.5.0 piped to the parent, otherwise they will be inherited from the parent, see the `'pipe'` and `'inherit'` options for [`child_process.spawn()`][]'s [`stdio`][] for more details (Default: `false`) + * `stdio` {Array} Supports the array version of [`child_process.spawn()`][]'s + [`stdio`][] option. When this option is provided, it overrides `silent`. * `uid` {Number} Sets the user identity of the process. (See setuid(2).) * `gid` {Number} Sets the group identity of the process. (See setgid(2).) * Return: {ChildProcess} diff --git a/lib/child_process.js b/lib/child_process.js index ba8fe911f5c224..1df450e1aa4a28 100644 --- a/lib/child_process.js +++ b/lib/child_process.js @@ -44,10 +44,14 @@ exports.fork = function(modulePath /*, args, options*/) { args = execArgv.concat([modulePath], args); - // Leave stdin open for the IPC channel. stdout and stderr should be the - // same as the parent's if silent isn't set. - options.stdio = options.silent ? ['pipe', 'pipe', 'pipe', 'ipc'] : - [0, 1, 2, 'ipc']; + if (!Array.isArray(options.stdio)) { + // Leave stdin open for the IPC channel. stdout and stderr should be the + // same as the parent's if silent isn't set. + options.stdio = options.silent ? ['pipe', 'pipe', 'pipe', 'ipc'] : + [0, 1, 2, 'ipc']; + } else if (options.stdio.indexOf('ipc') === -1) { + throw new TypeError('Forked processes must have an IPC channel'); + } options.execPath = options.execPath || process.execPath; diff --git a/test/parallel/test-child-process-fork-stdio.js b/test/parallel/test-child-process-fork-stdio.js new file mode 100644 index 00000000000000..89a5ff98e5f8a8 --- /dev/null +++ b/test/parallel/test-child-process-fork-stdio.js @@ -0,0 +1,54 @@ +'use strict'; +const common = require('../common'); +const assert = require('assert'); +const cp = require('child_process'); +const net = require('net'); + +if (process.argv[2] === 'child') { + process.stdout.write('this should be ignored'); + process.stderr.write('this should not be ignored'); + + const pipe = new net.Socket({ fd: 4 }); + + process.on('disconnect', () => { + pipe.unref(); + }); + + pipe.setEncoding('utf8'); + pipe.on('data', (data) => { + process.send(data); + }); +} else { + assert.throws(() => { + cp.fork(__filename, {stdio: ['pipe', 'pipe', 'pipe', 'pipe']}); + }, /Forked processes must have an IPC channel/); + + let ipc = ''; + let stderr = ''; + const buf = Buffer.from('data to send via pipe'); + const child = cp.fork(__filename, ['child'], { + stdio: [0, 'ignore', 'pipe', 'ipc', 'pipe'] + }); + + assert.strictEqual(child.stdout, null); + + child.on('message', (msg) => { + ipc += msg; + + if (ipc === buf.toString()) { + child.disconnect(); + } + }); + + child.stderr.on('data', (chunk) => { + stderr += chunk; + }); + + child.on('exit', common.mustCall((code, signal) => { + assert.strictEqual(code, 0); + assert.strictEqual(signal, null); + assert.strictEqual(stderr, 'this should not be ignored'); + })); + + child.stdio[4].write(buf); +}