From ec7dea9778ab879eb086ee19ec210e5807d8acd0 Mon Sep 17 00:00:00 2001 From: Robert Nagy Date: Sun, 26 Apr 2020 11:38:47 +0200 Subject: [PATCH] stream: don't emit end after close Readable stream could emit 'end' after 'close'. PR-URL: https://github.com/nodejs/node/pull/33076 Reviewed-By: Matteo Collina Reviewed-By: James M Snell Reviewed-By: Anna Henningsen --- lib/_stream_readable.js | 7 ++++++- lib/internal/streams/destroy.js | 6 +++++- test/parallel/test-stream-duplex-destroy.js | 2 +- test/parallel/test-stream-readable-destroy.js | 2 +- .../test-stream-readable-end-destroyed.js | 17 +++++++++++++++++ 5 files changed, 30 insertions(+), 4 deletions(-) create mode 100644 test/parallel/test-stream-readable-end-destroyed.js diff --git a/lib/_stream_readable.js b/lib/_stream_readable.js index 9c1441dd9ce405..ec9f86d7eacf4a 100644 --- a/lib/_stream_readable.js +++ b/lib/_stream_readable.js @@ -150,6 +150,10 @@ function ReadableState(options, stream, isDuplex) { // Indicates whether the stream has finished destroying. this.closed = false; + // True if close has been emitted or would have been emitted + // depending on emitClose. + this.closeEmitted = false; + // Crypto is kind of old and crusty. Historically, its default string // encoding is 'binary' so we have to make this configurable. // Everything else in the universe uses 'utf8', though. @@ -1213,7 +1217,8 @@ function endReadableNT(state, stream) { debug('endReadableNT', state.endEmitted, state.length); // Check that we didn't get one last unshift. - if (!state.errorEmitted && !state.endEmitted && state.length === 0) { + if (!state.errorEmitted && !state.closeEmitted && + !state.endEmitted && state.length === 0) { state.endEmitted = true; stream.emit('end'); diff --git a/lib/internal/streams/destroy.js b/lib/internal/streams/destroy.js index f17034c3b7aac5..ce476467a151dc 100644 --- a/lib/internal/streams/destroy.js +++ b/lib/internal/streams/destroy.js @@ -76,6 +76,9 @@ function emitCloseNT(self) { if (w) { w.closeEmitted = true; } + if (r) { + r.closeEmitted = true; + } if ((w && w.emitClose) || (r && r.emitClose)) { self.emit('close'); @@ -106,12 +109,13 @@ function undestroy() { if (r) { r.closed = false; + r.closeEmitted = false; r.destroyed = false; r.errored = false; + r.errorEmitted = false; r.reading = false; r.ended = false; r.endEmitted = false; - r.errorEmitted = false; } if (w) { diff --git a/test/parallel/test-stream-duplex-destroy.js b/test/parallel/test-stream-duplex-destroy.js index e7c91ec797beb3..c7d294e144bb5e 100644 --- a/test/parallel/test-stream-duplex-destroy.js +++ b/test/parallel/test-stream-duplex-destroy.js @@ -124,7 +124,7 @@ const assert = require('assert'); duplex.removeListener('end', fail); duplex.removeListener('finish', fail); - duplex.on('end', common.mustCall()); + duplex.on('end', common.mustNotCall()); duplex.on('finish', common.mustCall()); assert.strictEqual(duplex.destroyed, true); } diff --git a/test/parallel/test-stream-readable-destroy.js b/test/parallel/test-stream-readable-destroy.js index 6caf88a1f151e8..3d1ac8c92f9bd3 100644 --- a/test/parallel/test-stream-readable-destroy.js +++ b/test/parallel/test-stream-readable-destroy.js @@ -113,7 +113,7 @@ const assert = require('assert'); read.destroy(); read.removeListener('end', fail); - read.on('end', common.mustCall()); + read.on('end', common.mustNotCall()); assert.strictEqual(read.destroyed, true); } diff --git a/test/parallel/test-stream-readable-end-destroyed.js b/test/parallel/test-stream-readable-end-destroyed.js new file mode 100644 index 00000000000000..4b60bf4614770a --- /dev/null +++ b/test/parallel/test-stream-readable-end-destroyed.js @@ -0,0 +1,17 @@ +'use strict'; + +const common = require('../common'); +const { Readable } = require('stream'); + +{ + // Don't emit 'end' after 'close'. + + const r = new Readable(); + + r.on('end', common.mustNotCall()); + r.resume(); + r.destroy(); + r.on('close', common.mustCall(() => { + r.push(null); + })); +}