From 701112a9709ca1edb24f34a2846a3188ebdb104d Mon Sep 17 00:00:00 2001 From: Robert Nagy Date: Tue, 3 Aug 2021 09:01:31 +0200 Subject: [PATCH] stream: don't emit 'data' after 'error' or 'close' As per doc we should not emit further events after 'close' and only 'close' after 'error'. --- lib/internal/streams/readable.js | 5 +- test/parallel/test-stream-readable-destroy.js | 83 +++++++++++++++++++ 2 files changed, 87 insertions(+), 1 deletion(-) diff --git a/lib/internal/streams/readable.js b/lib/internal/streams/readable.js index a6a1913961bb41..4dd2117546862a 100644 --- a/lib/internal/streams/readable.js +++ b/lib/internal/streams/readable.js @@ -276,6 +276,8 @@ function readableAddChunk(stream, chunk, encoding, addToFront) { if (addToFront) { if (state.endEmitted) errorOrDestroy(stream, new ERR_STREAM_UNSHIFT_AFTER_END_EVENT()); + else if (state.destroyed || state.errored) + return false; else addChunk(stream, state, chunk, true); } else if (state.ended) { @@ -316,6 +318,7 @@ function addChunk(stream, state, chunk, addToFront) { } else { state.awaitDrainWriters = null; } + state.dataEmitted = true; stream.emit('data', chunk); } else { @@ -542,7 +545,7 @@ Readable.prototype.read = function(n) { endReadable(this); } - if (ret !== null) { + if (ret !== null && !state.errorEmitted && !state.closeEmitted) { state.dataEmitted = true; this.emit('data', ret); } diff --git a/test/parallel/test-stream-readable-destroy.js b/test/parallel/test-stream-readable-destroy.js index 9ba3f9cd3653d6..759375c7ca81e7 100644 --- a/test/parallel/test-stream-readable-destroy.js +++ b/test/parallel/test-stream-readable-destroy.js @@ -319,3 +319,86 @@ const assert = require('assert'); })(), /AbortError/); setTimeout(() => controller.abort(), 0); } + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.on('error', common.mustCall((e) => { + read.read(); + })); + read.on('close', common.mustCall((e) => { + read.read(); + })); + read.destroy(new Error('asd')); +} + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.on('close', common.mustCall((e) => { + read.read(); + })); + read.destroy(); +} + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.on('close', common.mustCall((e) => { + read.unshift('asd'); + })); + read.destroy(); +} + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.destroy(); + read.unshift('asd'); +} + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.on('close', common.mustCall((e) => { + read.push('asd'); + })); + read.destroy(); +} + +{ + const read = new Readable({ + read() { + }, + }); + read.push('asd'); + + read.on('data', common.mustNotCall()); + read.destroy(); + read.push('asd'); +}