Skip to content

Commit 74df749

Browse files
committed
async_hooks: add AsyncResource.bind utility
Creates an internal AsyncResource and binds a function to it, ensuring that the function is invoked within execution context in which bind was called. PR-URL: #34574 Reviewed-By: Stephen Belanger <[email protected]> Reviewed-By: Gus Caplan <[email protected]> Reviewed-By: Matteo Collina <[email protected]> Reviewed-By: Andrey Pechkurov <[email protected]> Reviewed-By: Gerhard Stöbich <[email protected]>
1 parent 744a284 commit 74df749

File tree

3 files changed

+90
-5
lines changed

3 files changed

+90
-5
lines changed

doc/api/async_hooks.md

+31-5
Original file line numberDiff line numberDiff line change
@@ -729,6 +729,32 @@ class DBQuery extends AsyncResource {
729729
}
730730
```
731731

732+
#### `static AsyncResource.bind(fn[, type])`
733+
<!-- YAML
734+
added: REPLACEME
735+
-->
736+
737+
* `fn` {Function} The function to bind to the current execution context.
738+
* `type` {string} An optional name to associate with the underlying
739+
`AsyncResource`.
740+
741+
Binds the given function to the current execution context.
742+
743+
The returned function will have an `asyncResource` property referencing
744+
the `AsyncResource` to which the function is bound.
745+
746+
#### `asyncResource.bind(fn)`
747+
<!-- YAML
748+
added: REPLACEME
749+
-->
750+
751+
* `fn` {Function} The function to bind to the current `AsyncResource`.
752+
753+
Binds the given function to execute to this `AsyncResource`'s scope.
754+
755+
The returned function will have an `asyncResource` property referencing
756+
the `AsyncResource` to which the function is bound.
757+
732758
#### `asyncResource.runInAsyncScope(fn[, thisArg, ...args])`
733759
<!-- YAML
734760
added: v9.6.0
@@ -900,12 +926,12 @@ const { createServer } = require('http');
900926
const { AsyncResource, executionAsyncId } = require('async_hooks');
901927

902928
const server = createServer((req, res) => {
903-
const asyncResource = new AsyncResource('request');
904-
// The listener will always run in the execution context of `asyncResource`.
905-
req.on('close', asyncResource.runInAsyncScope.bind(asyncResource, () => {
906-
// Prints: true
907-
console.log(asyncResource.asyncId() === executionAsyncId());
929+
req.on('close', AsyncResource.bind(() => {
930+
// Execution context is bound to the current outer scope.
908931
}));
932+
req.on('close', () => {
933+
// Execution context is bound to the scope that caused 'close' to emit.
934+
});
909935
res.end();
910936
}).listen(3000);
911937
```

lib/async_hooks.js

+24
Original file line numberDiff line numberDiff line change
@@ -2,13 +2,15 @@
22

33
const {
44
NumberIsSafeInteger,
5+
ObjectDefineProperties,
56
ReflectApply,
67
Symbol,
78
} = primordials;
89

910
const {
1011
ERR_ASYNC_CALLBACK,
1112
ERR_ASYNC_TYPE,
13+
ERR_INVALID_ARG_TYPE,
1214
ERR_INVALID_ASYNC_ID
1315
} = require('internal/errors').codes;
1416
const { validateString } = require('internal/validators');
@@ -211,6 +213,28 @@ class AsyncResource {
211213
triggerAsyncId() {
212214
return this[trigger_async_id_symbol];
213215
}
216+
217+
bind(fn) {
218+
if (typeof fn !== 'function')
219+
throw new ERR_INVALID_ARG_TYPE('fn', 'Function', fn);
220+
const ret = this.runInAsyncScope.bind(this, fn);
221+
ObjectDefineProperties(ret, {
222+
'length': {
223+
enumerable: true,
224+
value: fn.length,
225+
},
226+
'asyncResource': {
227+
enumerable: true,
228+
value: this,
229+
}
230+
});
231+
return ret;
232+
}
233+
234+
static bind(fn, type) {
235+
type = type || fn.name;
236+
return (new AsyncResource(type || 'bound-anonymous-fn')).bind(fn);
237+
}
214238
}
215239

216240
const storageList = [];
+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const assert = require('assert');
5+
const { AsyncResource, executionAsyncId } = require('async_hooks');
6+
7+
const fn = common.mustCall(AsyncResource.bind(() => {
8+
return executionAsyncId();
9+
}));
10+
11+
setImmediate(() => {
12+
const asyncId = executionAsyncId();
13+
assert.notStrictEqual(asyncId, fn());
14+
});
15+
16+
const asyncResource = new AsyncResource('test');
17+
18+
[1, false, '', {}, []].forEach((i) => {
19+
assert.throws(() => asyncResource.bind(i), {
20+
code: 'ERR_INVALID_ARG_TYPE'
21+
});
22+
});
23+
24+
const fn2 = asyncResource.bind((a, b) => {
25+
return executionAsyncId();
26+
});
27+
28+
assert.strictEqual(fn2.asyncResource, asyncResource);
29+
assert.strictEqual(fn2.length, 2);
30+
31+
setImmediate(() => {
32+
const asyncId = executionAsyncId();
33+
assert.strictEqual(asyncResource.asyncId(), fn2());
34+
assert.notStrictEqual(asyncId, fn2());
35+
});

0 commit comments

Comments
 (0)