UNPKG

19.6 kBJavaScriptView Raw
1import assert from "assert";
2import cri from "chrome-remote-interface";
3import { SourceType } from "istanbulize";
4import { InspectorServer } from "node-inspector-server";
5export 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}
34async 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":""}