Skip to content
This repository was archived by the owner on Feb 26, 2024. It is now read-only.

Commit 8d64c10

Browse files
committed
feat(async): fix #740, use async hooks to wrap nodejs async api
1 parent b92b6e3 commit 8d64c10

7 files changed

+251
-56
lines changed

lib/node/node_async_check.ts

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
/**
2+
* @license
3+
* Copyright Google Inc. All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
Zone.__load_patch('node_async_check', (global: any, Zone: ZoneType, api: _ZonePrivate) => {
10+
try {
11+
require('async_hooks');
12+
(process as any)._rawDebug('load async_hooks');
13+
// nodejs 8.x with async_hooks support.
14+
// disable original Zone patch
15+
global['__Zone_disable_ZoneAwarePromise'] = true;
16+
global['__Zone_disable_node_timers'] = true;
17+
global['__Zone_disable_nextTick'] = true;
18+
global['__Zone_disable_handleUnhandledPromiseRejection'] = true;
19+
global['__Zone_disable_crypto'] = true;
20+
global['__Zone_disable_fs'] = true;
21+
} catch (err) {
22+
global['__Zone_disable_node_async_hooks'] = true;
23+
}
24+
});

lib/node/node_asynchooks.ts

+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
/**
2+
* @license
3+
* Copyright Google Inc. All Rights Reserved.
4+
*
5+
* Use of this source code is governed by an MIT-style license that can be
6+
* found in the LICENSE file at https://angular.io/license
7+
*/
8+
9+
/**
10+
* patch nodejs async operations (timer, promise, net...) with
11+
* nodejs async_hooks
12+
*/
13+
Zone.__load_patch('node_async_hooks', (global: any, Zone: ZoneType, api: _ZonePrivate) => {
14+
let async_hooks;
15+
const BEFORE_RUN_TASK_STATUS = 'BEFORE_RUN_TASK_STATUS';
16+
17+
async_hooks = require('async_hooks');
18+
19+
const idTaskMap: {[key: number]: Task} = (Zone as any)[Zone.__symbol__('nodeTasks')] = {};
20+
21+
const noop = function() {};
22+
23+
function init(id: number, provider: string, parentId: number, parentHandle: any) {
24+
// @JiaLiPassion, check which tasks are microTask or macroTask
25+
//(process as any)._rawDebug('init hook', id , provider);
26+
if (provider === 'TIMERWRAP') {
27+
return;
28+
}
29+
// create microTask if 'PROMISE'
30+
if (provider === 'PROMISE') {
31+
const task = idTaskMap[id] = Zone.current.scheduleMicroTask(provider, noop, null, noop);
32+
//(process as any)._rawDebug('after init', id, 'status', task.state);
33+
return;
34+
}
35+
// create macroTask in other cases
36+
if (provider === 'Timeout' || provider === 'Immediate' || provider === 'FSREQWRAP') {
37+
idTaskMap[id] = Zone.current.scheduleMacroTask(provider, noop, null, noop, noop);
38+
}
39+
}
40+
41+
function before(id: number) {
42+
//(process as any)._rawDebug('before hook', id);
43+
// call Zone.beforeRunTask
44+
const task: Task = idTaskMap[id];
45+
if (!task) {
46+
return;
47+
}
48+
(task as any)[Zone.__symbol__(BEFORE_RUN_TASK_STATUS)] = api.beforeRunTask(task.zone, task);
49+
}
50+
51+
function after(id: number) {
52+
//(process as any)._rawDebug('after hook', id);
53+
const task: Task = idTaskMap[id];
54+
if (!task) {
55+
return;
56+
}
57+
const beforeRunTask: BeforeRunTaskStatus = (task as any)[Zone.__symbol__(BEFORE_RUN_TASK_STATUS)];
58+
if (beforeRunTask) {
59+
return;
60+
}
61+
(task as any)[Zone.__symbol__(BEFORE_RUN_TASK_STATUS)] = null;
62+
api.afterRunTask(task.zone, beforeRunTask, task);
63+
}
64+
65+
function destroy(id: number) {
66+
// try to cancel the task if is not canceled
67+
const task: Task = idTaskMap[id];
68+
if (task && task.state === 'scheduled') {
69+
task.zone.cancelTask(task);
70+
}
71+
idTaskMap[id] = null;
72+
}
73+
74+
process.on('uncaughtException', (err: any) => {
75+
const task = Zone.currentTask;
76+
if (task) {
77+
const beforeRunTask: BeforeRunTaskStatus = (task as any)[Zone.__symbol__(BEFORE_RUN_TASK_STATUS)];
78+
if (beforeRunTask) {
79+
if ((task.zone as any)._zoneDelegate.handleError(Zone.current, err)) {
80+
throw err;
81+
}
82+
}
83+
}
84+
});
85+
86+
global[Zone.__symbol__('setTimeout')] = global.setTimeout;
87+
global[Zone.__symbol__('setInterval')] = global.setInterval;
88+
global[Zone.__symbol__('setImmediate')] = global.setImmediate;
89+
90+
async_hooks.createHook({ init, before, after, destroy }).enable();
91+
});

