-
Notifications
You must be signed in to change notification settings - Fork 410
/
Copy pathasync_hooks.js
246 lines (221 loc) · 6.4 KB
/
async_hooks.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
'use strict'
var test = require('tap').test
var helper = require('../../lib/agent_helper')
var asyncHooks = require('async_hooks')
function testSegments(t, segmentMap) {
global.gc()
// Give the gc some time to work.
setTimeout(function() {
t.notOk(segmentMap.size, 'segments should be cleared after gc')
t.end()
}, 10)
}
test('await', function(t) {
var agent = setupAgent(t)
helper.runInTransaction(agent, async function(txn) {
var transaction = agent.getTransaction()
t.equal(
transaction && transaction.id,
txn.id,
'should start in a transaction'
)
await Promise.resolve("i'll be back")
transaction = agent.getTransaction()
t.equal(
transaction && transaction.id,
txn.id,
'should resume in the same transaction after await'
)
var segmentMap = require('../../../lib/instrumentation/core/async_hooks')._segmentMap
txn.end(function afterTransactionEnd() {
// Segments won't be cleared till a gc cycle clears the promises
// they are related with.
t.ok(segmentMap.size, 'segments should still be in the map')
if (global.gc) {
// Unroll all the stack frames to let go of the refs to the
// promises we want to gc, then call the segment tester.
return setImmediate(testSegments, t, segmentMap)
}
t.end()
})
})
})
test("the agent's async hook", function(t) {
class TestResource extends asyncHooks.AsyncResource {
constructor(id) {
super('PROMISE', id)
}
doStuff(callback) {
process.nextTick(() => {
this.emitBefore()
callback()
this.emitAfter()
})
}
}
t.autoend()
t.test('does not crash on multiple resolve calls', function(t) {
var agent = setupAgent(t)
helper.runInTransaction(agent, function(txn) {
var called = false
t.doesNotThrow(function () {
new Promise(function(res, rej) {
res()
res()
}).then(t.end)
})
})
})
t.test('does not restore a segment for a resource created outside a transaction', function(t) {
var agent = setupAgent(t)
var res = new TestResource(1)
helper.runInTransaction(agent, function(txn) {
var root = agent.tracer.segment
var segmentMap = require('../../../lib/instrumentation/core/async_hooks')._segmentMap
t.equal(segmentMap.size, 0, 'no segments should be tracked')
res.doStuff(function() {
t.ok(agent.tracer.segment, 'should be in a transaction')
t.equal(
agent.tracer.segment.name,
root.name,
'the agent loses transaction state for resources created outside of a transaction'
)
t.end()
})
})
})
t.test('restores context in inactive transactions', function(t) {
var agent = setupAgent(t)
helper.runInTransaction(agent, function(txn) {
var res = new TestResource(1)
var root = agent.tracer.segment
txn.end()
res.doStuff(function() {
t.equal(
agent.tracer.segment,
root,
'the hooks restore a segment when its transaction has been ended'
)
t.end()
})
})
})
t.test('handles multientry callbacks correctly', function(t) {
var agent = setupAgent(t)
var segmentMap = require('../../../lib/instrumentation/core/async_hooks')._segmentMap
helper.runInTransaction(agent, function(txn) {
var root = agent.tracer.segment
var aSeg = agent.tracer.createSegment('A')
agent.tracer.segment = aSeg
var resA = new TestResource(1)
var bSeg = agent.tracer.createSegment('B')
agent.tracer.segment = bSeg
var resB = new TestResource(2)
agent.tracer.segment = root
t.equal(segmentMap.size, 2, 'all resources should create an entry on init')
resA.doStuff(() => {
t.equal(
agent.tracer.segment.name,
aSeg.name,
'calling emitBefore should restore the segment active when a resource was made'
)
resB.doStuff(() => {
t.equal(
agent.tracer.segment.name,
bSeg.name,
'calling emitBefore should restore the segment active when a resource was made'
)
t.end()
})
t.equal(
agent.tracer.segment.name,
aSeg.name,
'calling emitAfter should restore the segment active when a callback was called'
)
})
t.equal(
agent.tracer.segment.name,
root.name,
'root should be restored after we are finished'
)
resA.doStuff(() => {
t.equal(
agent.tracer.segment.name,
aSeg.name,
'calling emitBefore should restore the segment active when a resource was made'
)
})
})
})
})
function checkCallMetrics(t, testMetrics) {
t.equal(testMetrics.initCalled, 2, 'two promises were created')
t.equal(testMetrics.beforeCalled, 1, 'before hook called for all async promises')
t.equal(
testMetrics.beforeCalled,
testMetrics.afterCalled,
'before should be called as many times as after'
)
if (global.gc) {
global.gc()
return setTimeout(function() {
t.equal(
testMetrics.initCalled,
testMetrics.destroyCalled,
'all promises created were destroyed'
)
t.end()
}, 10)
}
t.end()
}
test('promise hooks', function(t) {
t.autoend()
var testMetrics = {
initCalled: 0,
beforeCalled: 0,
afterCalled: 0,
destroyCalled: 0
}
var promiseIds = {}
var hook = asyncHooks.createHook({
init: function initHook(id, type, triggerAsyncId) {
if (type === 'PROMISE') {
promiseIds[id] = true
testMetrics.initCalled++
}
},
before: function beforeHook(id) {
if (promiseIds[id]) {
testMetrics.beforeCalled++
}
},
after: function afterHook(id) {
if (promiseIds[id]) {
testMetrics.afterCalled++
}
},
destroy: function destHook(id) {
if (promiseIds[id]) {
testMetrics.destroyCalled++
}
}
})
hook.enable()
t.test('are only called once during the lifetime of a promise', function(t) {
new Promise(function(res, rej) {
setTimeout(res, 10)
}).then(function() {
setImmediate(checkCallMetrics, t, testMetrics)
})
})
})
function setupAgent(t) {
var agent = helper.instrumentMockedAgent({
await_support: true
})
t.tearDown(function() {
helper.unloadAgent(agent)
})
return agent
}