-
Notifications
You must be signed in to change notification settings - Fork 471
/
Copy pathasync_context.js
122 lines (112 loc) · 3.25 KB
/
async_context.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
'use strict';
const assert = require('assert');
const common = require('./common');
// we only check async hooks on 8.x an higher were
// they are closer to working properly
const nodeVersion = process.versions.node.split('.')[0];
let asyncHooks;
function checkAsyncHooks () {
if (nodeVersion >= 8) {
if (asyncHooks === undefined) {
asyncHooks = require('async_hooks');
}
return true;
}
return false;
}
module.exports = common.runTest(test);
function installAsyncHooksForTest (resName) {
return new Promise((resolve, reject) => {
let id;
const events = [];
/**
* TODO(legendecas): investigate why resolving & disabling hooks in
* destroy callback causing crash with case 'callbackscope.js'.
*/
let destroyed = false;
const hook = asyncHooks.createHook({
init (asyncId, type, triggerAsyncId, resource) {
if (id === undefined && type === resName) {
id = asyncId;
events.push({ eventName: 'init', type, triggerAsyncId, resource });
}
},
before (asyncId) {
if (asyncId === id) {
events.push({ eventName: 'before' });
}
},
after (asyncId) {
if (asyncId === id) {
events.push({ eventName: 'after' });
}
},
destroy (asyncId) {
if (asyncId === id) {
events.push({ eventName: 'destroy' });
destroyed = true;
}
}
}).enable();
const interval = setInterval(() => {
if (destroyed) {
hook.disable();
clearInterval(interval);
resolve(events);
}
}, 10);
});
}
async function makeCallbackWithResource (binding) {
const hooks = installAsyncHooksForTest('async_context_test');
const triggerAsyncId = asyncHooks.executionAsyncId();
await new Promise((resolve, reject) => {
binding.asynccontext.makeCallback(common.mustCall(), { foo: 'foo' });
hooks.then(actual => {
assert.deepStrictEqual(actual, [
{
eventName: 'init',
type: 'async_context_test',
triggerAsyncId,
resource: { foo: 'foo' }
},
{ eventName: 'before' },
{ eventName: 'after' },
{ eventName: 'destroy' }
]);
}).catch(common.mustNotCall());
resolve();
});
}
async function makeCallbackWithoutResource (binding) {
const hooks = installAsyncHooksForTest('async_context_no_res_test');
const triggerAsyncId = asyncHooks.executionAsyncId();
await new Promise((resolve, reject) => {
binding.asynccontext.makeCallbackNoResource(common.mustCall());
hooks.then(actual => {
assert.deepStrictEqual(actual, [
{
eventName: 'init',
type: 'async_context_no_res_test',
triggerAsyncId,
resource: { }
},
{ eventName: 'before' },
{ eventName: 'after' },
{ eventName: 'destroy' }
]);
}).catch(common.mustNotCall());
resolve();
});
}
function assertAsyncContextReturnsCorrectEnv (binding) {
assert.strictEqual(binding.asynccontext.asyncCxtReturnCorrectEnv(), true);
}
async function test (binding) {
if (!checkAsyncHooks()) {
return;
}
await makeCallbackWithResource(binding);
await makeCallbackWithoutResource(binding);
assertAsyncContextReturnsCorrectEnv(binding);
}