lib/node/rollup-main.ts

+2
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@
77
*/
88

99
import '../zone';
10+
import '../node_async_check';
11+
import '../node_asynchooks';
1012
import '../common/promise';
1113
import '../common/to-string';
1214
import './node';

lib/zone.ts

+49-20
Original file line numberDiff line numberDiff line change
@@ -312,6 +312,12 @@ interface ZoneType {
312312
/** @internal */
313313
type _PatchFn = (global: Window, Zone: ZoneType, api: _ZonePrivate) => void;
314314

315+
/** @internal */
316+
interface BeforeRunTaskStatus {
317+
reEntryGuard: boolean;
318+
previousTask: Task;
319+
}
320+
315321
/** @internal */
316322
interface _ZonePrivate {
317323
currentZoneFrame: () => _ZoneFrame;
@@ -323,6 +329,8 @@ interface _ZonePrivate {
323329
patchEventTargetMethods:
324330
(obj: any, addFnName?: string, removeFnName?: string, metaCreator?: any) => boolean;
325331
patchOnProperties: (obj: any, properties: string[]) => void;
332+
beforeRunTask: (zone: Zone, task: Task) => BeforeRunTaskStatus;
333+
afterRunTask: (zone: Zone, beforeRunTaskStatus: BeforeRunTaskStatus, task: Task) => void;
326334
}
327335

328336
/** @internal */
@@ -757,8 +765,7 @@ const Zone: ZoneType = (function(global: any) {
757765
}
758766
}
759767

760-
761-
runTask(task: Task, applyThis?: any, applyArgs?: any): any {
768+
beforeRunTask(task: Task) {
762769
if (task.zone != this) {
763770
throw new Error(
764771
'A task can only be run in the zone of creation! (Creation: ' +
@@ -772,7 +779,7 @@ const Zone: ZoneType = (function(global: any) {
772779
// typescript compiler will complain below
773780
const isNotScheduled = task.state === notScheduled;
774781
if (isNotScheduled && task.type === eventTask) {
775-
return;
782+
return null;
776783
}
777784

778785
const reEntryGuard = task.state != running;
@@ -781,6 +788,33 @@ const Zone: ZoneType = (function(global: any) {
781788
const previousTask = _currentTask;
782789
_currentTask = task;
783790
_currentZoneFrame = {parent: _currentZoneFrame, zone: this};
791+
//(process as any)._rawDebug('currentFrame increase ', _currentZoneFrame && _currentZoneFrame.zone.name, task.source);
792+
return {
793+
reEntryGuard: reEntryGuard,
794+
previousTask: previousTask
795+
}
796+
}
797+
798+
afterRunTask(beforeRunTaskStatus: BeforeRunTaskStatus, task: Task) {
799+
// if the task's state is notScheduled or unknown, then it has already been cancelled
800+
// we should not reset the state to scheduled
801+
if (task.state !== notScheduled && task.state !== unknown) {
802+
if (task.type == eventTask || (task.data && task.data.isPeriodic)) {
803+
beforeRunTaskStatus.reEntryGuard && (task as ZoneTask<any>)._transitionTo(scheduled, running);
804+
} else {
805+
task.runCount = 0;
806+
this._updateTaskCount(task as ZoneTask<any>, -1);
807+
beforeRunTaskStatus.reEntryGuard &&
808+
(task as ZoneTask<any>)._transitionTo(notScheduled, running, notScheduled);
809+
}
810+
}
811+
_currentZoneFrame = _currentZoneFrame.parent;
812+
//(process as any)._rawDebug('currentFrame decrease ', _currentZoneFrame && _currentZoneFrame.zone.name, task.source);
813+
_currentTask = beforeRunTaskStatus.previousTask;
814+
}
815+
816+
runTask(task: Task, applyThis?: any, applyArgs?: any): any {
817+
const beforeRunTaskStatus = this.beforeRunTask(task);
784818
try {
785819
if (task.type == macroTask && task.data && !task.data.isPeriodic) {
786820
task.cancelFn = null;
@@ -793,20 +827,7 @@ const Zone: ZoneType = (function(global: any) {
793827
}
794828
}
795829
} finally {
796-
// if the task's state is notScheduled or unknown, then it has already been cancelled
797-
// we should not reset the state to scheduled
798-
if (task.state !== notScheduled && task.state !== unknown) {
799-
if (task.type == eventTask || (task.data && task.data.isPeriodic)) {
800-
reEntryGuard && (task as ZoneTask<any>)._transitionTo(scheduled, running);
801-
} else {
802-
task.runCount = 0;
803-
this._updateTaskCount(task as ZoneTask<any>, -1);
804-
reEntryGuard &&
805-
(task as ZoneTask<any>)._transitionTo(notScheduled, running, notScheduled);
806-
}
807-
}
808-
_currentZoneFrame = _currentZoneFrame.parent;
809-
_currentTask = previousTask;
830+
this.afterRunTask(beforeRunTaskStatus, task);
810831
}
811832
}
812833

@@ -1241,10 +1262,13 @@ const Zone: ZoneType = (function(global: any) {
12411262
// we must bootstrap the initial task creation by manually scheduling the drain
12421263
if (_numberOfNestedTaskFrames === 0 && _microTaskQueue.length === 0) {
12431264
// We are not running in Task, so we need to kickstart the microtask queue.
1265+
// @JiaLiPassion, use native promise if async_hooks is available
12441266
if (global[symbolPromise]) {
12451267
global[symbolPromise].resolve(0)[symbolThen](drainMicroTaskQueue);
1246-
} else {
1268+
} else if (global[symbolSetTimeout]) {
12471269
global[symbolSetTimeout](drainMicroTaskQueue, 0);
1270+
} else {
1271+
Promise.resolve(0).then(drainMicroTaskQueue);
12481272
}
12491273
}
12501274
task && _microTaskQueue.push(task);
@@ -1294,7 +1318,13 @@ const Zone: ZoneType = (function(global: any) {
12941318
scheduleMicroTask: scheduleMicroTask,
12951319
showUncaughtError: () => !(Zone as any)[__symbol__('ignoreConsoleErrorUncaughtError')],
12961320
patchEventTargetMethods: () => false,
1297-
patchOnProperties: noop
1321+
patchOnProperties: noop,
1322+
beforeRunTask: (zone: Zone, task: Task) => {
1323+
return zone.beforeRunTask(task);
1324+
},
1325+
afterRunTask: (zone: Zone, beforeRunTaskStatus: BeforeRunTaskStatus, task: Task) => {
1326+
return zone.afterRunTask(beforeRunTaskStatus, task);
1327+
}
12981328
};
12991329
let _currentZoneFrame: _ZoneFrame = {parent: null, zone: new Zone(null, null)};
13001330
let _currentTask: Task = null;
@@ -1306,7 +1336,6 @@ const Zone: ZoneType = (function(global: any) {
13061336
return '__zone_symbol__' + name;
13071337
}
13081338

1309-
13101339
performanceMeasure('Zone', 'Zone');
13111340
return global['Zone'] = Zone;
13121341
})(typeof window !== 'undefined' && window || typeof self !== 'undefined' && self || global);

0 commit comments

Comments
 (0)