Description
Background
The current implementation of promise integration into async_hooks
is.
- For each new
[[Promise]]
object, call theinit
hook with aPromiseWrap
referencing that Promise. - At
[[resolve]]
or[[reject]]
call theresolve
hook. - For each
[[then]]
callback, call thebefore
hook before calling the callback. - For each
[[then]]
callback, call theafter
hook after calling the callback. - When the Promise object is garbage collected call the destroy hook.
I think this implementation is fundamentally wrong, as it intertwines the promise lifecycle with async_hooks
. This causes a number of issues that are currently blocking us from making async_hooks
stable.
- Performance issues caused by listening for the garbage collection event.
- Thenables are not tracked when used by a native function that creates a microtrask.
destroy
hook is not called ifasync_hook
is enabled after Promise creation.- Tracking the async boundary when multiple
.then()
calls are used on the same promise object, is not possible.
Proposal
My proposal is to rework the promise integration into async_hooks
such that the async barrier is around the [[then]]
call, not creating a new Promise
object.
- at the call of
[[then]]
on apromise
orthenable
create a resource object (or use the promise/thenable object created by[[then]]
) then call theinit
hook. - at the call of the
[[then]]
callback, thebefore
hook is called. - at the end of the
[[then]]
callback, call theafter
hook, immediately after call thedestroy
hook.
How it solves the above-mentioned issues
Performance issues caused by listening for the garbage collection event.
Because the before
and after
hooks are only called once per resource, the destroy
hook can be called immediately after the after
hook. Thus completely eliminating the need to track promise objects in the garbage collector.
Thenables are not tracked when used by a native function that creates a microtrask.
This can now be solved because we don't need to know when the object was created or destroyed. The only knowledge that is required is when the [[then]]
method of the thenable is called by the native JS APIs which invokes the microtask queue. That is actually doable, as we could hook into those APIs. Only manual calls to [[then]]
on a thenable will not be tracked. But I don't see that as a concern, because that is not an async action.
destroy
hook is not called if async_hook
is enabled after Promise creation.
Again, because the destroy
hook is called with the after
hook, calling the destroy
hook becomes trivial.
Tracking the async boundary when multiple .then()
calls are used on the same promise object, is not possible.
This directly confronts this issue, by making the async boundary the [[then]]
call.
Tracking the lifetime of promises
This proposal removes features from async_hooks
that provide insight into promises
. That information is still valuable.
To keep providing that information, I propose making a dedicated promise_hooks
module. A user can then connect the promise lifecycle with async_hooks
via a promiseId
that is exposed both in promise_hook
and via the resource in async_hooks
.
Compatability
This does change the default async chain, as the start point of the async-barrier is now the .then()
call and not the new Promise
call. However, I think this is actually a preferred default. For example, lazyloading a resource with new Promise
, would currently not be tracked correctly, but with the proposed change it will.
Even though this changes the async graph, the proposed async graph will still be valid and useful in most cases. And the existing async graph can be restored by integrating the proposed promise_hooks
module.
implementing this proposal should be part of a major release.
Open questions
- It is unclear to me how this would integrate with
async/await
. As I don't have a good grasp of how the internal[[then]]
calls are used and wrapped. It has been proposed that we supply our own customMicrotaskQueue
(async_hooks performance/stability improvement plans #376 (comment)) to solve this. I think this could also be a good approach to hook into the native promise APIs, which will be necessary to track[[then]]
calls on thenables.