diff --git a/doc/api/errors.md b/doc/api/errors.md index e4a709d45a8031..72e61c49bb970a 100644 --- a/doc/api/errors.md +++ b/doc/api/errors.md @@ -1694,6 +1694,12 @@ A `Transform` stream finished with data still in the write buffer. The initialization of a TTY failed due to a system error. + +### ERR_TTY_WRITABLE_NOT_READABLE + +This `Error` is thrown when a read is attempted on a TTY `WriteStream`, +such as `process.stdout.on('data')`. + ### ERR_UNCAUGHT_EXCEPTION_CAPTURE_ALREADY_SET diff --git a/lib/internal/errors.js b/lib/internal/errors.js index 89b51a4cecc9d8..0520fbbc6f6668 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -828,6 +828,9 @@ E('ERR_TRANSFORM_ALREADY_TRANSFORMING', E('ERR_TRANSFORM_WITH_LENGTH_0', 'Calling transform done when writableState.length != 0', Error); E('ERR_TTY_INIT_FAILED', 'TTY initialization failed', SystemError); +E('ERR_TTY_WRITABLE_NOT_READABLE', + 'The Writable side of a TTY is not Readable', + Error); E('ERR_UNCAUGHT_EXCEPTION_CAPTURE_ALREADY_SET', '`process.setupUncaughtExceptionCapture()` was called while a capture ' + 'callback was already active', diff --git a/lib/tty.js b/lib/tty.js index b36adf2534c326..5e0e528c7e4593 100644 --- a/lib/tty.js +++ b/lib/tty.js @@ -25,7 +25,11 @@ const { inherits, _extend } = require('util'); const net = require('net'); const { TTY, isTTY } = process.binding('tty_wrap'); const errors = require('internal/errors'); -const { ERR_INVALID_FD, ERR_TTY_INIT_FAILED } = errors.codes; +const { + ERR_INVALID_FD, + ERR_TTY_INIT_FAILED, + ERR_TTY_WRITABLE_NOT_READABLE +} = errors.codes; const { getColorDepth } = require('internal/tty'); // Lazy loaded for startup performance. @@ -122,6 +126,13 @@ WriteStream.prototype._refreshSize = function() { } }; +// A WriteStream is not readable from, so _read become a no-op. +// this method could still be called because net.Socket() +// is a duplex +WriteStream.prototype._read = function() { + this.destroy(new ERR_TTY_WRITABLE_NOT_READABLE()); +}; + // Backwards-compat WriteStream.prototype.cursorTo = function(x, y) { if (readline === undefined) readline = require('readline'); diff --git a/test/pseudo-tty/test-tty-write-stream-resume-crash.js b/test/pseudo-tty/test-tty-write-stream-resume-crash.js new file mode 100644 index 00000000000000..12ec1df45ad598 --- /dev/null +++ b/test/pseudo-tty/test-tty-write-stream-resume-crash.js @@ -0,0 +1,17 @@ +'use strict'; + +const common = require('../common'); +const { WriteStream } = require('tty'); +const fd = common.getTTYfd(); + +// Calling resume on a tty.WriteStream should be a no-op +// Ref: /~https://github.com/nodejs/node/issues/21203 + +const stream = new WriteStream(fd); +stream.resume(); + +stream.on('error', common.expectsError({ + code: 'ERR_TTY_WRITABLE_NOT_READABLE', + type: Error, + message: 'The Writable side of a TTY is not Readable' +})); diff --git a/test/pseudo-tty/test-tty-write-stream-resume-crash.out b/test/pseudo-tty/test-tty-write-stream-resume-crash.out new file mode 100644 index 00000000000000..e69de29bb2d1d6