UNPKG

20 kBJavaScriptView Raw
1"use strict";
2var __importDefault = (this && this.__importDefault) || function (mod) {
3 return (mod && mod.__esModule) ? mod : { "default": mod };
4};
5Object.defineProperty(exports, "__esModule", { value: true });
6const assert_1 = __importDefault(require("assert"));
7const chrome_remote_interface_1 = __importDefault(require("chrome-remote-interface"));
8const istanbulize_1 = require("istanbulize");
9const node_inspector_server_1 = require("node-inspector-server");
10async function spawnInspected(file, args, options) {
11 const processCovs = [];
12 const srv = await node_inspector_server_1.InspectorServer.open();
13 return new Promise((resolve, reject) => {
14 srv
15 .subscribe(async (ev) => {
16 try {
17 // if (ev.rootProcess !== undefined && options.onRootProcess !== undefined) {
18 // options.onRootProcess(ev.rootProcess);
19 // }
20 // const args: ReadonlyArray<string> = ["--inspect=0", ...ev.args];
21 // const proxy: ChildProcessProxy = ev.proxySpawn(args);
22 // const debuggerPort: number = await getDebuggerPort(proxy);
23 const processCov = await getCoverage(ev.url, options.filter, options.timeout);
24 processCovs.push(processCov);
25 }
26 catch (err) {
27 reject(err);
28 }
29 }, reject, () => resolve(processCovs));
30 const child = srv.spawn(file, args, options);
31 if (options.onRootProcess !== undefined) {
32 options.onRootProcess(child);
33 }
34 child.on("close", () => {
35 srv.closeSync();
36 });
37 });
38}
39exports.spawnInspected = spawnInspected;
40async function getCoverage(url, filter, timeout) {
41 return new Promise(async (resolve, reject) => {
42 const timeoutId = timeout !== undefined ? setTimeout(onTimeout, timeout) : undefined;
43 let session;
44 let mainExecutionContextId;
45 const scriptIdToMeta = new Map();
46 let state = "WaitingForMainContext"; // TODO: enum
47 try {
48 session = await chrome_remote_interface_1.default({ target: url });
49 session.once("Runtime.executionContextCreated", onMainContextCreation);
50 session.on("Runtime.executionContextDestroyed", onContextDestruction);
51 session.on("Debugger.scriptParsed", onScriptParsed);
52 await session.Profiler.enable();
53 await session.Profiler.startPreciseCoverage({ callCount: true, detailed: true });
54 await session.Debugger.enable();
55 await session.Runtime.enable();
56 await session.Runtime.runIfWaitingForDebugger();
57 }
58 catch (err) {
59 removeListeners();
60 reject(err);
61 }
62 function onMainContextCreation(ev) {
63 assert_1.default(state === "WaitingForMainContext");
64 mainExecutionContextId = ev.context.id;
65 state = "WaitingForMainContextDestruction";
66 }
67 function onScriptParsed(ev) {
68 const collect = filter !== undefined ? filter(ev) : true;
69 if (collect) {
70 let sourceType = istanbulize_1.SourceType.Script;
71 if (ev.isModule !== undefined) {
72 sourceType = ev.isModule ? istanbulize_1.SourceType.Module : istanbulize_1.SourceType.Script;
73 }
74 let sourceMapUrl;
75 if (ev.sourceMapURL !== undefined && ev.sourceMapURL !== "") {
76 sourceMapUrl = ev.sourceMapURL;
77 }
78 scriptIdToMeta.set(ev.scriptId, {
79 sourceType,
80 sourceMapUrl,
81 });
82 }
83 }
84 async function onContextDestruction(ev) {
85 assert_1.default(state === "WaitingForMainContextDestruction");
86 if (ev.executionContextId !== mainExecutionContextId) {
87 return;
88 }
89 state = "WaitingForCoverage";
90 try {
91 // await session.Profiler.stopPreciseCoverage();
92 await session.HeapProfiler.collectGarbage();
93 const { result: scriptCovs } = await session.Profiler.takePreciseCoverage();
94 const result = [];
95 for (const scriptCov of scriptCovs) {
96 const meta = scriptIdToMeta.get(scriptCov.scriptId);
97 if (meta === undefined) {
98 // `undefined` means that the script was filtered out.
99 continue;
100 }
101 const { scriptSource } = await session.Debugger.getScriptSource({ scriptId: scriptCov.scriptId });
102 result.push(Object.assign(Object.assign(Object.assign({}, scriptCov), { sourceText: scriptSource }), meta));
103 }
104 resolve({ result });
105 }
106 catch (err) {
107 reject(err);
108 }
109 finally {
110 removeListeners();
111 }
112 }
113 function onTimeout() {
114 removeListeners();
115 reject(new Error("Unable to get V8 coverage (timeout)"));
116 }
117 function removeListeners() {
118 if (session === undefined) {
119 // Failure before the session is created
120 return;
121 }
122 session.removeListener("Runtime.executionContextCreated", onMainContextCreation);
123 session.removeListener("Runtime.executionContextDestroyed", onContextDestruction);
124 session.removeListener("Runtime.scriptParsed", onScriptParsed);
125 if (timeoutId !== undefined) {
126 clearTimeout(timeoutId);
127 }
128 session.close();
129 }
130 });
131}
132
133//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["_src/spawn-inspected.ts"],"names":[],"mappings":"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","file":"spawn-inspected.js","sourcesContent":["import { ProcessCov, ScriptCov } from \"@c88/v8-coverage\";\nimport assert from \"assert\";\nimport cp from \"child_process\";\nimport cri from \"chrome-remote-interface\";\nimport Protocol from \"devtools-protocol\";\nimport events from \"events\";\nimport { SourceType } from \"istanbulize\";\nimport { InspectorClient, InspectorServer } from \"node-inspector-server\";\nimport { CoverageFilter } from \"./filter\";\n\nexport interface ScriptMeta {\n  sourceText: string;\n  sourceType: SourceType;\n  sourceMapUrl?: string;\n}\n\nexport interface RichScriptCov extends ScriptCov, ScriptMeta {\n}\n\nexport interface RichProcessCov extends ProcessCov {\n  result: RichScriptCov[];\n}\n\nexport interface SpawnInspectedOptions extends cp.SpawnOptions {\n  filter?: CoverageFilter;\n\n  timeout?: number;\n\n  onRootProcess?(process: cp.ChildProcess): any;\n}\n\nexport async function spawnInspected(\n  file: string,\n  args: ReadonlyArray<string>,\n  options: SpawnInspectedOptions,\n): Promise<RichProcessCov[]> {\n  const processCovs: RichProcessCov[] = [];\n\n  const srv: InspectorServer = await InspectorServer.open();\n\n  return new Promise<RichProcessCov[]>((resolve, reject) => {\n    srv\n      .subscribe(\n        async (ev: InspectorClient) => {\n          try {\n            // if (ev.rootProcess !== undefined && options.onRootProcess !== undefined) {\n            //   options.onRootProcess(ev.rootProcess);\n            // }\n            // const args: ReadonlyArray<string> = [\"--inspect=0\", ...ev.args];\n            // const proxy: ChildProcessProxy = ev.proxySpawn(args);\n            // const debuggerPort: number = await getDebuggerPort(proxy);\n            const processCov: RichProcessCov = await getCoverage(ev.url, options.filter, options.timeout);\n            processCovs.push(processCov);\n          } catch (err) {\n            reject(err);\n          }\n        },\n        reject,\n        () => resolve(processCovs),\n      );\n\n    const child: cp.ChildProcess = srv.spawn(file, args, options);\n    if (options.onRootProcess !== undefined) {\n      options.onRootProcess(child);\n    }\n\n    child.on(\"close\", () => {\n      srv.closeSync();\n    });\n  });\n}\n\nasync function getCoverage(url: string, filter?: CoverageFilter, timeout?: number): Promise<RichProcessCov> {\n  return new Promise<RichProcessCov>(async (resolve, reject) => {\n    const timeoutId: NodeJS.Timer | undefined = timeout !== undefined ? setTimeout(onTimeout, timeout) : undefined;\n    let session: any;\n    let mainExecutionContextId: Protocol.Runtime.ExecutionContextId | undefined;\n    const scriptIdToMeta: Map<Protocol.Runtime.ScriptId, Partial<ScriptMeta>> = new Map();\n    let state: string = \"WaitingForMainContext\"; // TODO: enum\n    try {\n      session = await cri({target: url});\n      (session as any as events.EventEmitter).once(\"Runtime.executionContextCreated\", onMainContextCreation);\n      (session as any as events.EventEmitter).on(\"Runtime.executionContextDestroyed\", onContextDestruction);\n      (session as any as events.EventEmitter).on(\"Debugger.scriptParsed\", onScriptParsed);\n\n      await session.Profiler.enable();\n      await session.Profiler.startPreciseCoverage({callCount: true, detailed: true});\n      await session.Debugger.enable();\n      await session.Runtime.enable();\n      await session.Runtime.runIfWaitingForDebugger();\n    } catch (err) {\n      removeListeners();\n      reject(err);\n    }\n\n    function onMainContextCreation(ev: Protocol.Runtime.ExecutionContextCreatedEvent) {\n      assert(state === \"WaitingForMainContext\");\n      mainExecutionContextId = ev.context.id;\n      state = \"WaitingForMainContextDestruction\";\n    }\n\n    function onScriptParsed(ev: Protocol.Debugger.ScriptParsedEvent) {\n      const collect: boolean = filter !== undefined ? filter(ev) : true;\n      if (collect) {\n        let sourceType: SourceType = SourceType.Script;\n        if (ev.isModule !== undefined) {\n          sourceType = ev.isModule ? SourceType.Module : SourceType.Script;\n        }\n        let sourceMapUrl: string | undefined;\n        if (ev.sourceMapURL !== undefined && ev.sourceMapURL !== \"\") {\n          sourceMapUrl = ev.sourceMapURL;\n        }\n        scriptIdToMeta.set(\n          ev.scriptId,\n          {\n            sourceType,\n            sourceMapUrl,\n          },\n        );\n      }\n    }\n\n    async function onContextDestruction(ev: Protocol.Runtime.ExecutionContextDestroyedEvent): Promise<void> {\n      assert(state === \"WaitingForMainContextDestruction\");\n      if (ev.executionContextId !== mainExecutionContextId) {\n        return;\n      }\n      state = \"WaitingForCoverage\";\n\n      try {\n        // await session.Profiler.stopPreciseCoverage();\n        await session.HeapProfiler.collectGarbage();\n        const {result: scriptCovs} = await session.Profiler.takePreciseCoverage();\n        const result: RichScriptCov[] = [];\n        for (const scriptCov of scriptCovs) {\n          const meta: Partial<ScriptMeta> | undefined = scriptIdToMeta.get(scriptCov.scriptId);\n          if (meta === undefined) {\n            // `undefined` means that the script was filtered out.\n            continue;\n          }\n          const {scriptSource} = await session.Debugger.getScriptSource({scriptId: scriptCov.scriptId});\n          result.push({\n            ...scriptCov,\n            sourceText: scriptSource,\n            ...meta,\n          } as RichScriptCov);\n        }\n        resolve({result});\n      } catch (err) {\n        reject(err);\n      } finally {\n        removeListeners();\n      }\n    }\n\n    function onTimeout(): void {\n      removeListeners();\n      reject(new Error(\"Unable to get V8 coverage (timeout)\"));\n    }\n\n    function removeListeners(): void {\n      if (session === undefined) {\n        // Failure before the session is created\n        return;\n      }\n\n      (session as any as events.EventEmitter).removeListener(\"Runtime.executionContextCreated\", onMainContextCreation);\n      (session as any as events.EventEmitter).removeListener(\"Runtime.executionContextDestroyed\", onContextDestruction);\n      (session as any as events.EventEmitter).removeListener(\"Runtime.scriptParsed\", onScriptParsed);\n      if (timeoutId !== undefined) {\n        clearTimeout(timeoutId);\n      }\n      (session as any).close();\n    }\n  });\n}\n"],"sourceRoot":""}