@rpcbase/test 0.317.0 → 0.318.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/coverage/console-text-report.d.ts +4 -1
- package/dist/coverage/console-text-report.d.ts.map +1 -1
- package/dist/coverage/console-text-report.js +107 -7
- package/dist/coverage/console-text-report.js.map +1 -1
- package/dist/coverage/report.js +2 -2
- package/dist/coverage/report.js.map +1 -1
- package/package.json +1 -1
|
@@ -1,2 +1,5 @@
|
|
|
1
|
-
export declare function generateConsoleTextCoverageReport(reports: any, context: any):
|
|
1
|
+
export declare function generateConsoleTextCoverageReport(reports: any, context: any): {
|
|
2
|
+
output: string;
|
|
3
|
+
summarizer: string | undefined;
|
|
4
|
+
};
|
|
2
5
|
//# sourceMappingURL=console-text-report.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"console-text-report.d.ts","sourceRoot":"","sources":["../../src/coverage/console-text-report.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"console-text-report.d.ts","sourceRoot":"","sources":["../../src/coverage/console-text-report.ts"],"names":[],"mappings":"AAMA,wBAAgB,iCAAiC,CAC/C,OAAO,EAAE,GAAG,EACZ,OAAO,EAAE,GAAG,GACX;IAAE,MAAM,EAAE,MAAM,CAAC;IAAC,UAAU,EAAE,MAAM,GAAG,SAAS,CAAA;CAAE,CAWpD"}
|
|
@@ -1,14 +1,15 @@
|
|
|
1
|
+
import path from "node:path";
|
|
1
2
|
const DEFAULT_CONSOLE_MAX_COLS = 80;
|
|
2
|
-
const CONSOLE_SUMMARIZER = "
|
|
3
|
+
const CONSOLE_SUMMARIZER = "rbCwd";
|
|
3
4
|
function generateConsoleTextCoverageReport(reports, context) {
|
|
4
5
|
const debug = resolveConsoleReportDebugInfo();
|
|
5
|
-
const summarizer =
|
|
6
|
+
const summarizer = registerConsoleSummarizer(context);
|
|
6
7
|
writeConsoleReportDebugLog({ ...debug, summarizer });
|
|
7
8
|
const reportOutput = createConsoleTextReportOutput(reports, context, {
|
|
8
9
|
maxCols: debug.maxCols + 20,
|
|
9
10
|
summarizer
|
|
10
11
|
});
|
|
11
|
-
return stripUncoveredLineNumbersColumn(reportOutput);
|
|
12
|
+
return { output: stripUncoveredLineNumbersColumn(reportOutput), summarizer };
|
|
12
13
|
}
|
|
13
14
|
function resolveConsoleReportDebugInfo() {
|
|
14
15
|
const isTTY = Boolean(process.stdout.isTTY);
|
|
@@ -74,7 +75,7 @@ function stripUncoveredLineNumbersColumn(output) {
|
|
|
74
75
|
});
|
|
75
76
|
return filtered.join("\n");
|
|
76
77
|
}
|
|
77
|
-
function
|
|
78
|
+
function registerConsoleSummarizer(context) {
|
|
78
79
|
const summarizerFactory = context?._summarizerFactory;
|
|
79
80
|
if (!summarizerFactory || typeof summarizerFactory !== "object") {
|
|
80
81
|
return void 0;
|
|
@@ -87,11 +88,12 @@ function registerCollapsedPkgSummarizer(context) {
|
|
|
87
88
|
if (!pkgRoot) {
|
|
88
89
|
return void 0;
|
|
89
90
|
}
|
|
90
|
-
summarizerFactory[CONSOLE_SUMMARIZER] =
|
|
91
|
+
summarizerFactory[CONSOLE_SUMMARIZER] = createCollapsedCwdTree(pkgRoot, process.cwd());
|
|
91
92
|
return CONSOLE_SUMMARIZER;
|
|
92
93
|
}
|
|
93
|
-
function
|
|
94
|
-
const
|
|
94
|
+
function createCollapsedCwdTree(root, cwd) {
|
|
95
|
+
const treeRoot = createCwdTreeFromCoverage(root, cwd);
|
|
96
|
+
const clonedRoot = cloneReportNode(treeRoot, null);
|
|
95
97
|
return {
|
|
96
98
|
getRoot: () => clonedRoot,
|
|
97
99
|
visit: (visitor, state) => {
|
|
@@ -102,6 +104,104 @@ function createCollapsedPkgTree(root) {
|
|
|
102
104
|
}
|
|
103
105
|
};
|
|
104
106
|
}
|
|
107
|
+
function createCwdTreeFromCoverage(root, cwd) {
|
|
108
|
+
const proto = Object.getPrototypeOf(root);
|
|
109
|
+
const pathCtor = root?.path?.constructor;
|
|
110
|
+
const fileCoverages = [];
|
|
111
|
+
collectFileCoverages(root, fileCoverages);
|
|
112
|
+
const treeRoot = Object.create(proto);
|
|
113
|
+
treeRoot.path = pathCtor ? new pathCtor([]) : root.path;
|
|
114
|
+
treeRoot.fileCoverage = null;
|
|
115
|
+
treeRoot.parent = null;
|
|
116
|
+
treeRoot.children = [];
|
|
117
|
+
const dirNodes = /* @__PURE__ */ new Map();
|
|
118
|
+
dirNodes.set("", treeRoot);
|
|
119
|
+
for (const fileCoverage of fileCoverages) {
|
|
120
|
+
const absolutePath = path.resolve(String(fileCoverage?.path ?? ""));
|
|
121
|
+
if (!absolutePath) {
|
|
122
|
+
continue;
|
|
123
|
+
}
|
|
124
|
+
let relativePath = toPosix(path.relative(cwd, absolutePath));
|
|
125
|
+
if (!relativePath) {
|
|
126
|
+
relativePath = toPosix(absolutePath);
|
|
127
|
+
}
|
|
128
|
+
if (isInsideCwd(relativePath) && (relativePath === "src" || relativePath.startsWith("src/"))) {
|
|
129
|
+
relativePath = relativePath === "src" ? "" : relativePath.slice("src/".length);
|
|
130
|
+
}
|
|
131
|
+
if (!relativePath) {
|
|
132
|
+
relativePath = toPosix(path.basename(absolutePath));
|
|
133
|
+
}
|
|
134
|
+
const segments = relativePath.split("/").filter(Boolean);
|
|
135
|
+
if (segments.length === 0) {
|
|
136
|
+
continue;
|
|
137
|
+
}
|
|
138
|
+
let parent = treeRoot;
|
|
139
|
+
let prefix = "";
|
|
140
|
+
for (let i = 0; i < segments.length - 1; i += 1) {
|
|
141
|
+
prefix = prefix ? `${prefix}/${segments[i]}` : segments[i];
|
|
142
|
+
const existing = dirNodes.get(prefix);
|
|
143
|
+
if (existing) {
|
|
144
|
+
parent = existing;
|
|
145
|
+
continue;
|
|
146
|
+
}
|
|
147
|
+
const dirNode = Object.create(proto);
|
|
148
|
+
dirNode.path = pathCtor ? new pathCtor(prefix.split("/")) : root.path;
|
|
149
|
+
dirNode.fileCoverage = null;
|
|
150
|
+
dirNode.parent = parent;
|
|
151
|
+
dirNode.children = [];
|
|
152
|
+
parent.children.push(dirNode);
|
|
153
|
+
dirNodes.set(prefix, dirNode);
|
|
154
|
+
parent = dirNode;
|
|
155
|
+
}
|
|
156
|
+
const fileNode = Object.create(proto);
|
|
157
|
+
fileNode.path = pathCtor ? new pathCtor(segments) : root.path;
|
|
158
|
+
fileNode.fileCoverage = fileCoverage;
|
|
159
|
+
fileNode.parent = parent;
|
|
160
|
+
fileNode.children = [];
|
|
161
|
+
parent.children.push(fileNode);
|
|
162
|
+
}
|
|
163
|
+
sortReportTree(treeRoot);
|
|
164
|
+
return treeRoot;
|
|
165
|
+
}
|
|
166
|
+
function collectFileCoverages(node, out) {
|
|
167
|
+
if (!node || typeof node.isSummary !== "function") {
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
if (!node.isSummary()) {
|
|
171
|
+
const fileCoverage = typeof node.getFileCoverage === "function" ? node.getFileCoverage() : node.fileCoverage;
|
|
172
|
+
if (fileCoverage) {
|
|
173
|
+
out.push(fileCoverage);
|
|
174
|
+
}
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
const children = typeof node.getChildren === "function" ? node.getChildren() : [];
|
|
178
|
+
for (const child of children) {
|
|
179
|
+
collectFileCoverages(child, out);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
function sortReportTree(node) {
|
|
183
|
+
const children = Array.isArray(node?.children) ? node.children : [];
|
|
184
|
+
if (children.length === 0) {
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
children.sort((a, b) => {
|
|
188
|
+
const aKey = typeof a?.path?.toString === "function" ? a.path.toString() : "";
|
|
189
|
+
const bKey = typeof b?.path?.toString === "function" ? b.path.toString() : "";
|
|
190
|
+
return aKey < bKey ? -1 : aKey > bKey ? 1 : 0;
|
|
191
|
+
});
|
|
192
|
+
for (const child of children) {
|
|
193
|
+
sortReportTree(child);
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
function isInsideCwd(relativePosix) {
|
|
197
|
+
if (!relativePosix) {
|
|
198
|
+
return false;
|
|
199
|
+
}
|
|
200
|
+
return !(relativePosix === ".." || relativePosix.startsWith("../"));
|
|
201
|
+
}
|
|
202
|
+
function toPosix(input) {
|
|
203
|
+
return String(input ?? "").split(path.sep).join("/");
|
|
204
|
+
}
|
|
105
205
|
function toFullVisitor(visitor) {
|
|
106
206
|
const call = (method) => (node, state) => {
|
|
107
207
|
if (typeof visitor?.[method] === "function") {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"console-text-report.js","sources":["../../src/coverage/console-text-report.ts"],"sourcesContent":["const DEFAULT_CONSOLE_MAX_COLS = 80\nconst CONSOLE_SUMMARIZER = \"collapsedPkg\"\n\nexport function generateConsoleTextCoverageReport(reports: any, context: any): string {\n const debug = resolveConsoleReportDebugInfo()\n const summarizer = registerCollapsedPkgSummarizer(context)\n writeConsoleReportDebugLog({ ...debug, summarizer })\n\n const reportOutput = createConsoleTextReportOutput(reports, context, {\n maxCols: debug.maxCols + 20,\n summarizer,\n })\n\n return stripUncoveredLineNumbersColumn(reportOutput)\n}\n\nfunction resolveConsoleReportDebugInfo(): {\n isTTY: boolean\n stdoutColumns: number | undefined\n envColumns: string | undefined\n maxCols: number\n} {\n const isTTY = Boolean(process.stdout.isTTY)\n const stdoutColumns = process.stdout.columns\n const envColumns = process.env.COLUMNS\n const maxCols = resolveConsoleMaxCols(stdoutColumns, envColumns)\n\n return { isTTY, stdoutColumns, envColumns, maxCols }\n}\n\nfunction resolveConsoleMaxCols(stdoutColumns: number | undefined, envColumns: string | undefined): number {\n if (typeof stdoutColumns === \"number\" && Number.isFinite(stdoutColumns) && stdoutColumns > 0) {\n return stdoutColumns\n }\n\n const parsed = typeof envColumns === \"string\" ? Number.parseInt(envColumns, 10) : Number.NaN\n if (Number.isFinite(parsed) && parsed > 0) {\n return parsed\n }\n\n return DEFAULT_CONSOLE_MAX_COLS\n}\n\nfunction writeConsoleReportDebugLog({\n isTTY,\n stdoutColumns,\n envColumns,\n maxCols,\n summarizer,\n}: {\n isTTY: boolean\n stdoutColumns: number | undefined\n envColumns: string | undefined\n maxCols: number\n summarizer: string | undefined\n}): void {\n const message = [\n \"[coverage] debug\",\n `stdout.isTTY=${isTTY ? \"1\" : \"0\"}`,\n `stdout.columns=${stdoutColumns ?? \"undefined\"}`,\n `env.COLUMNS=${envColumns ?? \"undefined\"}`,\n `resolvedMaxCols=${maxCols}`,\n `summarizer=${summarizer ?? \"undefined\"}`,\n ].join(\" \")\n\n process.stderr.write(`${message}\\n`)\n}\n\nfunction createConsoleTextReportOutput(\n reports: any,\n context: any,\n options: { maxCols: number; summarizer?: string },\n): string {\n const fileWriter = context?.writer\n const fileWriterCtor = fileWriter?.constructor\n\n if (\n !fileWriterCtor\n || typeof fileWriterCtor.startCapture !== \"function\"\n || typeof fileWriterCtor.stopCapture !== \"function\"\n || typeof fileWriterCtor.getOutput !== \"function\"\n || typeof fileWriterCtor.resetOutput !== \"function\"\n ) {\n reports.create(\"text\", { maxCols: options.maxCols, summarizer: options.summarizer }).execute(context)\n return \"\"\n }\n\n fileWriterCtor.resetOutput()\n fileWriterCtor.startCapture()\n try {\n reports.create(\"text\", { maxCols: options.maxCols, summarizer: options.summarizer }).execute(context)\n } finally {\n fileWriterCtor.stopCapture()\n }\n const output = fileWriterCtor.getOutput()\n fileWriterCtor.resetOutput()\n return output\n}\n\nfunction stripUncoveredLineNumbersColumn(output: string): string {\n const lines = output.split(\"\\n\")\n const filtered = lines.map((line) => {\n const parts = line.split(\"|\")\n if (parts.length < 6) {\n return line.trimEnd()\n }\n return parts.slice(0, 5).join(\"|\").trimEnd()\n })\n return filtered.join(\"\\n\")\n}\n\nfunction registerCollapsedPkgSummarizer(context: any): string | undefined {\n const summarizerFactory = context?._summarizerFactory\n if (!summarizerFactory || typeof summarizerFactory !== \"object\") {\n return undefined\n }\n\n if (CONSOLE_SUMMARIZER in summarizerFactory) {\n return CONSOLE_SUMMARIZER\n }\n\n const pkgTree = context.getTree?.(\"pkg\")\n const pkgRoot = pkgTree?.getRoot?.()\n if (!pkgRoot) {\n return undefined\n }\n\n summarizerFactory[CONSOLE_SUMMARIZER] = createCollapsedPkgTree(pkgRoot)\n return CONSOLE_SUMMARIZER\n}\n\nfunction createCollapsedPkgTree(root: any): { getRoot: () => any; visit: (visitor: any, state: any) => void } {\n const clonedRoot = cloneReportNode(root, null)\n return {\n getRoot: () => clonedRoot,\n visit: (visitor: any, state: any) => {\n const fullVisitor = toFullVisitor(visitor)\n fullVisitor.onStart(clonedRoot, state)\n clonedRoot.visit(fullVisitor, state)\n fullVisitor.onEnd(clonedRoot, state)\n },\n }\n}\n\nfunction toFullVisitor(visitor: any): {\n onStart: (node: any, state: any) => void\n onEnd: (node: any, state: any) => void\n onSummary: (node: any, state: any) => void\n onSummaryEnd: (node: any, state: any) => void\n onDetail: (node: any, state: any) => void\n} {\n const call = (method: string) => (node: any, state: any) => {\n if (typeof visitor?.[method] === \"function\") {\n visitor[method](node, state)\n }\n }\n\n return {\n onStart: call(\"onStart\"),\n onEnd: call(\"onEnd\"),\n onSummary: call(\"onSummary\"),\n onSummaryEnd: call(\"onSummaryEnd\"),\n onDetail: call(\"onDetail\"),\n }\n}\n\nfunction cloneReportNode(node: any, parent: any): any {\n const clone = Object.create(Object.getPrototypeOf(node))\n clone.path = node.path\n clone.fileCoverage = node.fileCoverage\n clone.parent = parent\n\n const children: any[] = typeof node.getChildren === \"function\" ? node.getChildren() : []\n clone.children = children.flatMap((child) => cloneOrCollapseNode(child, clone))\n\n return clone\n}\n\nfunction cloneOrCollapseNode(node: any, parent: any): any[] {\n if (\n typeof node?.isSummary === \"function\"\n && node.isSummary()\n && typeof node.isRoot === \"function\"\n && !node.isRoot()\n && typeof node.getChildren === \"function\"\n ) {\n const children = node.getChildren()\n if (\n Array.isArray(children)\n && children.length === 1\n && typeof children[0]?.isSummary === \"function\"\n && !children[0].isSummary()\n ) {\n return [cloneReportNode(children[0], parent)]\n }\n }\n\n return [cloneReportNode(node, parent)]\n}\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"console-text-report.js","sources":["../../src/coverage/console-text-report.ts"],"sourcesContent":["import path from \"node:path\"\n\n\nconst DEFAULT_CONSOLE_MAX_COLS = 80\nconst CONSOLE_SUMMARIZER = \"rbCwd\"\n\nexport function generateConsoleTextCoverageReport(\n reports: any,\n context: any,\n): { output: string; summarizer: string | undefined } {\n const debug = resolveConsoleReportDebugInfo()\n const summarizer = registerConsoleSummarizer(context)\n writeConsoleReportDebugLog({ ...debug, summarizer })\n\n const reportOutput = createConsoleTextReportOutput(reports, context, {\n maxCols: debug.maxCols + 20,\n summarizer,\n })\n\n return { output: stripUncoveredLineNumbersColumn(reportOutput), summarizer }\n}\n\nfunction resolveConsoleReportDebugInfo(): {\n isTTY: boolean\n stdoutColumns: number | undefined\n envColumns: string | undefined\n maxCols: number\n} {\n const isTTY = Boolean(process.stdout.isTTY)\n const stdoutColumns = process.stdout.columns\n const envColumns = process.env.COLUMNS\n const maxCols = resolveConsoleMaxCols(stdoutColumns, envColumns)\n\n return { isTTY, stdoutColumns, envColumns, maxCols }\n}\n\nfunction resolveConsoleMaxCols(stdoutColumns: number | undefined, envColumns: string | undefined): number {\n if (typeof stdoutColumns === \"number\" && Number.isFinite(stdoutColumns) && stdoutColumns > 0) {\n return stdoutColumns\n }\n\n const parsed = typeof envColumns === \"string\" ? Number.parseInt(envColumns, 10) : Number.NaN\n if (Number.isFinite(parsed) && parsed > 0) {\n return parsed\n }\n\n return DEFAULT_CONSOLE_MAX_COLS\n}\n\nfunction writeConsoleReportDebugLog({\n isTTY,\n stdoutColumns,\n envColumns,\n maxCols,\n summarizer,\n}: {\n isTTY: boolean\n stdoutColumns: number | undefined\n envColumns: string | undefined\n maxCols: number\n summarizer: string | undefined\n}): void {\n const message = [\n \"[coverage] debug\",\n `stdout.isTTY=${isTTY ? \"1\" : \"0\"}`,\n `stdout.columns=${stdoutColumns ?? \"undefined\"}`,\n `env.COLUMNS=${envColumns ?? \"undefined\"}`,\n `resolvedMaxCols=${maxCols}`,\n `summarizer=${summarizer ?? \"undefined\"}`,\n ].join(\" \")\n\n process.stderr.write(`${message}\\n`)\n}\n\nfunction createConsoleTextReportOutput(\n reports: any,\n context: any,\n options: { maxCols: number; summarizer?: string },\n): string {\n const fileWriter = context?.writer\n const fileWriterCtor = fileWriter?.constructor\n\n if (\n !fileWriterCtor\n || typeof fileWriterCtor.startCapture !== \"function\"\n || typeof fileWriterCtor.stopCapture !== \"function\"\n || typeof fileWriterCtor.getOutput !== \"function\"\n || typeof fileWriterCtor.resetOutput !== \"function\"\n ) {\n reports.create(\"text\", { maxCols: options.maxCols, summarizer: options.summarizer }).execute(context)\n return \"\"\n }\n\n fileWriterCtor.resetOutput()\n fileWriterCtor.startCapture()\n try {\n reports.create(\"text\", { maxCols: options.maxCols, summarizer: options.summarizer }).execute(context)\n } finally {\n fileWriterCtor.stopCapture()\n }\n const output = fileWriterCtor.getOutput()\n fileWriterCtor.resetOutput()\n return output\n}\n\nfunction stripUncoveredLineNumbersColumn(output: string): string {\n const lines = output.split(\"\\n\")\n const filtered = lines.map((line) => {\n const parts = line.split(\"|\")\n if (parts.length < 6) {\n return line.trimEnd()\n }\n return parts.slice(0, 5).join(\"|\").trimEnd()\n })\n return filtered.join(\"\\n\")\n}\n\nfunction registerConsoleSummarizer(context: any): string | undefined {\n const summarizerFactory = context?._summarizerFactory\n if (!summarizerFactory || typeof summarizerFactory !== \"object\") {\n return undefined\n }\n\n if (CONSOLE_SUMMARIZER in summarizerFactory) {\n return CONSOLE_SUMMARIZER\n }\n\n const pkgTree = context.getTree?.(\"pkg\")\n const pkgRoot = pkgTree?.getRoot?.()\n if (!pkgRoot) {\n return undefined\n }\n\n summarizerFactory[CONSOLE_SUMMARIZER] = createCollapsedCwdTree(pkgRoot, process.cwd())\n return CONSOLE_SUMMARIZER\n}\n\nfunction createCollapsedCwdTree(\n root: any,\n cwd: string,\n): { getRoot: () => any; visit: (visitor: any, state: any) => void } {\n const treeRoot = createCwdTreeFromCoverage(root, cwd)\n const clonedRoot = cloneReportNode(treeRoot, null)\n return {\n getRoot: () => clonedRoot,\n visit: (visitor: any, state: any) => {\n const fullVisitor = toFullVisitor(visitor)\n fullVisitor.onStart(clonedRoot, state)\n clonedRoot.visit(fullVisitor, state)\n fullVisitor.onEnd(clonedRoot, state)\n },\n }\n}\n\nfunction createCwdTreeFromCoverage(root: any, cwd: string): any {\n const proto = Object.getPrototypeOf(root)\n const pathCtor = root?.path?.constructor\n\n const fileCoverages: any[] = []\n collectFileCoverages(root, fileCoverages)\n\n const treeRoot = Object.create(proto)\n treeRoot.path = pathCtor ? new pathCtor([]) : root.path\n treeRoot.fileCoverage = null\n treeRoot.parent = null\n treeRoot.children = []\n\n const dirNodes = new Map<string, any>()\n dirNodes.set(\"\", treeRoot)\n\n for (const fileCoverage of fileCoverages) {\n const absolutePath = path.resolve(String(fileCoverage?.path ?? \"\"))\n if (!absolutePath) {\n continue\n }\n\n let relativePath = toPosix(path.relative(cwd, absolutePath))\n if (!relativePath) {\n relativePath = toPosix(absolutePath)\n }\n\n if (isInsideCwd(relativePath) && (relativePath === \"src\" || relativePath.startsWith(\"src/\"))) {\n relativePath = relativePath === \"src\" ? \"\" : relativePath.slice(\"src/\".length)\n }\n\n if (!relativePath) {\n relativePath = toPosix(path.basename(absolutePath))\n }\n\n const segments = relativePath.split(\"/\").filter(Boolean)\n if (segments.length === 0) {\n continue\n }\n\n let parent = treeRoot\n let prefix = \"\"\n\n for (let i = 0; i < segments.length - 1; i += 1) {\n prefix = prefix ? `${prefix}/${segments[i]}` : segments[i]\n const existing = dirNodes.get(prefix)\n if (existing) {\n parent = existing\n continue\n }\n\n const dirNode = Object.create(proto)\n dirNode.path = pathCtor ? new pathCtor(prefix.split(\"/\")) : root.path\n dirNode.fileCoverage = null\n dirNode.parent = parent\n dirNode.children = []\n parent.children.push(dirNode)\n dirNodes.set(prefix, dirNode)\n parent = dirNode\n }\n\n const fileNode = Object.create(proto)\n fileNode.path = pathCtor ? new pathCtor(segments) : root.path\n fileNode.fileCoverage = fileCoverage\n fileNode.parent = parent\n fileNode.children = []\n parent.children.push(fileNode)\n }\n\n sortReportTree(treeRoot)\n\n return treeRoot\n}\n\nfunction collectFileCoverages(node: any, out: any[]): void {\n if (!node || typeof node.isSummary !== \"function\") {\n return\n }\n\n if (!node.isSummary()) {\n const fileCoverage = typeof node.getFileCoverage === \"function\" ? node.getFileCoverage() : node.fileCoverage\n if (fileCoverage) {\n out.push(fileCoverage)\n }\n return\n }\n\n const children: any[] = typeof node.getChildren === \"function\" ? node.getChildren() : []\n for (const child of children) {\n collectFileCoverages(child, out)\n }\n}\n\nfunction sortReportTree(node: any): void {\n const children: any[] = Array.isArray(node?.children) ? node.children : []\n if (children.length === 0) {\n return\n }\n\n children.sort((a, b) => {\n const aKey = typeof a?.path?.toString === \"function\" ? a.path.toString() : \"\"\n const bKey = typeof b?.path?.toString === \"function\" ? b.path.toString() : \"\"\n return aKey < bKey ? -1 : aKey > bKey ? 1 : 0\n })\n\n for (const child of children) {\n sortReportTree(child)\n }\n}\n\nfunction isInsideCwd(relativePosix: string): boolean {\n if (!relativePosix) {\n return false\n }\n return !(relativePosix === \"..\" || relativePosix.startsWith(\"../\"))\n}\n\nfunction toPosix(input: string): string {\n return String(input ?? \"\").split(path.sep).join(\"/\")\n}\n\nfunction toFullVisitor(visitor: any): {\n onStart: (node: any, state: any) => void\n onEnd: (node: any, state: any) => void\n onSummary: (node: any, state: any) => void\n onSummaryEnd: (node: any, state: any) => void\n onDetail: (node: any, state: any) => void\n} {\n const call = (method: string) => (node: any, state: any) => {\n if (typeof visitor?.[method] === \"function\") {\n visitor[method](node, state)\n }\n }\n\n return {\n onStart: call(\"onStart\"),\n onEnd: call(\"onEnd\"),\n onSummary: call(\"onSummary\"),\n onSummaryEnd: call(\"onSummaryEnd\"),\n onDetail: call(\"onDetail\"),\n }\n}\n\nfunction cloneReportNode(node: any, parent: any): any {\n const clone = Object.create(Object.getPrototypeOf(node))\n clone.path = node.path\n clone.fileCoverage = node.fileCoverage\n clone.parent = parent\n\n const children: any[] = typeof node.getChildren === \"function\" ? node.getChildren() : []\n clone.children = children.flatMap((child) => cloneOrCollapseNode(child, clone))\n\n return clone\n}\n\nfunction cloneOrCollapseNode(node: any, parent: any): any[] {\n if (\n typeof node?.isSummary === \"function\"\n && node.isSummary()\n && typeof node.isRoot === \"function\"\n && !node.isRoot()\n && typeof node.getChildren === \"function\"\n ) {\n const children = node.getChildren()\n if (\n Array.isArray(children)\n && children.length === 1\n && typeof children[0]?.isSummary === \"function\"\n && !children[0].isSummary()\n ) {\n return [cloneReportNode(children[0], parent)]\n }\n }\n\n return [cloneReportNode(node, parent)]\n}\n"],"names":[],"mappings":"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"}
|
package/dist/coverage/report.js
CHANGED
|
@@ -49,9 +49,9 @@ async function generateCoverageReport(config) {
|
|
|
49
49
|
coverageMap,
|
|
50
50
|
defaultSummarizer: "pkg"
|
|
51
51
|
});
|
|
52
|
-
const reportOutput = generateConsoleTextCoverageReport(reports, context);
|
|
52
|
+
const { output: reportOutput, summarizer } = generateConsoleTextCoverageReport(reports, context);
|
|
53
53
|
process.stdout.write(reportOutput);
|
|
54
|
-
reports.create("text", { file: TEXT_REPORT_FILENAME }).execute(context);
|
|
54
|
+
reports.create("text", { file: TEXT_REPORT_FILENAME, summarizer }).execute(context);
|
|
55
55
|
console.log(`[coverage] Full text report saved to ${path.join(config.coverageReportDir, TEXT_REPORT_FILENAME)}`);
|
|
56
56
|
const summary = coverageMap.getCoverageSummary();
|
|
57
57
|
enforceThresholds(summary, config.thresholds, "global");
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"report.js","sources":["../../src/coverage/report.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\n\nimport * as libCoverage from \"istanbul-lib-coverage\"\nimport * as libInstrument from \"istanbul-lib-instrument\"\nimport fg from \"fast-glob\"\nimport picomatch from \"picomatch\"\nimport v8ToIstanbul from \"v8-to-istanbul\"\n\nimport { createCollectCoverageMatcher, toPosix } from \"./collect\"\nimport { generateConsoleTextCoverageReport } from \"./console-text-report\"\nimport { findCoverageFiles } from \"./files\"\nimport type { CoverageConfig, CoverageThresholdTarget, CoverageThresholds } from \"./types\"\n\n\nconst TEXT_REPORT_FILENAME = \"coverage.txt\"\n\nexport class CoverageThresholdError extends Error {\n constructor(message: string) {\n super(message)\n this.name = \"CoverageThresholdError\"\n }\n}\n\nexport async function generateCoverageReport(config: CoverageConfig): Promise<void> {\n const coverageFiles = await findCoverageFiles(config)\n\n if (coverageFiles.length === 0) {\n console.warn(\"[coverage] no V8 coverage artifacts were generated\")\n return\n }\n\n const coverageLib = resolveCoverageLib()\n const coverageMap = coverageLib.createCoverageMap({})\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n for (const file of coverageFiles) {\n const payload = await readCoverageFile(file)\n if (!payload) {\n continue\n }\n\n for (const script of payload.scripts as any[]) {\n await mergeScriptCoverage(coverageMap, script, config, matchesCollectCoverageFrom)\n }\n }\n\n if (config.includeAllFiles) {\n await includeUntestedFiles(coverageMap, config, matchesCollectCoverageFrom)\n }\n\n if (coverageMap.files().length === 0) {\n console.warn(\"[coverage] no library files matched the coverage filters\")\n return\n }\n\n await fs.rm(config.coverageReportDir, { recursive: true, force: true })\n await fs.mkdir(config.coverageReportDir, { recursive: true })\n\n const { createContext, reports } = await loadIstanbulModules()\n const context = createContext({\n dir: config.coverageReportDir,\n coverageMap,\n defaultSummarizer: \"pkg\",\n })\n\n const reportOutput = generateConsoleTextCoverageReport(reports, context)\n process.stdout.write(reportOutput)\n reports.create(\"text\", { file: TEXT_REPORT_FILENAME }).execute(context)\n\n console.log(`[coverage] Full text report saved to ${path.join(config.coverageReportDir, TEXT_REPORT_FILENAME)}`)\n\n const summary = coverageMap.getCoverageSummary()\n enforceThresholds(summary, config.thresholds, \"global\")\n\n const targets: CoverageThresholdTarget[] = Array.isArray(config.thresholdTargets) ? config.thresholdTargets : []\n if (targets.length > 0) {\n const fileSummaries = buildFileSummaries(coverageMap, config.rootDir)\n for (const target of targets) {\n const matcher = createGlobMatcher(target.pattern)\n const matchResult = collectTargetSummary(fileSummaries, matcher, coverageLib)\n\n if (matchResult.matched === 0) {\n console.warn(\n `[coverage] threshold pattern \"${target.pattern}\" did not match any files — skipping`,\n )\n continue\n }\n\n enforceThresholds(matchResult.summary, target.thresholds, target.pattern)\n }\n }\n}\n\nexport async function collectCoveredFiles(config: CoverageConfig): Promise<string[]> {\n const coverageFiles = await findCoverageFiles(config)\n if (coverageFiles.length === 0) {\n return []\n }\n\n const coverageLib = resolveCoverageLib()\n const coverageMap = coverageLib.createCoverageMap({})\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n for (const file of coverageFiles) {\n const payload = await readCoverageFile(file)\n if (!payload) {\n continue\n }\n\n for (const script of payload.scripts as any[]) {\n await mergeScriptCoverage(coverageMap, script, config, matchesCollectCoverageFrom)\n }\n }\n\n return coverageMap.files().sort()\n}\n\nasync function loadIstanbulModules(): Promise<{ createContext: (opts: any) => any; reports: any }> {\n maybeForceColor()\n const [libReportMod, reportsMod]: any[] = await Promise.all([\n import(\"istanbul-lib-report\"),\n import(\"istanbul-reports\"),\n ])\n\n const createContext = typeof libReportMod.createContext === \"function\"\n ? libReportMod.createContext\n : libReportMod.default?.createContext\n\n if (typeof createContext !== \"function\") {\n throw new Error(\"istanbul-lib-report exports are unavailable\")\n }\n\n return {\n createContext,\n reports: reportsMod.default ?? reportsMod,\n }\n}\n\nfunction maybeForceColor(): void {\n if (process.stdout.isTTY) {\n return\n }\n\n if (process.env.FORCE_COLOR !== undefined) {\n return\n }\n\n if (process.env.NO_COLOR !== undefined || process.env.NODE_DISABLE_COLORS !== undefined) {\n return\n }\n\n if (process.env.CI !== undefined) {\n return\n }\n\n process.env.FORCE_COLOR = \"1\"\n}\n\nasync function mergeScriptCoverage(\n coverageMap: any,\n script: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<void> {\n const scriptPath = script.absolutePath\n if (!scriptPath) {\n return\n }\n\n if (isNodeModulesPath(scriptPath)) {\n return\n }\n\n if (isViteVirtualModulePath(scriptPath)) {\n return\n }\n\n let source = script.source && script.source.length > 0\n ? script.source\n : \"\"\n\n if (!source) {\n try {\n source = await fs.readFile(scriptPath, \"utf8\")\n } catch (error: any) {\n const base = path.basename(scriptPath)\n if (error?.code === \"ENOENT\" && base && !base.includes(\".\")) {\n return\n }\n if (error?.code === \"ENOENT\" && !matchesCollectCoverageFrom(scriptPath)) {\n return\n }\n throw error\n }\n }\n\n const sourceMap = await loadSourceMapForScript(scriptPath, source)\n const converter = v8ToIstanbul(\n scriptPath,\n 0,\n sourceMap ? { source, sourceMap: { sourcemap: sourceMap } } : { source },\n )\n await converter.load()\n converter.applyCoverage(script.functions)\n\n const filtered = filterCoverageMap(converter.toIstanbul(), config, matchesCollectCoverageFrom)\n if (Object.keys(filtered).length > 0) {\n coverageMap.merge(filtered)\n }\n}\n\nasync function readCoverageFile(file: string): Promise<any | null> {\n try {\n const raw = await fs.readFile(file, \"utf8\")\n return JSON.parse(raw)\n } catch (error) {\n console.warn(`[coverage] failed to parse ${file}:`, error)\n return null\n }\n}\n\nfunction enforceThresholds(summary: any, thresholds: CoverageThresholds, label = \"global\"): void {\n const failures = []\n\n for (const metric of Object.keys(thresholds) as Array<keyof CoverageThresholds>) {\n const minimum = thresholds[metric]\n const actual = summary[metric]?.pct ?? 0\n if (actual < minimum) {\n failures.push({ metric, actual, minimum })\n }\n }\n\n if (failures.length === 0) {\n return\n }\n\n const details = failures\n .map(({ metric, actual, minimum }) => `${metric}: ${actual.toFixed(2)}% < ${minimum}%`)\n .join(\"; \")\n\n throw new CoverageThresholdError(`[coverage] thresholds not met (target: ${label}) — ${details}`)\n}\n\nfunction resolveCoverageLib(): any {\n const candidate: any = libCoverage as any\n if (typeof candidate.createCoverageMap === \"function\") {\n return candidate\n }\n\n if (candidate.default && typeof candidate.default.createCoverageMap === \"function\") {\n return candidate.default\n }\n\n throw new Error(\"istanbul-lib-coverage exports are unavailable\")\n}\n\nfunction resolveInstrumentLib(): any {\n const candidate: any = libInstrument as any\n if (typeof candidate.createInstrumenter === \"function\") {\n return candidate\n }\n\n if (candidate.default && typeof candidate.default.createInstrumenter === \"function\") {\n return candidate.default\n }\n\n throw new Error(\"istanbul-lib-instrument exports are unavailable\")\n}\n\nasync function includeUntestedFiles(\n coverageMap: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<void> {\n const existing = new Set(\n coverageMap.files().map((filePath: unknown) => path.resolve(String(filePath ?? \"\"))),\n )\n\n const candidates = await findCollectCoverageFiles(config, matchesCollectCoverageFrom)\n if (candidates.length === 0) {\n return\n }\n\n const instrumentLib = resolveInstrumentLib()\n const instrumenter = instrumentLib.createInstrumenter({\n esModules: true,\n parserPlugins: [\n \"typescript\",\n \"jsx\",\n \"classProperties\",\n \"classPrivateProperties\",\n \"classPrivateMethods\",\n \"decorators-legacy\",\n \"importMeta\",\n \"topLevelAwait\",\n ],\n })\n\n for (const filePath of candidates) {\n const normalized = path.resolve(filePath)\n if (existing.has(normalized)) {\n continue\n }\n\n const source = await fs.readFile(normalized, \"utf8\").catch(() => null)\n if (source === null) {\n continue\n }\n\n try {\n instrumenter.instrumentSync(source, normalized)\n const fileCoverage = instrumenter.lastFileCoverage()\n if (!fileCoverage) {\n continue\n }\n coverageMap.addFileCoverage(fileCoverage)\n existing.add(normalized)\n } catch (error) {\n const relative = path.relative(config.rootDir, normalized)\n console.warn(\n `[coverage] failed to instrument ${relative && !relative.startsWith(\"..\") ? relative : normalized}:`,\n error,\n )\n }\n }\n}\n\nconst DEFAULT_COLLECT_COVERAGE_IGNORES = [\n \"**/.git/**\",\n \"**/.next/**\",\n \"**/.turbo/**\",\n \"**/.vite/**\",\n \"**/.vitest/**\",\n \"**/build/**\",\n \"**/coverage/**\",\n \"**/dist/**\",\n \"**/node_modules/**\",\n \"**/playwright-report/**\",\n \"**/spec/**\",\n \"**/test/**\",\n \"**/test-results/**\",\n \"**/tests/**\",\n \"**/__tests__/**\",\n \"**/*.d.ts\",\n \"**/*.map\",\n]\n\nasync function findCollectCoverageFiles(\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<string[]> {\n const patterns = Array.isArray(config.collectCoverageFrom)\n ? config.collectCoverageFrom\n : []\n\n if (patterns.length === 0) {\n return []\n }\n\n const rawFiles = await fg(patterns, {\n cwd: config.rootDir,\n absolute: true,\n dot: true,\n onlyFiles: true,\n unique: true,\n followSymbolicLinks: false,\n ignore: DEFAULT_COLLECT_COVERAGE_IGNORES,\n }).catch(() => [])\n\n const collected = new Set<string>()\n\n for (const file of rawFiles) {\n const normalized = path.resolve(String(file ?? \"\"))\n if (!normalized) {\n continue\n }\n\n if (normalized.endsWith(\".d.ts\") || normalized.endsWith(\".map\")) {\n continue\n }\n\n if (isNodeModulesPath(normalized)) {\n continue\n }\n\n if (isViteVirtualModulePath(normalized)) {\n continue\n }\n\n if (!matchesCollectCoverageFrom(normalized)) {\n continue\n }\n\n collected.add(normalized)\n }\n\n return Array.from(collected).sort()\n}\n\nfunction buildFileSummaries(\n coverageMap: any,\n rootDir: string,\n): Array<{ summary: any; candidates: string[] }> {\n const normalizedRoot = path.resolve(rootDir)\n return coverageMap.files().map((filePath: string) => {\n const normalizedAbsolute = path.resolve(filePath)\n const summary = coverageMap.fileCoverageFor(filePath).toSummary()\n const relativePath = path.relative(normalizedRoot, normalizedAbsolute)\n const candidates = new Set<string>()\n\n if (relativePath) {\n const relativePosix = toPosix(relativePath)\n candidates.add(relativePosix)\n candidates.add(`./${relativePosix}`)\n } else {\n candidates.add(toPosix(path.basename(normalizedAbsolute)))\n }\n\n return {\n summary,\n candidates: Array.from(candidates),\n }\n })\n}\n\nfunction collectTargetSummary(\n fileSummaries: Array<{ summary: any; candidates: string[] }>,\n matcher: (candidate: string) => boolean,\n coverageLib: any,\n): { summary: any; matched: number } {\n const summary = coverageLib.createCoverageSummary()\n let matched = 0\n\n for (const file of fileSummaries) {\n if (file.candidates.some((candidate) => matcher(candidate))) {\n summary.merge(file.summary)\n matched += 1\n }\n }\n\n return { summary, matched }\n}\n\nfunction createGlobMatcher(pattern: string): (candidate: string) => boolean {\n const normalized = toPosix(String(pattern ?? \"\")).trim()\n if (!normalized) {\n return () => false\n }\n if (isAbsoluteGlobPattern(normalized)) {\n throw new Error(`[coverage] threshold patterns must be relative (absolute paths are not supported): \"${pattern}\"`)\n }\n return picomatch(normalized, { dot: true })\n}\n\nfunction isAbsoluteGlobPattern(pattern: string): boolean {\n const normalized = String(pattern ?? \"\").trim()\n if (!normalized) {\n return false\n }\n\n if (normalized.startsWith(\"/\")) {\n return true\n }\n\n if (normalized.startsWith(\"file://\")) {\n return true\n }\n\n return /^[A-Za-z]:\\//.test(normalized)\n}\n\nfunction stripQuery(url: string): string {\n const queryIndex = url.indexOf(\"?\")\n const hashIndex = url.indexOf(\"#\")\n\n const endIndex = Math.min(\n queryIndex === -1 ? Number.POSITIVE_INFINITY : queryIndex,\n hashIndex === -1 ? Number.POSITIVE_INFINITY : hashIndex,\n )\n\n if (!Number.isFinite(endIndex)) {\n return url\n }\n\n return url.slice(0, endIndex)\n}\n\nfunction extractSourceMappingUrl(source: string): string | null {\n const regex = /\\/\\/[#@]\\s*sourceMappingURL=([^\\s]+)/g\n\n let last = null\n let match = null\n\n while ((match = regex.exec(source)) !== null) {\n last = match[1]\n }\n\n return typeof last === \"string\" && last.length > 0 ? last : null\n}\n\nfunction resolveSourceMapPath(scriptPath: string, mappingUrl: string): string {\n const cleaned = stripQuery(mappingUrl)\n\n if (cleaned.startsWith(\"file://\")) {\n return fileURLToPath(cleaned)\n }\n\n if (path.isAbsolute(cleaned)) {\n return cleaned\n }\n\n return path.resolve(path.dirname(scriptPath), cleaned)\n}\n\nfunction filterCoverageMap(\n map: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Record<string, any> {\n if (!map || typeof map !== \"object\") {\n return {}\n }\n\n const filtered: Record<string, any> = {}\n\n for (const [filePath, fileCoverage] of Object.entries(map)) {\n const absolutePath = resolveCoveragePath(filePath, config.rootDir)\n if (!absolutePath) {\n continue\n }\n\n if (absolutePath.endsWith(\".d.ts\") || absolutePath.endsWith(\".map\")) {\n continue\n }\n\n if (isNodeModulesPath(absolutePath)) {\n continue\n }\n\n if (isViteVirtualModulePath(absolutePath)) {\n continue\n }\n\n if (!matchesCollectCoverageFrom(absolutePath)) {\n continue\n }\n\n if (fileCoverage && typeof fileCoverage === \"object\") {\n filtered[absolutePath] = { ...fileCoverage, path: absolutePath }\n } else {\n filtered[absolutePath] = fileCoverage\n }\n }\n\n return filtered\n}\n\nfunction resolveCoveragePath(filePath: string, rootDir: string): string | null {\n const raw = String(filePath ?? \"\").trim()\n if (!raw) {\n return null\n }\n\n const cleaned = stripQuery(raw)\n\n if (cleaned.startsWith(\"file://\")) {\n try {\n return path.normalize(fileURLToPath(cleaned))\n } catch {\n return null\n }\n }\n\n if (path.isAbsolute(cleaned)) {\n return path.normalize(cleaned)\n }\n\n if (cleaned.includes(\"://\")) {\n return null\n }\n\n return path.normalize(path.resolve(rootDir, cleaned))\n}\n\nfunction isNodeModulesPath(filePath: string): boolean {\n return path\n .normalize(String(filePath ?? \"\"))\n .split(path.sep)\n .includes(\"node_modules\")\n}\n\nfunction isViteVirtualModulePath(filePath: string): boolean {\n const normalized = path.normalize(String(filePath ?? \"\"))\n const baseName = path.basename(normalized)\n return baseName === \"__vite-browser-external\"\n || baseName.startsWith(\"__vite-browser-external:\")\n || baseName.startsWith(\"__vite-\")\n}\n\nfunction parseSourceMapPayload(raw: any): any | null {\n if (!raw || typeof raw !== \"object\") {\n return null\n }\n\n const sources = raw.sources\n if (!Array.isArray(sources) || sources.length === 0) {\n return null\n }\n\n return raw\n}\n\nfunction normalizeSourceMap(sourceMap: any, scriptPath: string): any {\n const root = typeof sourceMap.sourceRoot === \"string\"\n ? sourceMap.sourceRoot.replace(\"file://\", \"\")\n : \"\"\n\n const dir = path.dirname(scriptPath)\n const fixedSources = sourceMap.sources.map((source: unknown) => {\n const raw = String(source ?? \"\")\n const cleaned = stripQuery(raw)\n\n if (cleaned.startsWith(\"file://\")) {\n try {\n return path.normalize(fileURLToPath(cleaned))\n } catch {\n return cleaned\n }\n }\n\n const withoutWebpack = cleaned.replace(/^webpack:\\/\\//, \"\")\n const candidate = path.join(root, withoutWebpack)\n\n if (path.isAbsolute(candidate)) {\n return path.normalize(candidate)\n }\n\n const normalizedCandidate = candidate.split(\"/\").join(path.sep)\n\n if (dir.endsWith(`${path.sep}dist`) && !normalizedCandidate.startsWith(`..${path.sep}`)) {\n if (normalizedCandidate.startsWith(`src${path.sep}`) || normalizedCandidate.startsWith(`lib${path.sep}`)) {\n return path.normalize(path.resolve(dir, \"..\", normalizedCandidate))\n }\n }\n\n return path.normalize(path.resolve(dir, normalizedCandidate))\n })\n\n return {\n ...sourceMap,\n sources: fixedSources,\n }\n}\n\nasync function loadSourceMapForScript(scriptPath: string, source: string): Promise<any | null> {\n const mappingUrl = extractSourceMappingUrl(source)\n if (!mappingUrl) {\n return null\n }\n\n const cleaned = stripQuery(mappingUrl)\n\n if (cleaned.startsWith(\"data:\")) {\n const commaIndex = cleaned.indexOf(\",\")\n if (commaIndex === -1) {\n return null\n }\n\n const meta = cleaned.slice(0, commaIndex)\n const payload = cleaned.slice(commaIndex + 1)\n const raw = meta.includes(\";base64\")\n ? Buffer.from(payload, \"base64\").toString(\"utf8\")\n : decodeURIComponent(payload)\n\n try {\n const parsed = parseSourceMapPayload(JSON.parse(raw))\n return parsed ? normalizeSourceMap(parsed, scriptPath) : null\n } catch {\n return null\n }\n }\n\n try {\n const mapPath = resolveSourceMapPath(scriptPath, cleaned)\n const raw = await fs.readFile(mapPath, \"utf8\")\n const parsed = parseSourceMapPayload(JSON.parse(raw))\n return parsed ? normalizeSourceMap(parsed, scriptPath) : null\n } catch {\n return null\n }\n}\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"report.js","sources":["../../src/coverage/report.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { fileURLToPath } from \"node:url\"\n\nimport * as libCoverage from \"istanbul-lib-coverage\"\nimport * as libInstrument from \"istanbul-lib-instrument\"\nimport fg from \"fast-glob\"\nimport picomatch from \"picomatch\"\nimport v8ToIstanbul from \"v8-to-istanbul\"\n\nimport { createCollectCoverageMatcher, toPosix } from \"./collect\"\nimport { generateConsoleTextCoverageReport } from \"./console-text-report\"\nimport { findCoverageFiles } from \"./files\"\nimport type { CoverageConfig, CoverageThresholdTarget, CoverageThresholds } from \"./types\"\n\n\nconst TEXT_REPORT_FILENAME = \"coverage.txt\"\n\nexport class CoverageThresholdError extends Error {\n constructor(message: string) {\n super(message)\n this.name = \"CoverageThresholdError\"\n }\n}\n\nexport async function generateCoverageReport(config: CoverageConfig): Promise<void> {\n const coverageFiles = await findCoverageFiles(config)\n\n if (coverageFiles.length === 0) {\n console.warn(\"[coverage] no V8 coverage artifacts were generated\")\n return\n }\n\n const coverageLib = resolveCoverageLib()\n const coverageMap = coverageLib.createCoverageMap({})\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n for (const file of coverageFiles) {\n const payload = await readCoverageFile(file)\n if (!payload) {\n continue\n }\n\n for (const script of payload.scripts as any[]) {\n await mergeScriptCoverage(coverageMap, script, config, matchesCollectCoverageFrom)\n }\n }\n\n if (config.includeAllFiles) {\n await includeUntestedFiles(coverageMap, config, matchesCollectCoverageFrom)\n }\n\n if (coverageMap.files().length === 0) {\n console.warn(\"[coverage] no library files matched the coverage filters\")\n return\n }\n\n await fs.rm(config.coverageReportDir, { recursive: true, force: true })\n await fs.mkdir(config.coverageReportDir, { recursive: true })\n\n const { createContext, reports } = await loadIstanbulModules()\n const context = createContext({\n dir: config.coverageReportDir,\n coverageMap,\n defaultSummarizer: \"pkg\",\n })\n\n const { output: reportOutput, summarizer } = generateConsoleTextCoverageReport(reports, context)\n process.stdout.write(reportOutput)\n reports.create(\"text\", { file: TEXT_REPORT_FILENAME, summarizer }).execute(context)\n\n console.log(`[coverage] Full text report saved to ${path.join(config.coverageReportDir, TEXT_REPORT_FILENAME)}`)\n\n const summary = coverageMap.getCoverageSummary()\n enforceThresholds(summary, config.thresholds, \"global\")\n\n const targets: CoverageThresholdTarget[] = Array.isArray(config.thresholdTargets) ? config.thresholdTargets : []\n if (targets.length > 0) {\n const fileSummaries = buildFileSummaries(coverageMap, config.rootDir)\n for (const target of targets) {\n const matcher = createGlobMatcher(target.pattern)\n const matchResult = collectTargetSummary(fileSummaries, matcher, coverageLib)\n\n if (matchResult.matched === 0) {\n console.warn(\n `[coverage] threshold pattern \"${target.pattern}\" did not match any files — skipping`,\n )\n continue\n }\n\n enforceThresholds(matchResult.summary, target.thresholds, target.pattern)\n }\n }\n}\n\nexport async function collectCoveredFiles(config: CoverageConfig): Promise<string[]> {\n const coverageFiles = await findCoverageFiles(config)\n if (coverageFiles.length === 0) {\n return []\n }\n\n const coverageLib = resolveCoverageLib()\n const coverageMap = coverageLib.createCoverageMap({})\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n for (const file of coverageFiles) {\n const payload = await readCoverageFile(file)\n if (!payload) {\n continue\n }\n\n for (const script of payload.scripts as any[]) {\n await mergeScriptCoverage(coverageMap, script, config, matchesCollectCoverageFrom)\n }\n }\n\n return coverageMap.files().sort()\n}\n\nasync function loadIstanbulModules(): Promise<{ createContext: (opts: any) => any; reports: any }> {\n maybeForceColor()\n const [libReportMod, reportsMod]: any[] = await Promise.all([\n import(\"istanbul-lib-report\"),\n import(\"istanbul-reports\"),\n ])\n\n const createContext = typeof libReportMod.createContext === \"function\"\n ? libReportMod.createContext\n : libReportMod.default?.createContext\n\n if (typeof createContext !== \"function\") {\n throw new Error(\"istanbul-lib-report exports are unavailable\")\n }\n\n return {\n createContext,\n reports: reportsMod.default ?? reportsMod,\n }\n}\n\nfunction maybeForceColor(): void {\n if (process.stdout.isTTY) {\n return\n }\n\n if (process.env.FORCE_COLOR !== undefined) {\n return\n }\n\n if (process.env.NO_COLOR !== undefined || process.env.NODE_DISABLE_COLORS !== undefined) {\n return\n }\n\n if (process.env.CI !== undefined) {\n return\n }\n\n process.env.FORCE_COLOR = \"1\"\n}\n\nasync function mergeScriptCoverage(\n coverageMap: any,\n script: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<void> {\n const scriptPath = script.absolutePath\n if (!scriptPath) {\n return\n }\n\n if (isNodeModulesPath(scriptPath)) {\n return\n }\n\n if (isViteVirtualModulePath(scriptPath)) {\n return\n }\n\n let source = script.source && script.source.length > 0\n ? script.source\n : \"\"\n\n if (!source) {\n try {\n source = await fs.readFile(scriptPath, \"utf8\")\n } catch (error: any) {\n const base = path.basename(scriptPath)\n if (error?.code === \"ENOENT\" && base && !base.includes(\".\")) {\n return\n }\n if (error?.code === \"ENOENT\" && !matchesCollectCoverageFrom(scriptPath)) {\n return\n }\n throw error\n }\n }\n\n const sourceMap = await loadSourceMapForScript(scriptPath, source)\n const converter = v8ToIstanbul(\n scriptPath,\n 0,\n sourceMap ? { source, sourceMap: { sourcemap: sourceMap } } : { source },\n )\n await converter.load()\n converter.applyCoverage(script.functions)\n\n const filtered = filterCoverageMap(converter.toIstanbul(), config, matchesCollectCoverageFrom)\n if (Object.keys(filtered).length > 0) {\n coverageMap.merge(filtered)\n }\n}\n\nasync function readCoverageFile(file: string): Promise<any | null> {\n try {\n const raw = await fs.readFile(file, \"utf8\")\n return JSON.parse(raw)\n } catch (error) {\n console.warn(`[coverage] failed to parse ${file}:`, error)\n return null\n }\n}\n\nfunction enforceThresholds(summary: any, thresholds: CoverageThresholds, label = \"global\"): void {\n const failures = []\n\n for (const metric of Object.keys(thresholds) as Array<keyof CoverageThresholds>) {\n const minimum = thresholds[metric]\n const actual = summary[metric]?.pct ?? 0\n if (actual < minimum) {\n failures.push({ metric, actual, minimum })\n }\n }\n\n if (failures.length === 0) {\n return\n }\n\n const details = failures\n .map(({ metric, actual, minimum }) => `${metric}: ${actual.toFixed(2)}% < ${minimum}%`)\n .join(\"; \")\n\n throw new CoverageThresholdError(`[coverage] thresholds not met (target: ${label}) — ${details}`)\n}\n\nfunction resolveCoverageLib(): any {\n const candidate: any = libCoverage as any\n if (typeof candidate.createCoverageMap === \"function\") {\n return candidate\n }\n\n if (candidate.default && typeof candidate.default.createCoverageMap === \"function\") {\n return candidate.default\n }\n\n throw new Error(\"istanbul-lib-coverage exports are unavailable\")\n}\n\nfunction resolveInstrumentLib(): any {\n const candidate: any = libInstrument as any\n if (typeof candidate.createInstrumenter === \"function\") {\n return candidate\n }\n\n if (candidate.default && typeof candidate.default.createInstrumenter === \"function\") {\n return candidate.default\n }\n\n throw new Error(\"istanbul-lib-instrument exports are unavailable\")\n}\n\nasync function includeUntestedFiles(\n coverageMap: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<void> {\n const existing = new Set(\n coverageMap.files().map((filePath: unknown) => path.resolve(String(filePath ?? \"\"))),\n )\n\n const candidates = await findCollectCoverageFiles(config, matchesCollectCoverageFrom)\n if (candidates.length === 0) {\n return\n }\n\n const instrumentLib = resolveInstrumentLib()\n const instrumenter = instrumentLib.createInstrumenter({\n esModules: true,\n parserPlugins: [\n \"typescript\",\n \"jsx\",\n \"classProperties\",\n \"classPrivateProperties\",\n \"classPrivateMethods\",\n \"decorators-legacy\",\n \"importMeta\",\n \"topLevelAwait\",\n ],\n })\n\n for (const filePath of candidates) {\n const normalized = path.resolve(filePath)\n if (existing.has(normalized)) {\n continue\n }\n\n const source = await fs.readFile(normalized, \"utf8\").catch(() => null)\n if (source === null) {\n continue\n }\n\n try {\n instrumenter.instrumentSync(source, normalized)\n const fileCoverage = instrumenter.lastFileCoverage()\n if (!fileCoverage) {\n continue\n }\n coverageMap.addFileCoverage(fileCoverage)\n existing.add(normalized)\n } catch (error) {\n const relative = path.relative(config.rootDir, normalized)\n console.warn(\n `[coverage] failed to instrument ${relative && !relative.startsWith(\"..\") ? relative : normalized}:`,\n error,\n )\n }\n }\n}\n\nconst DEFAULT_COLLECT_COVERAGE_IGNORES = [\n \"**/.git/**\",\n \"**/.next/**\",\n \"**/.turbo/**\",\n \"**/.vite/**\",\n \"**/.vitest/**\",\n \"**/build/**\",\n \"**/coverage/**\",\n \"**/dist/**\",\n \"**/node_modules/**\",\n \"**/playwright-report/**\",\n \"**/spec/**\",\n \"**/test/**\",\n \"**/test-results/**\",\n \"**/tests/**\",\n \"**/__tests__/**\",\n \"**/*.d.ts\",\n \"**/*.map\",\n]\n\nasync function findCollectCoverageFiles(\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Promise<string[]> {\n const patterns = Array.isArray(config.collectCoverageFrom)\n ? config.collectCoverageFrom\n : []\n\n if (patterns.length === 0) {\n return []\n }\n\n const rawFiles = await fg(patterns, {\n cwd: config.rootDir,\n absolute: true,\n dot: true,\n onlyFiles: true,\n unique: true,\n followSymbolicLinks: false,\n ignore: DEFAULT_COLLECT_COVERAGE_IGNORES,\n }).catch(() => [])\n\n const collected = new Set<string>()\n\n for (const file of rawFiles) {\n const normalized = path.resolve(String(file ?? \"\"))\n if (!normalized) {\n continue\n }\n\n if (normalized.endsWith(\".d.ts\") || normalized.endsWith(\".map\")) {\n continue\n }\n\n if (isNodeModulesPath(normalized)) {\n continue\n }\n\n if (isViteVirtualModulePath(normalized)) {\n continue\n }\n\n if (!matchesCollectCoverageFrom(normalized)) {\n continue\n }\n\n collected.add(normalized)\n }\n\n return Array.from(collected).sort()\n}\n\nfunction buildFileSummaries(\n coverageMap: any,\n rootDir: string,\n): Array<{ summary: any; candidates: string[] }> {\n const normalizedRoot = path.resolve(rootDir)\n return coverageMap.files().map((filePath: string) => {\n const normalizedAbsolute = path.resolve(filePath)\n const summary = coverageMap.fileCoverageFor(filePath).toSummary()\n const relativePath = path.relative(normalizedRoot, normalizedAbsolute)\n const candidates = new Set<string>()\n\n if (relativePath) {\n const relativePosix = toPosix(relativePath)\n candidates.add(relativePosix)\n candidates.add(`./${relativePosix}`)\n } else {\n candidates.add(toPosix(path.basename(normalizedAbsolute)))\n }\n\n return {\n summary,\n candidates: Array.from(candidates),\n }\n })\n}\n\nfunction collectTargetSummary(\n fileSummaries: Array<{ summary: any; candidates: string[] }>,\n matcher: (candidate: string) => boolean,\n coverageLib: any,\n): { summary: any; matched: number } {\n const summary = coverageLib.createCoverageSummary()\n let matched = 0\n\n for (const file of fileSummaries) {\n if (file.candidates.some((candidate) => matcher(candidate))) {\n summary.merge(file.summary)\n matched += 1\n }\n }\n\n return { summary, matched }\n}\n\nfunction createGlobMatcher(pattern: string): (candidate: string) => boolean {\n const normalized = toPosix(String(pattern ?? \"\")).trim()\n if (!normalized) {\n return () => false\n }\n if (isAbsoluteGlobPattern(normalized)) {\n throw new Error(`[coverage] threshold patterns must be relative (absolute paths are not supported): \"${pattern}\"`)\n }\n return picomatch(normalized, { dot: true })\n}\n\nfunction isAbsoluteGlobPattern(pattern: string): boolean {\n const normalized = String(pattern ?? \"\").trim()\n if (!normalized) {\n return false\n }\n\n if (normalized.startsWith(\"/\")) {\n return true\n }\n\n if (normalized.startsWith(\"file://\")) {\n return true\n }\n\n return /^[A-Za-z]:\\//.test(normalized)\n}\n\nfunction stripQuery(url: string): string {\n const queryIndex = url.indexOf(\"?\")\n const hashIndex = url.indexOf(\"#\")\n\n const endIndex = Math.min(\n queryIndex === -1 ? Number.POSITIVE_INFINITY : queryIndex,\n hashIndex === -1 ? Number.POSITIVE_INFINITY : hashIndex,\n )\n\n if (!Number.isFinite(endIndex)) {\n return url\n }\n\n return url.slice(0, endIndex)\n}\n\nfunction extractSourceMappingUrl(source: string): string | null {\n const regex = /\\/\\/[#@]\\s*sourceMappingURL=([^\\s]+)/g\n\n let last = null\n let match = null\n\n while ((match = regex.exec(source)) !== null) {\n last = match[1]\n }\n\n return typeof last === \"string\" && last.length > 0 ? last : null\n}\n\nfunction resolveSourceMapPath(scriptPath: string, mappingUrl: string): string {\n const cleaned = stripQuery(mappingUrl)\n\n if (cleaned.startsWith(\"file://\")) {\n return fileURLToPath(cleaned)\n }\n\n if (path.isAbsolute(cleaned)) {\n return cleaned\n }\n\n return path.resolve(path.dirname(scriptPath), cleaned)\n}\n\nfunction filterCoverageMap(\n map: any,\n config: CoverageConfig,\n matchesCollectCoverageFrom: (absolutePath: string) => boolean,\n): Record<string, any> {\n if (!map || typeof map !== \"object\") {\n return {}\n }\n\n const filtered: Record<string, any> = {}\n\n for (const [filePath, fileCoverage] of Object.entries(map)) {\n const absolutePath = resolveCoveragePath(filePath, config.rootDir)\n if (!absolutePath) {\n continue\n }\n\n if (absolutePath.endsWith(\".d.ts\") || absolutePath.endsWith(\".map\")) {\n continue\n }\n\n if (isNodeModulesPath(absolutePath)) {\n continue\n }\n\n if (isViteVirtualModulePath(absolutePath)) {\n continue\n }\n\n if (!matchesCollectCoverageFrom(absolutePath)) {\n continue\n }\n\n if (fileCoverage && typeof fileCoverage === \"object\") {\n filtered[absolutePath] = { ...fileCoverage, path: absolutePath }\n } else {\n filtered[absolutePath] = fileCoverage\n }\n }\n\n return filtered\n}\n\nfunction resolveCoveragePath(filePath: string, rootDir: string): string | null {\n const raw = String(filePath ?? \"\").trim()\n if (!raw) {\n return null\n }\n\n const cleaned = stripQuery(raw)\n\n if (cleaned.startsWith(\"file://\")) {\n try {\n return path.normalize(fileURLToPath(cleaned))\n } catch {\n return null\n }\n }\n\n if (path.isAbsolute(cleaned)) {\n return path.normalize(cleaned)\n }\n\n if (cleaned.includes(\"://\")) {\n return null\n }\n\n return path.normalize(path.resolve(rootDir, cleaned))\n}\n\nfunction isNodeModulesPath(filePath: string): boolean {\n return path\n .normalize(String(filePath ?? \"\"))\n .split(path.sep)\n .includes(\"node_modules\")\n}\n\nfunction isViteVirtualModulePath(filePath: string): boolean {\n const normalized = path.normalize(String(filePath ?? \"\"))\n const baseName = path.basename(normalized)\n return baseName === \"__vite-browser-external\"\n || baseName.startsWith(\"__vite-browser-external:\")\n || baseName.startsWith(\"__vite-\")\n}\n\nfunction parseSourceMapPayload(raw: any): any | null {\n if (!raw || typeof raw !== \"object\") {\n return null\n }\n\n const sources = raw.sources\n if (!Array.isArray(sources) || sources.length === 0) {\n return null\n }\n\n return raw\n}\n\nfunction normalizeSourceMap(sourceMap: any, scriptPath: string): any {\n const root = typeof sourceMap.sourceRoot === \"string\"\n ? sourceMap.sourceRoot.replace(\"file://\", \"\")\n : \"\"\n\n const dir = path.dirname(scriptPath)\n const fixedSources = sourceMap.sources.map((source: unknown) => {\n const raw = String(source ?? \"\")\n const cleaned = stripQuery(raw)\n\n if (cleaned.startsWith(\"file://\")) {\n try {\n return path.normalize(fileURLToPath(cleaned))\n } catch {\n return cleaned\n }\n }\n\n const withoutWebpack = cleaned.replace(/^webpack:\\/\\//, \"\")\n const candidate = path.join(root, withoutWebpack)\n\n if (path.isAbsolute(candidate)) {\n return path.normalize(candidate)\n }\n\n const normalizedCandidate = candidate.split(\"/\").join(path.sep)\n\n if (dir.endsWith(`${path.sep}dist`) && !normalizedCandidate.startsWith(`..${path.sep}`)) {\n if (normalizedCandidate.startsWith(`src${path.sep}`) || normalizedCandidate.startsWith(`lib${path.sep}`)) {\n return path.normalize(path.resolve(dir, \"..\", normalizedCandidate))\n }\n }\n\n return path.normalize(path.resolve(dir, normalizedCandidate))\n })\n\n return {\n ...sourceMap,\n sources: fixedSources,\n }\n}\n\nasync function loadSourceMapForScript(scriptPath: string, source: string): Promise<any | null> {\n const mappingUrl = extractSourceMappingUrl(source)\n if (!mappingUrl) {\n return null\n }\n\n const cleaned = stripQuery(mappingUrl)\n\n if (cleaned.startsWith(\"data:\")) {\n const commaIndex = cleaned.indexOf(\",\")\n if (commaIndex === -1) {\n return null\n }\n\n const meta = cleaned.slice(0, commaIndex)\n const payload = cleaned.slice(commaIndex + 1)\n const raw = meta.includes(\";base64\")\n ? Buffer.from(payload, \"base64\").toString(\"utf8\")\n : decodeURIComponent(payload)\n\n try {\n const parsed = parseSourceMapPayload(JSON.parse(raw))\n return parsed ? normalizeSourceMap(parsed, scriptPath) : null\n } catch {\n return null\n }\n }\n\n try {\n const mapPath = resolveSourceMapPath(scriptPath, cleaned)\n const raw = await fs.readFile(mapPath, \"utf8\")\n const parsed = parseSourceMapPayload(JSON.parse(raw))\n return parsed ? normalizeSourceMap(parsed, scriptPath) : null\n } catch {\n return null\n }\n}\n"],"names":[],"mappings":"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"}
|