-
Notifications
You must be signed in to change notification settings - Fork 7.3k
Conversation
link to original discussion: stream-utils/raw-body#27 (comment) |
I agree the official documentation for "close" has never really been good. Officially it said "when the underlieing resource has been disposed of" but after reading all the classic streams code I formed an interpretation of the meaning of close not from the documentation but from the actual way that streams actually behaved. I see "close" as "this stream is totally ended, and will never emit another event" anything active about the stream has finished, flushed to disk, flushed to the kernel etc. Streams are an abstraction - not every stream has an underling resource - but that doesn't mean they shouldn't have a close event. The close event is still useful - but it's more useful if all streams have it. It sounds like @jonathanong has come to the same conclusion. There was a brief period in 0.8 where all core streams had |
@dominictarr, since node core cannot know exactly when a userland readable stream has finished its operations on its underlying resources, the 'close' event cannot always be emitted by node core. maybe we could require that 'close' should always be sent. This way it could be used as a cleanup-catch-all as @jonathanong seem to be using it. Do you see a need for optional reserved events related to the underlying ressources (like 'attached' or 'detached' ?) I think the semantic of 'close' is not clear because we have a kind of silent pre-roll, post-roll currently in streams2 :
The documentation seem to state that 'close' may be sent right after 6 ; you seem to agree with @jonathanong that 'close' should not be emitted before 8. in the hope that I am not just creating more noise, I add @isaacs in this thread maybe he can help us understand the role of the 'close' event since from what I remember he has worked very hard on 'stream2' at the time. |
close is usually more significant on a writable stream - when close is emitted on a fs writable then it means you are guaranteed to be able to open that file and read the same thing back out. emitting close should be the last thing that happens. I don't think there needs to be a standard for streams that happen to have many underling resources. not only is that rare, but if a user needs to know about that, they are already coupled to the particular type of stream. |
I think this would be self resolved by finishing up #6802. See, right now the |
@trevnorris I tend to agree that calling the callback in the correct phase of the event loop is a + (I have no idea of the impact such a move could have). nevertheless, finishing #6802 would not change the fact that we are not clear on the semantics of the 'close' event. If I read the current documentation, I understand that we could have a case where a descriptor is closed because no more data will be fecthed from it (and the 'close' callback is called). Some data could still be in internal buffer (because the downstream stream has reached its highWaterMark). Thus, in my understanding, the doc says it is ok to have 'close' before 'end'. This is what @dominictarr and @jonathanong say we should not have because 'close' should be the last event if we want it to be useful. @trevnorris do you agree with the fact that the 'close' event should be the last event ever emited by a stream (be it readable, writable or both) ? |
@jeromew That's not possible to guarantee with the new pull based Streams3 API. When incoming data was immediately fed to the user on the
So I disagree that
Can you please give an example scenario where a single stream is linked to multiple file descriptors? If that's possible then, IMO, it's a design flaw.
That's impossible to guarantee. Streams inherit from EventEmitter and can emit whenever and however they want. In fact, I could even hijack a TCP Stream and emit data from JS even though the data hasn't come over TCP. That's the double edge sword of an abstraction layer. But honestly, this is all just bikeshedding the name of events. For the sake of argument let's rename |
@trevnorris hmm, well that interpretation certainly worked well in classic streams. I think the problem here is that we are discussing this totally in abstract - such discussions normally get nowhere, and or wind up going down some weird path that you'd probably never actually use in practice. So - we should ask, what do we use the close event for, in practice. I can think of a few examples where I have used the close event. I also used the close event in the same way leveldb, writing to a stream, and waiting for the close event before reading again. Hence, the interpretation that 'close' means the stream is done. I am having trouble thinking of a situation where I have needed close on a readable stream. |
I can answer with both a theoric & concrete example. @trevnorris you ask what kind of stream could have 2 fd open. I can totally imagine a userland readable stream opening 2 files containing respectively data for left and right speakers ; ans mixin them. or a userland writable stream opening 2 files to separate audio from video in a video file. clearly your description corresponds to the current documentation for readable streams (by the way, the "close" event documentation is located in the Readable part) :
so in a Reading @dominictarr, the 'close' event seem to have also been used in userland for Writable streams. In the
@trevnorris do you agree that 'close' can be rightfully sent by a writable stream ? what happens with a duplex stream that writes to the input pin of a dsp and reads from the output pin ? should it send 2 events ? Now my practical issue from which all of this is coming is an incompatibility between the 2 modules
The problem comes when I use a
before I did a hack (0.0.5) to make sure 'close' is sent after 'end', In the case In my mind, According to @jonathanong 'close' should always be sent after 'end' and this is the way it goes in many modules. Does it makes things more clear on my initial pain point and my API-wise headache ? Here are a few questions for which the answer does not seem clear to me if you still have some courage (thanks a lot @trevnorris and @dominictarr for giving your very insightfull help on this)
|
@jeromew the most important part of my question is how do the clients of the streams interface with the close event? |
Closing this here. A new PR has been opened at http://github.com/nodejs/node |
per: nodejs/node-v0.x-archive#8209 originally submitted by @jeromew
per: nodejs/node-v0.x-archive#8209 originally submitted by @jeromew Reviewed By: Sakthipriyan Vairamani <thechargingvolcano@gmail.com> PR-URL: #2378
per: nodejs/node-v0.x-archive#8209 originally submitted by @jeromew Reviewed By: Sakthipriyan Vairamani <thechargingvolcano@gmail.com> PR-URL: #2378
After a discussion with @jonathanong and @dougwilson related to the raw-body module, we realized that maybe the documentation for the "close" event is not correct on node.js documentation.
This PR proposes to start a discussion on the meaning of the 'close' event or accept the definition given by @jonathanong which seem to be more in line with what is done in stream modules on npm.
The current definition seem to imply that the close event is only related to the closing of an underlying ressource, meaning that we could receive and 'end' event after the 'close' event if the underlying resource has been close but we still have some data in the internal buffers.
The current definition is also silent on the case when there are many underlying resources (for example many file descriptors). Should we have many 'close' event or just one ?
added to that, it seems that a dest.'close' event is also used to unpipe automatically a destination in /~https://github.com/joyent/node/blob/master/lib/_stream_readable.js#L564 which probably makes sense but seem to be far fetched if 'close' is only related to underlying ressources of the Readable part of a stream.
I don't know if they are all interested in this thread but their stream knowledge could be very beneficial if they can share their experience working with streams: @dominictarr, @substack, @chrisdickinson