-
Notifications
You must be signed in to change notification settings - Fork 29.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Race condition when ending a stream #40471
Comments
From my studying of the node code, I'm starting to think that the behavior of "waiting forever" is the correct one -- and therefore that v14.x and v16.x fixed issues. Let me explain (because I find that the documentation isn't always good enough at this level of details). This is my understanding and may contain also imprecisions.
Therefore, what happens in the case above:
I now think the faulty behaviors in v12.x and v14.x were due to a race or several races around the use of
I couldn't pinpoint which different code produces these different behaviors yet. |
This is more puzzling. I tried simplifying my testcase, removing all the piping. This is the console.log("START TEST");
const fixture = "WRITE SOMETHING";
const checker = new MyStream();
console.log("WRITE 1");
checker.write(fixture.slice(0, 3));
await nextTick();
console.log("WRITE 2");
checker.end(fixture.slice(3));
console.log("WAIT FOR PIPELINE END");
await finished(checker);
console.log("FINISHED"); Now, v12 gives me the error I expect: |
I fixed my issue by removing the call to |
Version
v14.17.5 and v16.11.1
Platform
Linux torri 5.13.14 #29 SMP Mon Sep 6 12:28:17 CEST 2021 x86_64 GNU/Linux
Subsystem
stream
What steps will reproduce the bug?
Clone https://github.com/julienw/streams-problems
Run the script
node-streams.js
with node v14 or node v16.Here is the content of this script:
(This needs to be run with
type:"module"
so it's probably easier to clone the repository)In v12, we get this output:
But in v14 and v16 we get this:
And this never ends.
As you see, we're calling
this.end()
in_write()
after a nextTick. The original idea was that this stream's purpose was finished (checking something) and callingend
would unpipe, and we wouldn't get more writes. The goal is to stop doing more work.How often does it reproduce? Is there a required condition?
Always
What is the expected behavior?
I think we should get one of these results:
end
([ERR_STREAM_WRITE_AFTER_END]: write after end
).What do you see instead?
Here it looks like that the pipeline isn't unpiped AND we don't get an error. So the written data isn't consumed by the stream that's been ended, and we're waiting forever.
Additional information
Replacing
process.nextTick(() => this.end());
with something likeawait nextTick(); this.end();
gets an error (behavior 2 above),which is different than node v12but at least we're not waiting forever and we get some clue. Update Oct 18: we get the same error in v12 and v14, but in v16 we get the behavior of waiting forever.Any insight will be very much appreciated. Especially can you point to what changed in v14 in this topic compared to v12 (I do know that a lot of the streams code changed in-between)? Also it seems to be than waiting forever isn't the right behavior.
Thanks!
The text was updated successfully, but these errors were encountered: