UNPKG

14.3 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.clearLeakDetector = exports.onAsyncHook = exports.stopAsyncTracing = exports.printHooks = exports.detectAsyncLeaks = exports.printAsyncStack = exports.trace = exports.startAsyncTracing = void 0;
4const asyncHooks = require("async_hooks");
5const util_1 = require("util");
6let hook;
7const asyncObjects = new Map();
8const objectMapping = new Map();
9function startAsyncTracing(stackTraces = false) {
10 hook = onAsyncHook(stackTraces);
11}
12exports.startAsyncTracing = startAsyncTracing;
13function trace(obj) {
14 let res = objectMapping.get(obj);
15 if (!res) {
16 // console.log(`trace: object not found: ${util.inspect(obj)}`);
17 return;
18 }
19 let asyncTrace = `== Tracing leaked object ${res.asyncId} ==`;
20 while (res) {
21 const { stack, ...rest } = res;
22 asyncTrace += `${(0, util_1.inspect)(rest)}\n${stack}`;
23 res = asyncObjects.get(res.triggerId);
24 }
25 console.log(asyncTrace);
26 return { obj, trace: asyncTrace };
27}
28exports.trace = trace;
29function printAsyncStack() {
30 console.log(`Async stack:`);
31 let res = asyncObjects.get(asyncHooks.executionAsyncId());
32 while (res) {
33 const { stack, ...rest } = res;
34 console.log(`%O\n${stack}`, rest);
35 res = asyncObjects.get(res.triggerId);
36 }
37}
38exports.printAsyncStack = printAsyncStack;
39function detectAsyncLeaks() {
40 const leaks = [];
41 process._getActiveHandles().forEach((h) => {
42 if (h !== process.stdout && h !== process.stderr) {
43 const leak = trace(h);
44 leak && leaks.push(leak);
45 }
46 });
47 process._getActiveRequests().forEach((h) => {
48 const leak = trace(h);
49 leak && leaks.push(leak);
50 });
51 return leaks;
52}
53exports.detectAsyncLeaks = detectAsyncLeaks;
54function printHooks() {
55 for (const obj of asyncObjects) {
56 console.log(`%O`, obj);
57 }
58}
59exports.printHooks = printHooks;
60function stopAsyncTracing() {
61 hook?.();
62}
63exports.stopAsyncTracing = stopAsyncTracing;
64function onAsyncHook(stackTraces) {
65 const hooks = {
66 init,
67 before,
68 after,
69 destroy,
70 promiseResolve
71 };
72 const asyncHook = asyncHooks.createHook(hooks);
73 asyncHook.enable();
74 return () => {
75 asyncHook.disable();
76 };
77 function init(asyncId, type, triggerId, resource) {
78 const obj = {
79 asyncId,
80 type,
81 triggerId,
82 resource,
83 state: "init",
84 startedCount: 0,
85 finishedCount: 0,
86 stack: stackTraces
87 ? new Error("stack:").stack.replace(/Error:/, "")
88 : undefined
89 };
90 asyncObjects.set(asyncId, obj);
91 objectMapping.set(resource, obj);
92 }
93 function destroy(asyncId) {
94 const obj = asyncObjects.get(asyncId);
95 if (obj) {
96 obj.state = "destroyed";
97 }
98 else {
99 // console.log(`destroyed: No obj ${asyncId}`);
100 }
101 }
102 function before(asyncId) {
103 const obj = asyncObjects.get(asyncId);
104 if (obj) {
105 obj.state = "before";
106 obj.startedCount++;
107 }
108 else {
109 // console.log(`before: No obj ${asyncId}`);
110 }
111 }
112 function after(asyncId) {
113 const obj = asyncObjects.get(asyncId);
114 if (obj) {
115 obj.state = "after";
116 obj.finishedCount++;
117 }
118 else {
119 // console.log(`after: No obj ${asyncId}`);
120 }
121 }
122 function promiseResolve(asyncId) {
123 const obj = asyncObjects.get(asyncId);
124 if (obj) {
125 obj.state = "resolved";
126 }
127 else {
128 // console.log(`resolved: No obj ${asyncId}`);
129 }
130 }
131}
132exports.onAsyncHook = onAsyncHook;
133function clearLeakDetector() {
134 asyncObjects.clear();
135 objectMapping.clear();
136}
137exports.clearLeakDetector = clearLeakDetector;
138//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"trace.js","sourceRoot":"","sources":["../../src/trace.ts"],"names":[],"mappings":"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","sourcesContent":["import * as asyncHooks from \"async_hooks\";\nimport { inspect } from \"util\";\n\nlet hook: () => void | undefined;\n\ntype AsyncState = \"init\" | \"before\" | \"after\" | \"destroyed\" | \"resolved\";\n\ninterface AsyncObject {\n    asyncId: number;\n    type: string;\n    triggerId: number;\n    resource: any;\n    state: AsyncState;\n    startedCount: number;\n    finishedCount: number;\n    stack?: string;\n}\n\nconst asyncObjects: Map<number, AsyncObject> = new Map();\nconst objectMapping: Map<object, AsyncObject> = new Map();\n\nexport function startAsyncTracing(stackTraces: boolean = false) {\n    hook = onAsyncHook(stackTraces);\n}\n\ninterface Trace {\n    obj: object;\n    trace: string;\n}\n\nexport function trace(obj: object): Trace | void {\n    let res = objectMapping.get(obj);\n    if (!res) {\n        // console.log(`trace: object not found: ${util.inspect(obj)}`);\n        return;\n    }\n    let asyncTrace = `== Tracing leaked object ${res.asyncId} ==`;\n    while (res) {\n        const { stack, ...rest } = res;\n        asyncTrace += `${inspect(rest)}\\n${stack}`;\n        res = asyncObjects.get(res.triggerId);\n    }\n    console.log(asyncTrace);\n    return { obj, trace: asyncTrace };\n}\n\nexport function printAsyncStack() {\n    console.log(`Async stack:`);\n    let res = asyncObjects.get(asyncHooks.executionAsyncId());\n    while (res) {\n        const { stack, ...rest } = res;\n        console.log(`%O\\n${stack}`, rest);\n        res = asyncObjects.get(res.triggerId);\n    }\n}\n\nexport function detectAsyncLeaks(): object[] {\n    const leaks: Trace[] = [];\n    (process as any)._getActiveHandles().forEach((h: object) => {\n        if (h !== process.stdout && h !== process.stderr) {\n            const leak = trace(h);\n            leak && leaks.push(leak);\n        }\n    });\n    (process as any)._getActiveRequests().forEach((h: object) => {\n        const leak = trace(h);\n        leak && leaks.push(leak);\n    });\n    return leaks;\n}\n\nexport function printHooks() {\n    for (const obj of asyncObjects) {\n        console.log(`%O`, obj);\n    }\n}\n\nexport function stopAsyncTracing() {\n    hook?.();\n}\n\nexport function onAsyncHook(stackTraces: boolean) {\n    const hooks: asyncHooks.HookCallbacks = {\n        init,\n        before,\n        after,\n        destroy,\n        promiseResolve\n    };\n\n    const asyncHook = asyncHooks.createHook(hooks);\n    asyncHook.enable();\n\n    return () => {\n        asyncHook.disable();\n    };\n\n    function init(asyncId: number, type: string, triggerId: number, resource: object) {\n        const obj: AsyncObject = {\n            asyncId,\n            type,\n            triggerId,\n            resource,\n            state: \"init\",\n            startedCount: 0,\n            finishedCount: 0,\n            stack: stackTraces\n                ? new Error(\"stack:\").stack!.replace(/Error:/, \"\")\n                : undefined\n        };\n        asyncObjects.set(asyncId, obj);\n        objectMapping.set(resource, obj);\n    }\n    function destroy(asyncId: number) {\n        const obj = asyncObjects.get(asyncId);\n        if (obj) {\n            obj.state = \"destroyed\";\n        } else {\n            // console.log(`destroyed: No obj ${asyncId}`);\n        }\n    }\n    function before(asyncId: number) {\n        const obj = asyncObjects.get(asyncId);\n        if (obj) {\n            obj.state = \"before\";\n            obj.startedCount++;\n        } else {\n            // console.log(`before: No obj ${asyncId}`);\n        }\n    }\n    function after(asyncId: number) {\n        const obj = asyncObjects.get(asyncId);\n        if (obj) {\n            obj.state = \"after\";\n            obj.finishedCount++;\n        } else {\n            // console.log(`after: No obj ${asyncId}`);\n        }\n    }\n    function promiseResolve(asyncId: number) {\n        const obj = asyncObjects.get(asyncId);\n        if (obj) {\n            obj.state = \"resolved\";\n        } else {\n            // console.log(`resolved: No obj ${asyncId}`);\n        }\n    }\n}\n\nexport function clearLeakDetector() {\n    asyncObjects.clear();\n    objectMapping.clear();\n}\n"]}
\No newline at end of file