@@ -10,7 +10,6 @@ const {
10
10
NumberIsSafeInteger,
11
11
ObjectDefineProperties,
12
12
ObjectFreeze,
13
- ObjectIs,
14
13
ReflectApply,
15
14
Symbol,
16
15
} = primordials ;
@@ -30,6 +29,8 @@ const {
30
29
} = require ( 'internal/validators' ) ;
31
30
const internal_async_hooks = require ( 'internal/async_hooks' ) ;
32
31
32
+ const AsyncContextFrame = require ( 'internal/async_context_frame' ) ;
33
+
33
34
// Get functions
34
35
// For userland AsyncResources, make sure to emit a destroy event when the
35
36
// resource gets gced.
@@ -158,6 +159,7 @@ function createHook(fns) {
158
159
// Embedder API //
159
160
160
161
const destroyedSymbol = Symbol ( 'destroyed' ) ;
162
+ const contextFrameSymbol = Symbol ( 'context_frame' ) ;
161
163
162
164
class AsyncResource {
163
165
constructor ( type , opts = kEmptyObject ) {
@@ -177,6 +179,8 @@ class AsyncResource {
177
179
throw new ERR_INVALID_ASYNC_ID ( 'triggerAsyncId' , triggerAsyncId ) ;
178
180
}
179
181
182
+ this [ contextFrameSymbol ] = AsyncContextFrame . current ( ) ;
183
+
180
184
const asyncId = newAsyncId ( ) ;
181
185
this [ async_id_symbol ] = asyncId ;
182
186
this [ trigger_async_id_symbol ] = triggerAsyncId ;
@@ -201,12 +205,12 @@ class AsyncResource {
201
205
const asyncId = this [ async_id_symbol ] ;
202
206
emitBefore ( asyncId , this [ trigger_async_id_symbol ] , this ) ;
203
207
208
+ const contextFrame = this [ contextFrameSymbol ] ;
209
+ const prior = AsyncContextFrame . exchange ( contextFrame ) ;
204
210
try {
205
- const ret =
206
- ReflectApply ( fn , thisArg , args ) ;
207
-
208
- return ret ;
211
+ return ReflectApply ( fn , thisArg , args ) ;
209
212
} finally {
213
+ AsyncContextFrame . set ( prior ) ;
210
214
if ( hasAsyncIdStack ( ) )
211
215
emitAfter ( asyncId ) ;
212
216
}
@@ -270,110 +274,15 @@ class AsyncResource {
270
274
}
271
275
}
272
276
273
- const storageList = [ ] ;
274
- const storageHook = createHook ( {
275
- init ( asyncId , type , triggerAsyncId , resource ) {
276
- const currentResource = executionAsyncResource ( ) ;
277
- // Value of currentResource is always a non null object
278
- for ( let i = 0 ; i < storageList . length ; ++ i ) {
279
- storageList [ i ] . _propagate ( resource , currentResource , type ) ;
280
- }
281
- } ,
282
- } ) ;
283
-
284
- class AsyncLocalStorage {
285
- constructor ( ) {
286
- this . kResourceStore = Symbol ( 'kResourceStore' ) ;
287
- this . enabled = false ;
288
- }
289
-
290
- static bind ( fn ) {
291
- return AsyncResource . bind ( fn ) ;
292
- }
293
-
294
- static snapshot ( ) {
295
- return AsyncLocalStorage . bind ( ( cb , ...args ) => cb ( ...args ) ) ;
296
- }
297
-
298
- disable ( ) {
299
- if ( this . enabled ) {
300
- this . enabled = false ;
301
- // If this.enabled, the instance must be in storageList
302
- ArrayPrototypeSplice ( storageList ,
303
- ArrayPrototypeIndexOf ( storageList , this ) , 1 ) ;
304
- if ( storageList . length === 0 ) {
305
- storageHook . disable ( ) ;
306
- }
307
- }
308
- }
309
-
310
- _enable ( ) {
311
- if ( ! this . enabled ) {
312
- this . enabled = true ;
313
- ArrayPrototypePush ( storageList , this ) ;
314
- storageHook . enable ( ) ;
315
- }
316
- }
317
-
318
- // Propagate the context from a parent resource to a child one
319
- _propagate ( resource , triggerResource , type ) {
320
- const store = triggerResource [ this . kResourceStore ] ;
321
- if ( this . enabled ) {
322
- resource [ this . kResourceStore ] = store ;
323
- }
324
- }
325
-
326
- enterWith ( store ) {
327
- this . _enable ( ) ;
328
- const resource = executionAsyncResource ( ) ;
329
- resource [ this . kResourceStore ] = store ;
330
- }
331
-
332
- run ( store , callback , ...args ) {
333
- // Avoid creation of an AsyncResource if store is already active
334
- if ( ObjectIs ( store , this . getStore ( ) ) ) {
335
- return ReflectApply ( callback , null , args ) ;
336
- }
337
-
338
- this . _enable ( ) ;
339
-
340
- const resource = executionAsyncResource ( ) ;
341
- const oldStore = resource [ this . kResourceStore ] ;
342
-
343
- resource [ this . kResourceStore ] = store ;
344
-
345
- try {
346
- return ReflectApply ( callback , null , args ) ;
347
- } finally {
348
- resource [ this . kResourceStore ] = oldStore ;
349
- }
350
- }
351
-
352
- exit ( callback , ...args ) {
353
- if ( ! this . enabled ) {
354
- return ReflectApply ( callback , null , args ) ;
355
- }
356
- this . disable ( ) ;
357
- try {
358
- return ReflectApply ( callback , null , args ) ;
359
- } finally {
360
- this . _enable ( ) ;
361
- }
362
- }
363
-
364
- getStore ( ) {
365
- if ( this . enabled ) {
366
- const resource = executionAsyncResource ( ) ;
367
- return resource [ this . kResourceStore ] ;
368
- }
369
- }
370
- }
371
-
372
277
// Placing all exports down here because the exported classes won't export
373
278
// otherwise.
374
279
module . exports = {
375
280
// Public API
376
- AsyncLocalStorage,
281
+ get AsyncLocalStorage ( ) {
282
+ return AsyncContextFrame . enabled ?
283
+ require ( 'internal/async_local_storage/native' ) :
284
+ require ( 'internal/async_local_storage/async_hooks' ) ;
285
+ } ,
377
286
createHook,
378
287
executionAsyncId,
379
288
triggerAsyncId,
0 commit comments