From 954217adda03457115e426e9126dcad845877f74 Mon Sep 17 00:00:00 2001 From: Robert Nagy Date: Wed, 15 Jul 2020 21:44:10 +0200 Subject: [PATCH] stream: error Duplex write/read if not writable/readable If writable/readable has been explicitly disabled then using a Duplex as writable/readable should fail. Fixes: https://github.com/nodejs/node/issues/34374 PR-URL: https://github.com/nodejs/node/pull/34385 Reviewed-By: James M Snell Reviewed-By: Matteo Collina Reviewed-By: Benjamin Gruenbaum Reviewed-By: Anna Henningsen --- .../bootstrap/switches/is_main_thread.js | 6 +-- lib/internal/child_process.js | 4 +- lib/internal/http2/core.js | 14 ++++-- lib/internal/streams/destroy.js | 10 ++-- lib/internal/streams/duplex.js | 20 ++++---- lib/tty.js | 11 ++--- src/tty_wrap.cc | 7 ++- src/tty_wrap.h | 1 - test/parallel/test-http2-compat-socket-set.js | 4 +- test/parallel/test-stdio-readable-writable.js | 33 ------------- .../test-stream-duplex-readable-writable.js | 46 +++++++++++++++++++ test/pseudo-tty/repl-dumb-tty.js | 20 ++++---- test/pseudo-tty/repl-dumb-tty.out | 4 +- 13 files changed, 97 insertions(+), 83 deletions(-) delete mode 100644 test/parallel/test-stdio-readable-writable.js create mode 100644 test/parallel/test-stream-duplex-readable-writable.js diff --git a/lib/internal/bootstrap/switches/is_main_thread.js b/lib/internal/bootstrap/switches/is_main_thread.js index 08623898edafac..01b2d5bd286071 100644 --- a/lib/internal/bootstrap/switches/is_main_thread.js +++ b/lib/internal/bootstrap/switches/is_main_thread.js @@ -148,11 +148,7 @@ function getStdin() { switch (guessHandleType(fd)) { case 'TTY': const tty = require('tty'); - stdin = new tty.ReadStream(fd, { - highWaterMark: 0, - readable: true, - writable: false - }); + stdin = new tty.ReadStream(fd); break; case 'FILE': diff --git a/lib/internal/child_process.js b/lib/internal/child_process.js index 7ff0832538f1e7..d0a4b2f6ef666d 100644 --- a/lib/internal/child_process.js +++ b/lib/internal/child_process.js @@ -324,7 +324,7 @@ function flushStdio(subprocess) { function createSocket(pipe, readable) { - return net.Socket({ handle: pipe, readable, writable: !readable }); + return net.Socket({ handle: pipe, readable }); } @@ -442,8 +442,6 @@ ChildProcess.prototype.spawn = function(options) { } if (stream.handle) { - // When i === 0 - we're dealing with stdin - // (which is the only one writable pipe). stream.socket = createSocket(this.pid !== 0 ? stream.handle : null, i > 0); diff --git a/lib/internal/http2/core.js b/lib/internal/http2/core.js index 1654f2460cdd5a..473c1244f75da7 100644 --- a/lib/internal/http2/core.js +++ b/lib/internal/http2/core.js @@ -358,10 +358,12 @@ function onSessionHeaders(handle, id, cat, flags, headers, sensitiveHeaders) { handle.destroy(); return; } - const opts = { readable: !endOfStream }; // session[kType] can be only one of two possible values if (type === NGHTTP2_SESSION_SERVER) { - stream = new ServerHttp2Stream(session, handle, id, opts, obj); + stream = new ServerHttp2Stream(session, handle, id, {}, obj); + if (endOfStream) { + stream.push(null); + } if (obj[HTTP2_HEADER_METHOD] === HTTP2_METHOD_HEAD) { // For head requests, there must not be a body... // end the writable side immediately. @@ -369,7 +371,10 @@ function onSessionHeaders(handle, id, cat, flags, headers, sensitiveHeaders) { stream[kState].flags |= STREAM_FLAGS_HEAD_REQUEST; } } else { - stream = new ClientHttp2Stream(session, handle, id, opts); + stream = new ClientHttp2Stream(session, handle, id, {}); + if (endOfStream) { + stream.push(null); + } stream.end(); } if (endOfStream) @@ -2675,7 +2680,6 @@ class ServerHttp2Stream extends Http2Stream { let headRequest = false; if (headers[HTTP2_HEADER_METHOD] === HTTP2_METHOD_HEAD) headRequest = options.endStream = true; - options.readable = false; const headersList = mapToHeaders(headers); @@ -2703,6 +2707,8 @@ class ServerHttp2Stream extends Http2Stream { const stream = new ServerHttp2Stream(session, ret, id, options, headers); stream[kSentHeaders] = headers; + stream.push(null); + if (options.endStream) stream.end(); diff --git a/lib/internal/streams/destroy.js b/lib/internal/streams/destroy.js index 4238e12074d769..df2d9f7f71987a 100644 --- a/lib/internal/streams/destroy.js +++ b/lib/internal/streams/destroy.js @@ -206,8 +206,8 @@ function undestroy() { r.errored = null; r.errorEmitted = false; r.reading = false; - r.ended = false; - r.endEmitted = false; + r.ended = r.readable === false; + r.endEmitted = r.readable === false; } if (w) { @@ -217,11 +217,11 @@ function undestroy() { w.closeEmitted = false; w.errored = null; w.errorEmitted = false; - w.ended = false; - w.ending = false; w.finalCalled = false; w.prefinished = false; - w.finished = false; + w.ended = w.writable === false; + w.ending = w.writable === false; + w.finished = w.writable === false; } } diff --git a/lib/internal/streams/duplex.js b/lib/internal/streams/duplex.js index 596d30a90a0418..3d65c15f201dd7 100644 --- a/lib/internal/streams/duplex.js +++ b/lib/internal/streams/duplex.js @@ -55,18 +55,20 @@ function Duplex(options) { Readable.call(this, options); Writable.call(this, options); - this.allowHalfOpen = true; - if (options) { - if (options.readable === false) - this.readable = false; + this.allowHalfOpen = options?.allowHalfOpen !== false; - if (options.writable === false) - this.writable = false; + if (options?.readable === false) { + this._readableState.readable = false; + this._readableState.ended = true; + this._readableState.endEmitted = true; + } - if (options.allowHalfOpen === false) { - this.allowHalfOpen = false; - } + if (options?.writable === false) { + this._writableState.writable = false; + this._writableState.ending = true; + this._writableState.ended = true; + this._writableState.finished = true; } } diff --git a/lib/tty.js b/lib/tty.js index e61a5c3ac3f905..d9c576b5fd6c77 100644 --- a/lib/tty.js +++ b/lib/tty.js @@ -51,16 +51,15 @@ function ReadStream(fd, options) { throw new ERR_INVALID_FD(fd); const ctx = {}; - const tty = new TTY(fd, true, ctx); + const tty = new TTY(fd, ctx); if (ctx.code !== undefined) { throw new ERR_TTY_INIT_FAILED(ctx); } net.Socket.call(this, { highWaterMark: 0, - readable: true, - writable: false, handle: tty, + manualStart: true, ...options }); @@ -89,15 +88,15 @@ function WriteStream(fd) { throw new ERR_INVALID_FD(fd); const ctx = {}; - const tty = new TTY(fd, false, ctx); + const tty = new TTY(fd, ctx); if (ctx.code !== undefined) { throw new ERR_TTY_INIT_FAILED(ctx); } net.Socket.call(this, { + highWaterMark: 0, handle: tty, - readable: false, - writable: true + manualStart: true }); // Prevents interleaved or dropped stdout/stderr output for terminals. diff --git a/src/tty_wrap.cc b/src/tty_wrap.cc index 401c2513dbc628..e9c7eb37a6621d 100644 --- a/src/tty_wrap.cc +++ b/src/tty_wrap.cc @@ -121,9 +121,9 @@ void TTYWrap::New(const FunctionCallbackInfo& args) { CHECK_GE(fd, 0); int err = 0; - new TTYWrap(env, args.This(), fd, args[1]->IsTrue(), &err); + new TTYWrap(env, args.This(), fd, &err); if (err != 0) { - env->CollectUVExceptionInfo(args[2], err, "uv_tty_init"); + env->CollectUVExceptionInfo(args[1], err, "uv_tty_init"); args.GetReturnValue().SetUndefined(); } } @@ -132,13 +132,12 @@ void TTYWrap::New(const FunctionCallbackInfo& args) { TTYWrap::TTYWrap(Environment* env, Local object, int fd, - bool readable, int* init_err) : LibuvStreamWrap(env, object, reinterpret_cast(&handle_), AsyncWrap::PROVIDER_TTYWRAP) { - *init_err = uv_tty_init(env->event_loop(), &handle_, fd, readable); + *init_err = uv_tty_init(env->event_loop(), &handle_, fd, 0); set_fd(fd); if (*init_err != 0) MarkAsUninitialized(); diff --git a/src/tty_wrap.h b/src/tty_wrap.h index fdf07e4242c1f8..84e1fbb74d3384 100644 --- a/src/tty_wrap.h +++ b/src/tty_wrap.h @@ -46,7 +46,6 @@ class TTYWrap : public LibuvStreamWrap { TTYWrap(Environment* env, v8::Local object, int fd, - bool readable, int* init_err); static void IsTTY(const v8::FunctionCallbackInfo& args); diff --git a/test/parallel/test-http2-compat-socket-set.js b/test/parallel/test-http2-compat-socket-set.js index b14a2f65d8e7fd..30bddc6df59030 100644 --- a/test/parallel/test-http2-compat-socket-set.js +++ b/test/parallel/test-http2-compat-socket-set.js @@ -26,9 +26,9 @@ server.on('request', common.mustCall(function(request, response) { assert.strictEqual(request.stream.destroyed, true); request.socket.destroyed = false; - assert.strictEqual(request.stream.readable, false); - request.socket.readable = true; assert.strictEqual(request.stream.readable, true); + request.socket.readable = false; + assert.strictEqual(request.stream.readable, false); assert.strictEqual(request.stream.writable, true); request.socket.writable = false; diff --git a/test/parallel/test-stdio-readable-writable.js b/test/parallel/test-stdio-readable-writable.js deleted file mode 100644 index 6178313bca3d3b..00000000000000 --- a/test/parallel/test-stdio-readable-writable.js +++ /dev/null @@ -1,33 +0,0 @@ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. - -'use strict'; -require('../common'); -const assert = require('assert'); - -assert(process.stdout.writable); -assert(!process.stdout.readable); - -assert(process.stderr.writable); -assert(!process.stderr.readable); - -assert(!process.stdin.writable); -assert(process.stdin.readable); diff --git a/test/parallel/test-stream-duplex-readable-writable.js b/test/parallel/test-stream-duplex-readable-writable.js new file mode 100644 index 00000000000000..aec88fc120b1c5 --- /dev/null +++ b/test/parallel/test-stream-duplex-readable-writable.js @@ -0,0 +1,46 @@ +'use strict'; + +const common = require('../common'); +const { Duplex } = require('stream'); +const assert = require('assert'); + +{ + const duplex = new Duplex({ + readable: false + }); + assert.strictEqual(duplex.readable, false); + duplex.push('asd'); + duplex.on('error', common.mustCall((err) => { + assert.strictEqual(err.code, 'ERR_STREAM_PUSH_AFTER_EOF'); + })); + duplex.on('data', common.mustNotCall()); + duplex.on('end', common.mustNotCall()); +} + +{ + const duplex = new Duplex({ + writable: false, + write: common.mustNotCall() + }); + assert.strictEqual(duplex.writable, false); + duplex.write('asd'); + duplex.on('error', common.mustCall((err) => { + assert.strictEqual(err.code, 'ERR_STREAM_WRITE_AFTER_END'); + })); + duplex.on('finish', common.mustNotCall()); +} + +{ + const duplex = new Duplex({ + readable: false + }); + assert.strictEqual(duplex.readable, false); + duplex.on('data', common.mustNotCall()); + duplex.on('end', common.mustNotCall()); + async function run() { + for await (const chunk of duplex) { + assert(false, chunk); + } + } + run().then(common.mustCall()); +} diff --git a/test/pseudo-tty/repl-dumb-tty.js b/test/pseudo-tty/repl-dumb-tty.js index 8c9b93a9f31ccd..7fb96457151a4e 100644 --- a/test/pseudo-tty/repl-dumb-tty.js +++ b/test/pseudo-tty/repl-dumb-tty.js @@ -1,5 +1,6 @@ 'use strict'; const common = require('../common'); +const process = require('process'); process.env.TERM = 'dumb'; @@ -7,15 +8,6 @@ const repl = require('repl'); const ArrayStream = require('../common/arraystream'); repl.start('> '); -process.stdin.push('conso'); // No completion preview. -process.stdin.push('le.log("foo")\n'); -process.stdin.push('1 + 2'); // No input preview. -process.stdin.push('\n'); -process.stdin.push('"str"\n'); -process.stdin.push('console.dir({ a: 1 })\n'); -process.stdin.push('{ a: 1 }\n'); -process.stdin.push('\n'); -process.stdin.push('.exit\n'); // Verify + D support. { @@ -34,3 +26,13 @@ process.stdin.push('.exit\n'); replServer.write(null, { ctrl: true, name: 's' }); replServer.write(null, { ctrl: true, name: 'd' }); } + +process.stdin.push('conso'); // No completion preview. +process.stdin.push('le.log("foo")\n'); +process.stdin.push('1 + 2'); // No input preview. +process.stdin.push('\n'); +process.stdin.push('"str"\n'); +process.stdin.push('console.dir({ a: 1 })\n'); +process.stdin.push('{ a: 1 }\n'); +process.stdin.push('\n'); +process.stdin.push('.exit\n'); diff --git a/test/pseudo-tty/repl-dumb-tty.out b/test/pseudo-tty/repl-dumb-tty.out index 3304faff0a4f4f..590267e9264b3f 100644 --- a/test/pseudo-tty/repl-dumb-tty.out +++ b/test/pseudo-tty/repl-dumb-tty.out @@ -1,4 +1,5 @@ -> console.log("foo") +> > +console.log("foo") foo undefined > 1 + 2 @@ -12,4 +13,3 @@ undefined { a: 1 } > > .exit ->