forked from elastic/apm-agent-nodejs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathAbstractRunContextManager.js
214 lines (189 loc) · 6.39 KB
/
AbstractRunContextManager.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
/*
* Copyright Elasticsearch B.V. and other contributors where applicable.
* Licensed under the BSD 2-Clause License; you may not use this file except in
* compliance with the BSD 2-Clause License.
*/
'use strict';
const { RunContext } = require('./RunContext');
const ADD_LISTENER_METHODS = [
'addListener',
'on',
'once',
'prependListener',
'prependOnceListener',
];
// An abstract base RunContextManager class that implements the following
// methods that all run context manager implementations can share:
// root()
// bindFn(runContext, target)
// bindEE(runContext, eventEmitter)
// isEEBound(eventEmitter)
// and stubs out the remaining public methods of the RunContextManager
// interface.
//
// (This class has largerly the same API as @opentelemetry/api `ContextManager`.
// The implementation is adapted from
// https://github.com/open-telemetry/opentelemetry-js/blob/main/packages/opentelemetry-context-async-hooks/src/AbstractAsyncHooksContextManager.ts)
class AbstractRunContextManager {
constructor(log, runContextClass = RunContext) {
this._log = log;
this._kListeners = Symbol('ElasticListeners');
this._root = new runContextClass(); // eslint-disable-line new-cap
}
// Get the root run context. This is always empty (no current trans or span).
//
// This is the equivalent of OTel JS API's `ROOT_CONTEXT` constant. Ours
// is not a top-level constant, because the RunContext class can be
// overriden.
root() {
return this._root;
}
enable() {
throw new Error('abstract method not implemented');
}
disable() {
throw new Error('abstract method not implemented');
}
// Reset state for re-use of this context manager by tests in the same process.
testReset() {
this.disable();
this.enable();
}
active() {
throw new Error('abstract method not implemented');
}
with(runContext, fn, thisArg, ...args) {
throw new Error('abstract method not implemented');
}
supersedeRunContext(runContext) {
throw new Error('abstract method not implemented');
}
// The OTel ContextManager API has a single .bind() like this:
//
// bind (runContext, target) {
// if (target instanceof EventEmitter) {
// return this._bindEventEmitter(runContext, target)
// }
// if (typeof target === 'function') {
// return this._bindFunction(runContext, target)
// }
// return target
// }
//
// Is there any value in this over our two separate `.bind*` methods?
bindFn(runContext, target) {
if (typeof target !== 'function') {
return target;
}
// this._log.trace('bind %s to fn "%s"', runContext, target.name)
const self = this;
const wrapper = function () {
return self.with(runContext, () => target.apply(this, arguments));
};
Object.defineProperty(wrapper, 'length', {
enumerable: false,
configurable: true,
writable: false,
value: target.length,
});
return wrapper;
}
// (This implementation is adapted from OTel's `_bindEventEmitter`.)
bindEE(runContext, ee) {
// Explicitly do *not* guard with `ee instanceof EventEmitter`. The
// `Request` object from the aws-sdk@2 module, for example, has an `on`
// with the EventEmitter API that we want to bind, but it is not otherwise
// an EventEmitter.
const map = this._getPatchMap(ee);
if (map !== undefined) {
// No double-binding.
return ee;
}
this._createPatchMap(ee);
// patch methods that add a listener to propagate context
ADD_LISTENER_METHODS.forEach((methodName) => {
if (ee[methodName] === undefined) return;
ee[methodName] = this._patchAddListener(ee, ee[methodName], runContext);
});
// patch methods that remove a listener
if (typeof ee.removeListener === 'function') {
ee.removeListener = this._patchRemoveListener(ee, ee.removeListener);
}
if (typeof ee.off === 'function') {
ee.off = this._patchRemoveListener(ee, ee.off);
}
// patch method that remove all listeners
if (typeof ee.removeAllListeners === 'function') {
ee.removeAllListeners = this._patchRemoveAllListeners(
ee,
ee.removeAllListeners,
);
}
return ee;
}
// Return true iff the given EventEmitter is already bound to a run context.
isEEBound(ee) {
return this._getPatchMap(ee) !== undefined;
}
// Patch methods that remove a given listener so that we match the "patched"
// version of that listener (the one that propagate context).
_patchRemoveListener(ee, original) {
const contextManager = this;
return function (event, listener) {
const map = contextManager._getPatchMap(ee);
const listeners = map && map[event];
if (listeners === undefined) {
return original.call(this, event, listener);
}
const patchedListener = listeners.get(listener);
return original.call(this, event, patchedListener || listener);
};
}
// Patch methods that remove all listeners so we remove our internal
// references for a given event.
_patchRemoveAllListeners(ee, original) {
const contextManager = this;
return function (event) {
const map = contextManager._getPatchMap(ee);
if (map !== undefined) {
if (arguments.length === 0) {
contextManager._createPatchMap(ee);
} else if (map[event] !== undefined) {
delete map[event];
}
}
return original.apply(this, arguments);
};
}
// Patch methods on an event emitter instance that can add listeners so we
// can force them to propagate a given context.
_patchAddListener(ee, original, runContext) {
const contextManager = this;
return function (event, listener) {
let map = contextManager._getPatchMap(ee);
if (map === undefined) {
map = contextManager._createPatchMap(ee);
}
let listeners = map[event];
if (listeners === undefined) {
listeners = new WeakMap();
map[event] = listeners;
}
const patchedListener = contextManager.bindFn(runContext, listener);
// store a weak reference of the user listener to ours
listeners.set(listener, patchedListener);
return original.call(this, event, patchedListener);
};
}
_createPatchMap(ee) {
const map = Object.create(null);
ee[this._kListeners] = map;
return map;
}
_getPatchMap(ee) {
return ee[this._kListeners];
}
}
module.exports = {
AbstractRunContextManager,
};