UNPKG

23.3 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 demurgos_spawn_wrap_1 = require("demurgos-spawn-wrap");
9const istanbulize_1 = require("istanbulize");
10const DEBUGGER_URI_RE = /ws:\/\/.*?:(\d+)\//;
11// In milliseconds (1s)
12const GET_DEBUGGER_PORT_TIMEOUT = 1000;
13// In milliseconds (10s)
14const GET_COVERAGE_TIMEOUT = 10000;
15async function spawnInspected(file, args, options) {
16 const processCovs = [];
17 return new Promise((resolve, reject) => {
18 demurgos_spawn_wrap_1.observeSpawn(file, args, options)
19 .subscribe(async (ev) => {
20 try {
21 if (ev.rootProcess !== undefined && options.onRootProcess !== undefined) {
22 options.onRootProcess(ev.rootProcess);
23 }
24 const args = ["--inspect=0", ...ev.args];
25 const proxy = ev.proxySpawn(args);
26 const debuggerPort = await getDebuggerPort(proxy);
27 const processCov = await getCoverage(debuggerPort, options.filter);
28 processCovs.push(processCov);
29 }
30 catch (err) {
31 reject(err);
32 }
33 }, reject, () => resolve(processCovs));
34 });
35}
36exports.spawnInspected = spawnInspected;
37async function getDebuggerPort(proc) {
38 return new Promise((resolve, reject) => {
39 const timeoutId = setTimeout(onTimeout, GET_DEBUGGER_PORT_TIMEOUT * 100);
40 let stderrBuffer = Buffer.alloc(0);
41 proc.stderr.on("data", onStderrData);
42 proc.stderr.on("close", onClose);
43 function onStderrData(chunk) {
44 stderrBuffer = Buffer.concat([stderrBuffer, chunk]);
45 const stderrStr = stderrBuffer.toString("UTF-8");
46 const match = DEBUGGER_URI_RE.exec(stderrStr);
47 if (match === null) {
48 return;
49 }
50 const result = parseInt(match[1], 10);
51 removeListeners();
52 resolve(result);
53 }
54 function onClose(code, signal) {
55 removeListeners();
56 reject(new Error(`Unable to hook inspector (early exit, ${code}, ${signal})`));
57 }
58 function onTimeout() {
59 removeListeners();
60 reject(new Error("Unable to hook inspector (timeout)"));
61 // proc.kill();
62 }
63 function removeListeners() {
64 proc.stderr.removeListener("data", onStderrData);
65 proc.stderr.removeListener("close", onClose);
66 clearTimeout(timeoutId);
67 }
68 });
69}
70exports.getDebuggerPort = getDebuggerPort;
71async function getCoverage(port, filter) {
72 return new Promise(async (resolve, reject) => {
73 const timeoutId = setTimeout(onTimeout, GET_COVERAGE_TIMEOUT);
74 let client;
75 let mainExecutionContextId;
76 const scriptIdToMeta = new Map();
77 let state = "WaitingForMainContext"; // TODO: enum
78 try {
79 client = await chrome_remote_interface_1.default({ port });
80 await client.Profiler.enable();
81 await client.Profiler.startPreciseCoverage({ callCount: true, detailed: true });
82 await client.Debugger.enable();
83 client.once("Runtime.executionContextCreated", onMainContextCreation);
84 client.on("Runtime.executionContextDestroyed", onContextDestruction);
85 client.on("Debugger.scriptParsed", onScriptParsed);
86 await client.Runtime.enable();
87 }
88 catch (err) {
89 removeListeners();
90 reject(err);
91 }
92 function onMainContextCreation(ev) {
93 assert_1.default(state === "WaitingForMainContext");
94 mainExecutionContextId = ev.context.id;
95 state = "WaitingForMainContextDestruction";
96 }
97 function onScriptParsed(ev) {
98 const collect = filter !== undefined ? filter(ev) : true;
99 if (collect) {
100 let sourceType = istanbulize_1.SourceType.Script;
101 if (ev.isModule !== undefined) {
102 sourceType = ev.isModule ? istanbulize_1.SourceType.Module : istanbulize_1.SourceType.Script;
103 }
104 let sourceMapUrl;
105 if (ev.sourceMapURL !== undefined && ev.sourceMapURL !== "") {
106 sourceMapUrl = ev.sourceMapURL;
107 }
108 scriptIdToMeta.set(ev.scriptId, {
109 sourceType,
110 sourceMapUrl,
111 });
112 }
113 }
114 async function onContextDestruction(ev) {
115 assert_1.default(state === "WaitingForMainContextDestruction");
116 if (ev.executionContextId !== mainExecutionContextId) {
117 return;
118 }
119 state = "WaitingForCoverage";
120 try {
121 // await client.Profiler.stopPreciseCoverage();
122 await client.HeapProfiler.collectGarbage();
123 const { result: scriptCovs } = await client.Profiler.takePreciseCoverage();
124 const result = [];
125 for (const scriptCov of scriptCovs) {
126 const meta = scriptIdToMeta.get(scriptCov.scriptId);
127 if (meta === undefined) {
128 // `undefined` means that the script was filtered out.
129 continue;
130 }
131 const { scriptSource } = await client.Debugger.getScriptSource({ scriptId: scriptCov.scriptId });
132 result.push(Object.assign({}, scriptCov, { sourceText: scriptSource }, meta));
133 }
134 resolve({ result });
135 }
136 catch (err) {
137 reject(err);
138 }
139 finally {
140 removeListeners();
141 }
142 }
143 function onTimeout() {
144 removeListeners();
145 reject(new Error("Unable to get V8 coverage (timeout)"));
146 }
147 function removeListeners() {
148 client.removeListener("Runtime.executionContextCreated", onMainContextCreation);
149 client.removeListener("Runtime.executionContextDestroyed", onContextDestruction);
150 client.removeListener("Runtime.scriptParsed", onScriptParsed);
151 clearTimeout(timeoutId);
152 client.close();
153 }
154 });
155}
156
157//# 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 { ChildProcessProxy, observeSpawn, ObserveSpawnOptions, SpawnEvent } from \"demurgos-spawn-wrap\";\nimport Protocol from \"devtools-protocol\";\nimport events from \"events\";\nimport { SourceType } from \"istanbulize\";\nimport { CoverageFilter } from \"./filter\";\n\nconst DEBUGGER_URI_RE: RegExp = /ws:\\/\\/.*?:(\\d+)\\//;\n// In milliseconds (1s)\nconst GET_DEBUGGER_PORT_TIMEOUT: number = 1000;\n// In milliseconds (10s)\nconst GET_COVERAGE_TIMEOUT: number = 10000;\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 ObserveSpawnOptions {\n  filter?: CoverageFilter;\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  return new Promise<RichProcessCov[]>((resolve, reject) => {\n    observeSpawn(file, args, options)\n      .subscribe(\n        async (ev: SpawnEvent) => {\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(debuggerPort, options.filter);\n            processCovs.push(processCov);\n          } catch (err) {\n            reject(err);\n          }\n        },\n        reject,\n        () => resolve(processCovs),\n      );\n  });\n}\n\nexport async function getDebuggerPort(proc: ChildProcessProxy): Promise<number> {\n  return new Promise<number>((resolve, reject) => {\n    const timeoutId: NodeJS.Timer = setTimeout(onTimeout, GET_DEBUGGER_PORT_TIMEOUT * 100);\n    let stderrBuffer: Buffer = Buffer.alloc(0);\n    proc.stderr.on(\"data\", onStderrData);\n    proc.stderr.on(\"close\", onClose);\n\n    function onStderrData(chunk: Buffer): void {\n      stderrBuffer = Buffer.concat([stderrBuffer, chunk]);\n      const stderrStr: string = stderrBuffer.toString(\"UTF-8\");\n      const match: RegExpExecArray | null = DEBUGGER_URI_RE.exec(stderrStr);\n      if (match === null) {\n        return;\n      }\n      const result: number = parseInt(match[1], 10);\n      removeListeners();\n      resolve(result);\n    }\n\n    function onClose(code: number | null, signal: string | null): void {\n      removeListeners();\n      reject(new Error(`Unable to hook inspector (early exit, ${code}, ${signal})`));\n    }\n\n    function onTimeout(): void {\n      removeListeners();\n      reject(new Error(\"Unable to hook inspector (timeout)\"));\n      // proc.kill();\n    }\n\n    function removeListeners(): void {\n      proc.stderr.removeListener(\"data\", onStderrData);\n      proc.stderr.removeListener(\"close\", onClose);\n      clearTimeout(timeoutId);\n    }\n  });\n}\n\nasync function getCoverage(port: number, filter?: CoverageFilter): Promise<RichProcessCov> {\n  return new Promise<RichProcessCov>(async (resolve, reject) => {\n    const timeoutId: NodeJS.Timer = setTimeout(onTimeout, GET_COVERAGE_TIMEOUT);\n    let client: 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      client = await cri({port});\n\n      await client.Profiler.enable();\n      await client.Profiler.startPreciseCoverage({callCount: true, detailed: true});\n      await client.Debugger.enable();\n\n      (client as any as events.EventEmitter).once(\"Runtime.executionContextCreated\", onMainContextCreation);\n      (client as any as events.EventEmitter).on(\"Runtime.executionContextDestroyed\", onContextDestruction);\n      (client as any as events.EventEmitter).on(\"Debugger.scriptParsed\", onScriptParsed);\n\n      await client.Runtime.enable();\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 client.Profiler.stopPreciseCoverage();\n        await client.HeapProfiler.collectGarbage();\n        const {result: scriptCovs} = await client.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 client.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      (client as any as events.EventEmitter).removeListener(\"Runtime.executionContextCreated\", onMainContextCreation);\n      (client as any as events.EventEmitter).removeListener(\"Runtime.executionContextDestroyed\", onContextDestruction);\n      (client as any as events.EventEmitter).removeListener(\"Runtime.scriptParsed\", onScriptParsed);\n      clearTimeout(timeoutId);\n      (client as any).close();\n    }\n  });\n}\n"],"sourceRoot":""}