Skip to content

Commit 0875837

Browse files
committed
stream: fix async iterator destroyed error order
There was an edge case where if _destroy calls the error callback later than one tick the iterator would complete early and not propgate the error. PR-URL: #31314 Reviewed-By: Matteo Collina <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]> Reviewed-By: Rich Trott <[email protected]> Reviewed-By: Minwoo Jung <[email protected]> Reviewed-By: James M Snell <[email protected]> PR-URL: #31700 Reviewed-By: Anna Henningsen <[email protected]>
1 parent 9a1c19b commit 0875837

File tree

2 files changed

+30
-11
lines changed

2 files changed

+30
-11
lines changed

lib/internal/streams/async_iterator.js

+13-11
Original file line numberDiff line numberDiff line change
@@ -96,18 +96,20 @@ const ReadableStreamAsyncIteratorPrototype = ObjectSetPrototypeOf({
9696
}
9797

9898
if (this[kStream].destroyed) {
99-
// We need to defer via nextTick because if .destroy(err) is
100-
// called, the error will be emitted via nextTick, and
101-
// we cannot guarantee that there is no error lingering around
102-
// waiting to be emitted.
10399
return new Promise((resolve, reject) => {
104-
process.nextTick(() => {
105-
if (this[kError]) {
106-
reject(this[kError]);
107-
} else {
108-
resolve(createIterResult(undefined, true));
109-
}
110-
});
100+
if (this[kError]) {
101+
reject(this[kError]);
102+
} else if (this[kEnded]) {
103+
resolve(createIterResult(undefined, true));
104+
} else {
105+
finished(this[kStream], (err) => {
106+
if (err && err.code !== 'ERR_STREAM_PREMATURE_CLOSE') {
107+
reject(err);
108+
} else {
109+
resolve(createIterResult(undefined, true));
110+
}
111+
});
112+
}
111113
});
112114
}
113115

test/parallel/test-stream-readable-async-iterators.js

+17
Original file line numberDiff line numberDiff line change
@@ -567,6 +567,23 @@ async function tests() {
567567
assert.strictEqual(e, err);
568568
})()]);
569569
}
570+
571+
{
572+
const _err = new Error('asd');
573+
const r = new Readable({
574+
read() {
575+
},
576+
destroy(err, callback) {
577+
setTimeout(() => callback(_err), 1);
578+
}
579+
});
580+
581+
r.destroy();
582+
const it = r[Symbol.asyncIterator]();
583+
it.next().catch(common.mustCall((err) => {
584+
assert.strictEqual(err, _err);
585+
}));
586+
}
570587
}
571588

572589
{

0 commit comments

Comments
 (0)