1 | import assert from "assert";
|
2 | import cri from "chrome-remote-interface";
|
3 | import { SourceType } from "istanbulize";
|
4 | import { InspectorServer } from "node-inspector-server";
|
5 | export async function spawnInspected(file, args, options) {
|
6 | const processCovs = [];
|
7 | const srv = await InspectorServer.open();
|
8 | return new Promise((resolve, reject) => {
|
9 | srv
|
10 | .subscribe(async (ev) => {
|
11 | try {
|
12 | // if (ev.rootProcess !== undefined && options.onRootProcess !== undefined) {
|
13 | // options.onRootProcess(ev.rootProcess);
|
14 | // }
|
15 | // const args: ReadonlyArray<string> = ["--inspect=0", ...ev.args];
|
16 | // const proxy: ChildProcessProxy = ev.proxySpawn(args);
|
17 | // const debuggerPort: number = await getDebuggerPort(proxy);
|
18 | const processCov = await getCoverage(ev.url, options.filter, options.timeout);
|
19 | processCovs.push(processCov);
|
20 | }
|
21 | catch (err) {
|
22 | reject(err);
|
23 | }
|
24 | }, reject, () => resolve(processCovs));
|
25 | const child = srv.spawn(file, args, options);
|
26 | if (options.onRootProcess !== undefined) {
|
27 | options.onRootProcess(child);
|
28 | }
|
29 | child.on("close", () => {
|
30 | srv.closeSync();
|
31 | });
|
32 | });
|
33 | }
|
34 | async function getCoverage(url, filter, timeout) {
|
35 | return new Promise(async (resolve, reject) => {
|
36 | const timeoutId = timeout !== undefined ? setTimeout(onTimeout, timeout) : undefined;
|
37 | let session;
|
38 | let mainExecutionContextId;
|
39 | const scriptIdToMeta = new Map();
|
40 | let state = "WaitingForMainContext"; // TODO: enum
|
41 | try {
|
42 | session = await cri({ target: url });
|
43 | session.once("Runtime.executionContextCreated", onMainContextCreation);
|
44 | session.on("Runtime.executionContextDestroyed", onContextDestruction);
|
45 | session.on("Debugger.scriptParsed", onScriptParsed);
|
46 | await session.Profiler.enable();
|
47 | await session.Profiler.startPreciseCoverage({ callCount: true, detailed: true });
|
48 | await session.Debugger.enable();
|
49 | await session.Runtime.enable();
|
50 | await session.Runtime.runIfWaitingForDebugger();
|
51 | }
|
52 | catch (err) {
|
53 | removeListeners();
|
54 | reject(err);
|
55 | }
|
56 | function onMainContextCreation(ev) {
|
57 | assert(state === "WaitingForMainContext");
|
58 | mainExecutionContextId = ev.context.id;
|
59 | state = "WaitingForMainContextDestruction";
|
60 | }
|
61 | function onScriptParsed(ev) {
|
62 | const collect = filter !== undefined ? filter(ev) : true;
|
63 | if (collect) {
|
64 | let sourceType = SourceType.Script;
|
65 | if (ev.isModule !== undefined) {
|
66 | sourceType = ev.isModule ? SourceType.Module : SourceType.Script;
|
67 | }
|
68 | let sourceMapUrl;
|
69 | if (ev.sourceMapURL !== undefined && ev.sourceMapURL !== "") {
|
70 | sourceMapUrl = ev.sourceMapURL;
|
71 | }
|
72 | scriptIdToMeta.set(ev.scriptId, {
|
73 | sourceType,
|
74 | sourceMapUrl,
|
75 | });
|
76 | }
|
77 | }
|
78 | async function onContextDestruction(ev) {
|
79 | assert(state === "WaitingForMainContextDestruction");
|
80 | if (ev.executionContextId !== mainExecutionContextId) {
|
81 | return;
|
82 | }
|
83 | state = "WaitingForCoverage";
|
84 | try {
|
85 | // await session.Profiler.stopPreciseCoverage();
|
86 | await session.HeapProfiler.collectGarbage();
|
87 | const { result: scriptCovs } = await session.Profiler.takePreciseCoverage();
|
88 | const result = [];
|
89 | for (const scriptCov of scriptCovs) {
|
90 | const meta = scriptIdToMeta.get(scriptCov.scriptId);
|
91 | if (meta === undefined) {
|
92 | // `undefined` means that the script was filtered out.
|
93 | continue;
|
94 | }
|
95 | const { scriptSource } = await session.Debugger.getScriptSource({ scriptId: scriptCov.scriptId });
|
96 | result.push(Object.assign(Object.assign(Object.assign({}, scriptCov), { sourceText: scriptSource }), meta));
|
97 | }
|
98 | resolve({ result });
|
99 | }
|
100 | catch (err) {
|
101 | reject(err);
|
102 | }
|
103 | finally {
|
104 | removeListeners();
|
105 | }
|
106 | }
|
107 | function onTimeout() {
|
108 | removeListeners();
|
109 | reject(new Error("Unable to get V8 coverage (timeout)"));
|
110 | }
|
111 | function removeListeners() {
|
112 | if (session === undefined) {
|
113 | // Failure before the session is created
|
114 | return;
|
115 | }
|
116 | session.removeListener("Runtime.executionContextCreated", onMainContextCreation);
|
117 | session.removeListener("Runtime.executionContextDestroyed", onContextDestruction);
|
118 | session.removeListener("Runtime.scriptParsed", onScriptParsed);
|
119 | if (timeoutId !== undefined) {
|
120 | clearTimeout(timeoutId);
|
121 | }
|
122 | session.close();
|
123 | }
|
124 | });
|
125 | }
|
126 |
|
127 | //# 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":""}
|