events: add once method to use promises with EventEmitter by mcollina · Pull Request #26078 · nodejs/node (original) (raw)
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 andprivacy statement. We’ll occasionally send you account related emails.
Already on GitHub?Sign in to your account
Conversation69 Commits1 Checks0 Files changed
Conversation
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.Learn more about bidirectional Unicode characters
[ Show hidden characters]({{ revealButtonHref }})
This change adds a EventEmitter.once() method that wraps ee.once in a
promise. I've been using this model for some months now, and it works extremely well for me.
const { once, EventEmitter } = require('events'); async function run() { const ee = new EventEmitter(); process.nextTick(() => { ee.emit('myevent', 42); }); const [value] = await once(ee, 'myevent'); console.log(value); const err = new Error('kaboom'); process.nextTick(() => { ee.emit('error', err); }); try { await once(ee, 'myevent'); } catch (err) { console.log('error happened', err); } } run();
Checklist
make -j4 test
(UNIX), orvcbuild test
(Windows) passes- tests and/or benchmarks are included
- documentation is changed or added
- commit message follows commit guidelines
bricss, ChALkeR, benjamingr, bajtos, keul, daliborgogic, alvaropinot, PazzaVlad, IgorNovozhilov, syuilo, and 8 more reacted with thumbs up emoji ChALkeR, bricss, alvaropinot, noinkling, gyj1278, and daliborgogic reacted with hooray emoji
mcollina added the semver-minor
PRs that contain new features and should be released in the next minor version.
label
Also @davidmarkclements that had the original idea: would you like to be included as "co-authored-by"?
Why in core? Can't this be an npm module?
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you! 💙
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same question as @lpinca -
Why in core? Can't this be an npm module?
I mean, I understand why it would be in core, I think -- that is, to make yet more node.js core apis usable with await
.
But I also think that since it doesn't appear to need to be in core, it should at very least have an npm module with adoption to show the weight of it's usefulness...
While this could live outside of core, it's been frequently requested and aligns with our promises support. I'm +1 on adding it
@@ -653,6 +653,46 @@ newListeners[0](); |
---|
emitter.emit('log'); |
``` |
## EventEmitter.once(emitter, name) |
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should not this be events.once(emitter, name)
? Otherwise, this seems to be a static EventEmitter
method like EventEmitter.listenerCount()
above. If I understand correctly, this would be called as events.once()
in the REPL. I see that it can be used as EventEmitter
static method, but in examples, it is imported alongside with the class and this can be confusing for users not aware of EventEmitter.EventEmitter = EventEmitter;
core binding.
If I'm not wrong it even works with a third party EventEmitter
assuming that it supports once()
and removeListener()
which kind of reinforces "why in core?". In my opinion this goes against the small core principle.
- It is not more efficient than a userland implementation.
- It doesn't add something that can't be done in userland.
- As far as I know It is not needed in core itself. Is the plan to make core API use it?
I think this API (or something similar) belongs in core for three main reasons:
- it aligns with our strategic initiative of “promisifying core APIs”.
- it is extremely hard to interact with stream using async/await without an utility like this.
- making sure that 12.0.0 has this will shorten the adoption cycle by 1 year.
I think we can use this API in a lot of test code within core.
I’m using it from a userland module, but tiny utility modules are hard to market at this point.
I’m happy to provide more examples if needed.
Yes, ok but that does not answer "why in core?". A npm module would be better for this in my opinion. Anyway I'm not blocking this and it already has a few TSC approvals.
cc @nodejs/tsc adding to tsc-agenda for visibility
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM. I don't disagree with the comment about this possibly belonging in user-land but I'm also happy with it in Node. Implementation is super clean 👍
} |
---|
resolve(args); |
}; |
let secondListener; |
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is just personal preference but having this at the start of the function body would be clearer for me. When I read the code, the secondListener is referenced before its defined which makes it somewhat confusing. I know it's kind of nitpicky tho so non-blocking...
(Maybe naming it errorRejectListener
would also be clearer?)
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
eventListener
and errorListener
(or something of the like) would be clearer than main
and second
.
No strong opinion, but if there's concerns especially around API design, maybe land as Experimental so we can change the API if we don't get it right the first time?
Trott removed the tsc-agenda
Issues and PRs to discuss during the meetings of the TSC.
label
targos pushed a commit to targos/node that referenced this pull request
This change adds a EventEmitter.once() method that wraps ee.once in a promise.
Co-authored-by: David Mark Clements david.mark.clements@gmail.com
PR-URL: nodejs#26078 Reviewed-By: James M Snell jasnell@gmail.com Reviewed-By: Ruben Bridgewater ruben@bridgewater.de Reviewed-By: Gus Caplan me@gus.host Reviewed-By: Anna Henningsen anna@addaleax.net Reviewed-By: Michaël Zasso targos@protonmail.com Reviewed-By: Anatoli Papirovski apapirovski@mac.com Reviewed-By: Benjamin Gruenbaum benjamingr@gmail.com Reviewed-By: Сковорода Никита Андреевич chalkerx@gmail.com
targos added a commit that referenced this pull request
Notable changes:
- events:
- Added a
once
function to useEventEmitter
with promises (#26078).
- Added a
- tty:
- v8:
- Added
v8.getHeapSnapshot
andv8.writeHeapSnapshot
to generate snapshots in the format used by tools such as Chrome DevTools (#26501).
- Added
- meta:
PR-URL: #26949
targos added a commit that referenced this pull request
Notable changes:
- crypto
- Allow deriving public from private keys (Tobias Nießen) #26278.
- events
- Added a
once
function to useEventEmitter
with promises (Matteo Collina) #26078.
- Added a
- tty
- v8
- Added
v8.getHeapSnapshot
andv8.writeHeapSnapshot
to generate snapshots in the format used by tools such as Chrome DevTools (James M Snell) #26501.
- Added
- worker
- Added
worker.moveMessagePortToContext
. This enables using MessagePorts in different vm.Contexts, aiding with the isolation that the vm module seeks to provide (Anna Henningsen) #26497.
- Added
- C++ API
- meta
- Gireesh Punathil is now a member of the Technical Steering Committee #26657.
- Added Yongsheng Zhang to collaborators #26730.
PR-URL: #26949
targos added a commit that referenced this pull request
Notable changes:
- crypto
- Allow deriving public from private keys (Tobias Nießen) #26278.
- events
- Added a
once
function to useEventEmitter
with promises (Matteo Collina) #26078.
- Added a
- tty
- v8
- Added
v8.getHeapSnapshot
andv8.writeHeapSnapshot
to generate snapshots in the format used by tools such as Chrome DevTools (James M Snell) #26501.
- Added
- worker
- Added
worker.moveMessagePortToContext
. This enables using MessagePorts in different vm.Contexts, aiding with the isolation that the vm module seeks to provide (Anna Henningsen) #26497.
- Added
- C++ API
- meta
- Gireesh Punathil is now a member of the Technical Steering Committee #26657.
- Added Yongsheng Zhang to collaborators #26730.
PR-URL: #26949
BethGriggs pushed a commit that referenced this pull request
Notable changes:
- crypto
- Allow deriving public from private keys (Tobias Nießen) #26278.
- events
- Added a
once
function to useEventEmitter
with promises (Matteo Collina) #26078.
- Added a
- tty
- v8
- Added
v8.getHeapSnapshot
andv8.writeHeapSnapshot
to generate snapshots in the format used by tools such as Chrome DevTools (James M Snell) #26501.
- Added
- worker
- Added
worker.moveMessagePortToContext
. This enables using MessagePorts in different vm.Contexts, aiding with the isolation that the vm module seeks to provide (Anna Henningsen) #26497.
- Added
- C++ API
- meta
- Gireesh Punathil is now a member of the Technical Steering Committee #26657.
- Added Yongsheng Zhang to collaborators #26730.
PR-URL: #26949
BethGriggs pushed a commit that referenced this pull request
This change adds a EventEmitter.once() method that wraps ee.once in a promise.
Co-authored-by: David Mark Clements david.mark.clements@gmail.com
PR-URL: #26078 Reviewed-By: James M Snell jasnell@gmail.com Reviewed-By: Ruben Bridgewater ruben@bridgewater.de Reviewed-By: Gus Caplan me@gus.host Reviewed-By: Anna Henningsen anna@addaleax.net Reviewed-By: Michaël Zasso targos@protonmail.com Reviewed-By: Anatoli Papirovski apapirovski@mac.com Reviewed-By: Benjamin Gruenbaum benjamingr@gmail.com Reviewed-By: Сковорода Никита Андреевич chalkerx@gmail.com
Why not do the same for on
?
Then it would be possible to have a bus based on promises
.once()
callbacks once, promises resolve once. .on()
callbacks repeatedly, so how could that work with promises, which cannot resolve multiple times?
Why not do the same for on?
We could expose a on(ee, 'event')
method that returns an async iterator. However there would be no "exit" condition from the loop.
for await (let event of on(ee, 'event')) { // do something with event // this loop will never end }
Would you mind opening a new issue to discuss? Do you think this would be worthwhile even if the loop would never end?
Well, problem I have so far, if I have an active subscription to events that can come forever until I unsubscribe, there is no way to have a nice handling of that with promises.
Basically I am looking for a "hot observable" kind of thing.
I would recommend using a stream instead. It’s async iterable already.
We could support an on function that returns an async iterator so long as we introduced a concept of a loop terminator.... That is, an object that would cause the loop to terminate...
E.g. something roughly like...
const foo = events.on(emitter, 'foo', term); for await (const f of foo) { // ... term.exit() }
@jasnell @mcollina actually break
and return
would work:
const foo = events.on(emitter, 'foo');
for await (const f of foo) {
break; // this calls .return
on the async iterable and can unsubscribe.
}
This is also true for regular generators. I'm +1 on exploring this.
@benjamingr ... yes, those work within the actual for await
but we would need to make sure we're able to clean up inside the async iterator itself (e.g. removing the event listener). It may be rather trivial to do, I just haven't tried it yet :-) ... Perhaps I'll take a break from QUIC implementation and give it a try :-)
Ok, quick investigation... it should be possible to implement an EventEmitter.on(emitter, event)
function that is essentially a simplification of the code used for stream.Readable
with a few notable differences:
- Unlike
stream.Readable
, there is no natural end to the event stream, and there is nodestroy()
method, so cleanup and termination of thefor await() {}
would be entirely dependent on usingbreak
andreturn
as suggested. - With
stream.Readable
, when there are multiple synchronouspush()
's, there's still just a single 'readable', and just in case there are multiple read() operations performed. For an arbitraryEventEmitter
, we would need a way of reliably handling synchronous emits.
Everything else looks like it should be fairly reasonable to implement if someone wants to take it on ;-)
moving to a separate issue
BethGriggs added a commit that referenced this pull request
Notable changes:
- deps:
- events: add once method to use promises with EventEmitter (Matteo Collina) #26078
- n-api: mark thread-safe function as stable (Gabriel Schulhof) #25556
- repl: support top-level for-await-of (Shelley Vohr) #23841
- zlib:
- add brotli support (Anna Henningsen) #24938
PR-URL: #27514
BethGriggs added a commit that referenced this pull request
Notable changes:
- deps:
- events:
- add once method to use promises with EventEmitter (Matteo Collina) #26078
- n-api:
- mark thread-safe function as stable (Gabriel Schulhof) #25556
- repl:
- support top-level for-await-of (Shelley Vohr) #23841
- zlib:
- add brotli support (Anna Henningsen) #24938
PR-URL: #27514
BethGriggs added a commit that referenced this pull request
Notable changes:
- deps:
- events:
- add once method to use promises with EventEmitter (Matteo Collina) #26078
- n-api:
- mark thread-safe function as stable (Gabriel Schulhof) #25556
- repl:
- support top-level for-await-of (Shelley Vohr) #23841
- zlib:
- add brotli support (Anna Henningsen) #24938
PR-URL: #27514
This was referenced
May 29, 2019
Labels
PRs that have at least one approval, no pending requests for changes, and a CI started.
Issues and PRs related to the events subsystem / EventEmitter.
PRs that contain new features and should be released in the next minor version.