@rpcbase/test 0.334.0 → 0.336.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.
Files changed (37) hide show
  1. package/dist/clearDatabase.js +4 -1
  2. package/dist/clearDatabase.js.map +1 -1
  3. package/dist/cli.js +130 -77
  4. package/dist/cli.js.map +1 -1
  5. package/dist/coverage/collect.js +3 -1
  6. package/dist/coverage/collect.js.map +1 -1
  7. package/dist/coverage/config-loader.js +20 -36
  8. package/dist/coverage/config-loader.js.map +1 -1
  9. package/dist/coverage/config.js +19 -8
  10. package/dist/coverage/config.js.map +1 -1
  11. package/dist/coverage/console-text-report.js +23 -13
  12. package/dist/coverage/console-text-report.js.map +1 -1
  13. package/dist/coverage/files.js +25 -28
  14. package/dist/coverage/files.js.map +1 -1
  15. package/dist/coverage/fixtures.js +3 -1
  16. package/dist/coverage/fixtures.js.map +1 -1
  17. package/dist/coverage/global-setup.js +4 -1
  18. package/dist/coverage/global-setup.js.map +1 -1
  19. package/dist/coverage/index.js +3 -1
  20. package/dist/coverage/index.js.map +1 -1
  21. package/dist/coverage/report.js +55 -62
  22. package/dist/coverage/report.js.map +1 -1
  23. package/dist/coverage/reporter.js +4 -1
  24. package/dist/coverage/reporter.js.map +1 -1
  25. package/dist/coverage/v8-tracker.js +18 -14
  26. package/dist/coverage/v8-tracker.js.map +1 -1
  27. package/dist/index.js +10 -3
  28. package/dist/index.js.map +1 -1
  29. package/dist/serverCoverage.js +6 -2
  30. package/dist/serverCoverage.js.map +1 -1
  31. package/dist/specs-map.d.ts +15 -0
  32. package/dist/specs-map.d.ts.map +1 -0
  33. package/dist/specs-map.js +50 -0
  34. package/dist/specs-map.js.map +1 -0
  35. package/dist/vitest.config.js +14 -14
  36. package/dist/vitest.config.js.map +1 -1
  37. package/package.json +6 -5
@@ -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 { 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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"}
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":["TEXT_REPORT_FILENAME","CoverageThresholdError","Error","constructor","message","name","generateCoverageReport","config","coverageFiles","findCoverageFiles","length","console","warn","coverageLib","resolveCoverageLib","coverageMap","createCoverageMap","matchesCollectCoverageFrom","createCollectCoverageMatcher","collectCoverageFrom","rootDir","file","payload","readCoverageFile","script","scripts","mergeScriptCoverage","includeAllFiles","includeUntestedFiles","files","fs","rm","coverageReportDir","recursive","force","mkdir","createContext","reports","loadIstanbulModules","context","dir","defaultSummarizer","output","reportOutput","summarizer","generateConsoleTextCoverageReport","process","stdout","write","create","execute","log","path","join","summary","getCoverageSummary","enforceThresholds","thresholds","targets","Array","isArray","thresholdTargets","fileSummaries","buildFileSummaries","target","matcher","createGlobMatcher","pattern","matchResult","collectTargetSummary","matched","collectCoveredFiles","sort","maybeForceColor","libReportMod","reportsMod","Promise","all","default","isTTY","env","FORCE_COLOR","undefined","NO_COLOR","NODE_DISABLE_COLORS","CI","scriptPath","absolutePath","isNodeModulesPath","isViteVirtualModulePath","source","readFile","error","base","basename","code","includes","sourceMap","loadSourceMapForScript","converter","v8ToIstanbul","sourcemap","load","applyCoverage","functions","filtered","filterCoverageMap","toIstanbul","Object","keys","merge","raw","JSON","parse","label","failures","metric","minimum","actual","pct","push","details","map","toFixed","candidate","libCoverage","resolveInstrumentLib","libInstrument","createInstrumenter","existing","Set","filePath","resolve","String","candidates","findCollectCoverageFiles","instrumentLib","instrumenter","esModules","parserPlugins","normalized","has","catch","instrumentSync","fileCoverage","lastFileCoverage","addFileCoverage","add","relative","startsWith","DEFAULT_COLLECT_COVERAGE_IGNORES","patterns","rawFiles","fg","cwd","absolute","dot","onlyFiles","unique","followSymbolicLinks","ignore","collected","endsWith","from","normalizedRoot","normalizedAbsolute","fileCoverageFor","toSummary","relativePath","relativePosix","toPosix","createCoverageSummary","some","trim","isAbsoluteGlobPattern","picomatch","test","stripQuery","url","queryIndex","indexOf","hashIndex","endIndex","Math","min","Number","POSITIVE_INFINITY","isFinite","slice","extractSourceMappingUrl","regex","last","match","exec","resolveSourceMapPath","mappingUrl","cleaned","fileURLToPath","isAbsolute","dirname","entries","resolveCoveragePath","normalize","split","sep","baseName","parseSourceMapPayload","sources","normalizeSourceMap","root","sourceRoot","replace","fixedSources","withoutWebpack","normalizedCandidate","commaIndex","meta","Buffer","toString","decodeURIComponent","parsed","mapPath"],"mappings":";;;;;;;;;;;AAgBA,MAAMA,uBAAuB;AAEtB,MAAMC,+BAA+BC,MAAM;AAAA,EAChDC,YAAYC,SAAiB;AAC3B,UAAMA,OAAO;AACb,SAAKC,OAAO;AAAA,EACd;AACF;AAEA,eAAsBC,uBAAuBC,QAAuC;AAClF,QAAMC,gBAAgB,MAAMC,kBAAkBF,MAAM;AAEpD,MAAIC,cAAcE,WAAW,GAAG;AAC9BC,YAAQC,KAAK,oDAAoD;AACjE;AAAA,EACF;AAEA,QAAMC,cAAcC,mBAAAA;AACpB,QAAMC,cAAcF,YAAYG,kBAAkB,EAAE;AACpD,QAAMC,6BAA6BC,6BAA6BX,OAAOY,qBAAqBZ,OAAOa,OAAO;AAE1G,aAAWC,QAAQb,eAAe;AAChC,UAAMc,UAAU,MAAMC,iBAAiBF,IAAI;AAC3C,QAAI,CAACC,SAAS;AACZ;AAAA,IACF;AAEA,eAAWE,UAAUF,QAAQG,SAAkB;AAC7C,YAAMC,oBAAoBX,aAAaS,QAAQjB,QAAQU,0BAA0B;AAAA,IACnF;AAAA,EACF;AAEA,MAAIV,OAAOoB,iBAAiB;AAC1B,UAAMC,qBAAqBb,aAAaR,QAAQU,0BAA0B;AAAA,EAC5E;AAEA,MAAIF,YAAYc,QAAQnB,WAAW,GAAG;AACpCC,YAAQC,KAAK,0DAA0D;AACvE;AAAA,EACF;AAEA,QAAMkB,GAAGC,GAAGxB,OAAOyB,mBAAmB;AAAA,IAAEC,WAAW;AAAA,IAAMC,OAAO;AAAA,EAAA,CAAM;AACtE,QAAMJ,GAAGK,MAAM5B,OAAOyB,mBAAmB;AAAA,IAAEC,WAAW;AAAA,EAAA,CAAM;AAE5D,QAAM;AAAA,IAAEG;AAAAA,IAAeC;AAAAA,EAAAA,IAAY,MAAMC,oBAAAA;AACzC,QAAMC,UAAUH,cAAc;AAAA,IAC5BI,KAAKjC,OAAOyB;AAAAA,IACZjB;AAAAA,IACA0B,mBAAmB;AAAA,EAAA,CACpB;AAED,QAAM;AAAA,IAAEC,QAAQC;AAAAA,IAAcC;AAAAA,EAAAA,IAAeC,kCAAkCR,SAASE,OAAO;AAC/FO,UAAQC,OAAOC,MAAML,YAAY;AACjCN,UAAQY,OAAO,QAAQ;AAAA,IAAE5B,MAAMrB;AAAAA,IAAsB4C;AAAAA,EAAAA,CAAY,EAAEM,QAAQX,OAAO;AAElF5B,UAAQwC,IAAI,wCAAwCC,KAAKC,KAAK9C,OAAOyB,mBAAmBhC,oBAAoB,CAAC,EAAE;AAE/G,QAAMsD,UAAUvC,YAAYwC,mBAAAA;AAC5BC,oBAAkBF,SAAS/C,OAAOkD,YAAY,QAAQ;AAEtD,QAAMC,UAAqCC,MAAMC,QAAQrD,OAAOsD,gBAAgB,IAAItD,OAAOsD,mBAAmB,CAAA;AAC9G,MAAIH,QAAQhD,SAAS,GAAG;AACtB,UAAMoD,gBAAgBC,mBAAmBhD,aAAaR,OAAOa,OAAO;AACpE,eAAW4C,UAAUN,SAAS;AAC5B,YAAMO,UAAUC,kBAAkBF,OAAOG,OAAO;AAChD,YAAMC,cAAcC,qBAAqBP,eAAeG,SAASpD,WAAW;AAE5E,UAAIuD,YAAYE,YAAY,GAAG;AAC7B3D,gBAAQC,KACN,iCAAiCoD,OAAOG,OAAO,sCACjD;AACA;AAAA,MACF;AAEAX,wBAAkBY,YAAYd,SAASU,OAAOP,YAAYO,OAAOG,OAAO;AAAA,IAC1E;AAAA,EACF;AACF;AAEA,eAAsBI,oBAAoBhE,QAA2C;AACnF,QAAMC,gBAAgB,MAAMC,kBAAkBF,MAAM;AACpD,MAAIC,cAAcE,WAAW,GAAG;AAC9B,WAAO,CAAA;AAAA,EACT;AAEA,QAAMG,cAAcC,mBAAAA;AACpB,QAAMC,cAAcF,YAAYG,kBAAkB,EAAE;AACpD,QAAMC,6BAA6BC,6BAA6BX,OAAOY,qBAAqBZ,OAAOa,OAAO;AAE1G,aAAWC,QAAQb,eAAe;AAChC,UAAMc,UAAU,MAAMC,iBAAiBF,IAAI;AAC3C,QAAI,CAACC,SAAS;AACZ;AAAA,IACF;AAEA,eAAWE,UAAUF,QAAQG,SAAkB;AAC7C,YAAMC,oBAAoBX,aAAaS,QAAQjB,QAAQU,0BAA0B;AAAA,IACnF;AAAA,EACF;AAEA,SAAOF,YAAYc,MAAAA,EAAQ2C,KAAAA;AAC7B;AAEA,eAAelC,sBAAoF;AACjGmC,kBAAAA;AACA,QAAM,CAACC,cAAcC,UAAU,IAAW,MAAMC,QAAQC,IAAI,CAC1D,OAAO,qBAAqB,GAC5B,OAAO,kBAAkB,CAAC,CAC3B;AAED,QAAMzC,gBAAgB,OAAOsC,aAAatC,kBAAkB,aACxDsC,aAAatC,gBACbsC,aAAaI,SAAS1C;AAE1B,MAAI,OAAOA,kBAAkB,YAAY;AACvC,UAAM,IAAIlC,MAAM,6CAA6C;AAAA,EAC/D;AAEA,SAAO;AAAA,IACLkC;AAAAA,IACAC,SAASsC,WAAWG,WAAWH;AAAAA,EAAAA;AAEnC;AAEA,SAASF,kBAAwB;AAC/B,MAAI3B,QAAQC,OAAOgC,OAAO;AACxB;AAAA,EACF;AAEA,MAAIjC,QAAQkC,IAAIC,gBAAgBC,QAAW;AACzC;AAAA,EACF;AAEA,MAAIpC,QAAQkC,IAAIG,aAAaD,UAAapC,QAAQkC,IAAII,wBAAwBF,QAAW;AACvF;AAAA,EACF;AAEA,MAAIpC,QAAQkC,IAAIK,OAAOH,QAAW;AAChC;AAAA,EACF;AAEApC,UAAQkC,IAAIC,cAAc;AAC5B;AAEA,eAAevD,oBACbX,aACAS,QACAjB,QACAU,4BACe;AACf,QAAMqE,aAAa9D,OAAO+D;AAC1B,MAAI,CAACD,YAAY;AACf;AAAA,EACF;AAEA,MAAIE,kBAAkBF,UAAU,GAAG;AACjC;AAAA,EACF;AAEA,MAAIG,wBAAwBH,UAAU,GAAG;AACvC;AAAA,EACF;AAEA,MAAII,SAASlE,OAAOkE,UAAUlE,OAAOkE,OAAOhF,SAAS,IACjDc,OAAOkE,SACP;AAEJ,MAAI,CAACA,QAAQ;AACX,QAAI;AACFA,eAAS,MAAM5D,GAAG6D,SAASL,YAAY,MAAM;AAAA,IAC/C,SAASM,OAAY;AACnB,YAAMC,OAAOzC,KAAK0C,SAASR,UAAU;AACrC,UAAIM,OAAOG,SAAS,YAAYF,QAAQ,CAACA,KAAKG,SAAS,GAAG,GAAG;AAC3D;AAAA,MACF;AACA,UAAIJ,OAAOG,SAAS,YAAY,CAAC9E,2BAA2BqE,UAAU,GAAG;AACvE;AAAA,MACF;AACA,YAAMM;AAAAA,IACR;AAAA,EACF;AAEA,QAAMK,YAAY,MAAMC,uBAAuBZ,YAAYI,MAAM;AACjE,QAAMS,YAAYC,aAChBd,YACA,GACAW,YAAY;AAAA,IAAEP;AAAAA,IAAQO,WAAW;AAAA,MAAEI,WAAWJ;AAAAA,IAAAA;AAAAA,EAAU,IAAM;AAAA,IAAEP;AAAAA,EAAAA,CAClE;AACA,QAAMS,UAAUG,KAAAA;AAChBH,YAAUI,cAAc/E,OAAOgF,SAAS;AAExC,QAAMC,WAAWC,kBAAkBP,UAAUQ,WAAAA,GAAcpG,QAAQU,0BAA0B;AAC7F,MAAI2F,OAAOC,KAAKJ,QAAQ,EAAE/F,SAAS,GAAG;AACpCK,gBAAY+F,MAAML,QAAQ;AAAA,EAC5B;AACF;AAEA,eAAelF,iBAAiBF,MAAmC;AACjE,MAAI;AACF,UAAM0F,MAAM,MAAMjF,GAAG6D,SAAStE,MAAM,MAAM;AAC1C,WAAO2F,KAAKC,MAAMF,GAAG;AAAA,EACvB,SAASnB,OAAO;AACdjF,YAAQC,KAAK,8BAA8BS,IAAI,KAAKuE,KAAK;AACzD,WAAO;AAAA,EACT;AACF;AAEA,SAASpC,kBAAkBF,SAAcG,YAAgCyD,QAAQ,UAAgB;AAC/F,QAAMC,WAAW,CAAA;AAEjB,aAAWC,UAAUR,OAAOC,KAAKpD,UAAU,GAAsC;AAC/E,UAAM4D,UAAU5D,WAAW2D,MAAM;AACjC,UAAME,SAAShE,QAAQ8D,MAAM,GAAGG,OAAO;AACvC,QAAID,SAASD,SAAS;AACpBF,eAASK,KAAK;AAAA,QAAEJ;AAAAA,QAAQE;AAAAA,QAAQD;AAAAA,MAAAA,CAAS;AAAA,IAC3C;AAAA,EACF;AAEA,MAAIF,SAASzG,WAAW,GAAG;AACzB;AAAA,EACF;AAEA,QAAM+G,UAAUN,SACbO,IAAI,CAAC;AAAA,IAAEN;AAAAA,IAAQE;AAAAA,IAAQD;AAAAA,EAAAA,MAAc,GAAGD,MAAM,KAAKE,OAAOK,QAAQ,CAAC,CAAC,OAAON,OAAO,GAAG,EACrFhE,KAAK,IAAI;AAEZ,QAAM,IAAIpD,uBAAuB,0CAA0CiH,KAAK,OAAOO,OAAO,EAAE;AAClG;AAEA,SAAS3G,qBAA0B;AACjC,QAAM8G,YAAiBC;AACvB,MAAI,OAAOD,UAAU5G,sBAAsB,YAAY;AACrD,WAAO4G;AAAAA,EACT;AAEA,MAAIA,UAAU9C,WAAW,OAAO8C,UAAU9C,QAAQ9D,sBAAsB,YAAY;AAClF,WAAO4G,UAAU9C;AAAAA,EACnB;AAEA,QAAM,IAAI5E,MAAM,+CAA+C;AACjE;AAEA,SAAS4H,uBAA4B;AACnC,QAAMF,YAAiBG;AACvB,MAAI,OAAOH,UAAUI,uBAAuB,YAAY;AACtD,WAAOJ;AAAAA,EACT;AAEA,MAAIA,UAAU9C,WAAW,OAAO8C,UAAU9C,QAAQkD,uBAAuB,YAAY;AACnF,WAAOJ,UAAU9C;AAAAA,EACnB;AAEA,QAAM,IAAI5E,MAAM,iDAAiD;AACnE;AAEA,eAAe0B,qBACbb,aACAR,QACAU,4BACe;AACf,QAAMgH,WAAW,IAAIC,IACnBnH,YAAYc,MAAAA,EAAQ6F,IAAI,CAACS,aAAsB/E,KAAKgF,QAAQC,OAAOF,YAAY,EAAE,CAAC,CAAC,CACrF;AAEA,QAAMG,aAAa,MAAMC,yBAAyBhI,QAAQU,0BAA0B;AACpF,MAAIqH,WAAW5H,WAAW,GAAG;AAC3B;AAAA,EACF;AAEA,QAAM8H,gBAAgBV,qBAAAA;AACtB,QAAMW,eAAeD,cAAcR,mBAAmB;AAAA,IACpDU,WAAW;AAAA,IACXC,eAAe,CACb,cACA,OACA,mBACA,0BACA,uBACA,qBACA,cACA,eAAe;AAAA,EAAA,CAElB;AAED,aAAWR,YAAYG,YAAY;AACjC,UAAMM,aAAaxF,KAAKgF,QAAQD,QAAQ;AACxC,QAAIF,SAASY,IAAID,UAAU,GAAG;AAC5B;AAAA,IACF;AAEA,UAAMlD,SAAS,MAAM5D,GAAG6D,SAASiD,YAAY,MAAM,EAAEE,MAAM,MAAM,IAAI;AACrE,QAAIpD,WAAW,MAAM;AACnB;AAAA,IACF;AAEA,QAAI;AACF+C,mBAAaM,eAAerD,QAAQkD,UAAU;AAC9C,YAAMI,eAAeP,aAAaQ,iBAAAA;AAClC,UAAI,CAACD,cAAc;AACjB;AAAA,MACF;AACAjI,kBAAYmI,gBAAgBF,YAAY;AACxCf,eAASkB,IAAIP,UAAU;AAAA,IACzB,SAAShD,OAAO;AACd,YAAMwD,WAAWhG,KAAKgG,SAAS7I,OAAOa,SAASwH,UAAU;AACzDjI,cAAQC,KACN,mCAAmCwI,YAAY,CAACA,SAASC,WAAW,IAAI,IAAID,WAAWR,UAAU,KACjGhD,KACF;AAAA,IACF;AAAA,EACF;AACF;AAEA,MAAM0D,mCAAmC,CACvC,cACA,eACA,gBACA,eACA,iBACA,eACA,kBACA,cACA,sBACA,2BACA,cACA,cACA,sBACA,eACA,mBACA,aACA,UAAU;AAGZ,eAAef,yBACbhI,QACAU,4BACmB;AACnB,QAAMsI,WAAW5F,MAAMC,QAAQrD,OAAOY,mBAAmB,IACrDZ,OAAOY,sBACP,CAAA;AAEJ,MAAIoI,SAAS7I,WAAW,GAAG;AACzB,WAAO,CAAA;AAAA,EACT;AAEA,QAAM8I,WAAW,MAAMC,GAAGF,UAAU;AAAA,IAClCG,KAAKnJ,OAAOa;AAAAA,IACZuI,UAAU;AAAA,IACVC,KAAK;AAAA,IACLC,WAAW;AAAA,IACXC,QAAQ;AAAA,IACRC,qBAAqB;AAAA,IACrBC,QAAQV;AAAAA,EAAAA,CACT,EAAER,MAAM,MAAM,EAAE;AAEjB,QAAMmB,gCAAgB/B,IAAAA;AAEtB,aAAW7G,QAAQmI,UAAU;AAC3B,UAAMZ,aAAaxF,KAAKgF,QAAQC,OAAOhH,QAAQ,EAAE,CAAC;AAClD,QAAI,CAACuH,YAAY;AACf;AAAA,IACF;AAEA,QAAIA,WAAWsB,SAAS,OAAO,KAAKtB,WAAWsB,SAAS,MAAM,GAAG;AAC/D;AAAA,IACF;AAEA,QAAI1E,kBAAkBoD,UAAU,GAAG;AACjC;AAAA,IACF;AAEA,QAAInD,wBAAwBmD,UAAU,GAAG;AACvC;AAAA,IACF;AAEA,QAAI,CAAC3H,2BAA2B2H,UAAU,GAAG;AAC3C;AAAA,IACF;AAEAqB,cAAUd,IAAIP,UAAU;AAAA,EAC1B;AAEA,SAAOjF,MAAMwG,KAAKF,SAAS,EAAEzF,KAAAA;AAC/B;AAEA,SAAST,mBACPhD,aACAK,SAC+C;AAC/C,QAAMgJ,iBAAiBhH,KAAKgF,QAAQhH,OAAO;AAC3C,SAAOL,YAAYc,MAAAA,EAAQ6F,IAAI,CAACS,aAAqB;AACnD,UAAMkC,qBAAqBjH,KAAKgF,QAAQD,QAAQ;AAChD,UAAM7E,UAAUvC,YAAYuJ,gBAAgBnC,QAAQ,EAAEoC,UAAAA;AACtD,UAAMC,eAAepH,KAAKgG,SAASgB,gBAAgBC,kBAAkB;AACrE,UAAM/B,iCAAiBJ,IAAAA;AAEvB,QAAIsC,cAAc;AAChB,YAAMC,gBAAgBC,QAAQF,YAAY;AAC1ClC,iBAAWa,IAAIsB,aAAa;AAC5BnC,iBAAWa,IAAI,KAAKsB,aAAa,EAAE;AAAA,IACrC,OAAO;AACLnC,iBAAWa,IAAIuB,QAAQtH,KAAK0C,SAASuE,kBAAkB,CAAC,CAAC;AAAA,IAC3D;AAEA,WAAO;AAAA,MACL/G;AAAAA,MACAgF,YAAY3E,MAAMwG,KAAK7B,UAAU;AAAA,IAAA;AAAA,EAErC,CAAC;AACH;AAEA,SAASjE,qBACPP,eACAG,SACApD,aACmC;AACnC,QAAMyC,UAAUzC,YAAY8J,sBAAAA;AAC5B,MAAIrG,UAAU;AAEd,aAAWjD,QAAQyC,eAAe;AAChC,QAAIzC,KAAKiH,WAAWsC,KAAMhD,eAAc3D,QAAQ2D,SAAS,CAAC,GAAG;AAC3DtE,cAAQwD,MAAMzF,KAAKiC,OAAO;AAC1BgB,iBAAW;AAAA,IACb;AAAA,EACF;AAEA,SAAO;AAAA,IAAEhB;AAAAA,IAASgB;AAAAA,EAAAA;AACpB;AAEA,SAASJ,kBAAkBC,SAAiD;AAC1E,QAAMyE,aAAa8B,QAAQrC,OAAOlE,WAAW,EAAE,CAAC,EAAE0G,KAAAA;AAClD,MAAI,CAACjC,YAAY;AACf,WAAO,MAAM;AAAA,EACf;AACA,MAAIkC,sBAAsBlC,UAAU,GAAG;AACrC,UAAM,IAAI1I,MAAM,uFAAuFiE,OAAO,GAAG;AAAA,EACnH;AACA,SAAO4G,UAAUnC,YAAY;AAAA,IAAEgB,KAAK;AAAA,EAAA,CAAM;AAC5C;AAEA,SAASkB,sBAAsB3G,SAA0B;AACvD,QAAMyE,aAAaP,OAAOlE,WAAW,EAAE,EAAE0G,KAAAA;AACzC,MAAI,CAACjC,YAAY;AACf,WAAO;AAAA,EACT;AAEA,MAAIA,WAAWS,WAAW,GAAG,GAAG;AAC9B,WAAO;AAAA,EACT;AAEA,MAAIT,WAAWS,WAAW,SAAS,GAAG;AACpC,WAAO;AAAA,EACT;AAEA,SAAO,eAAe2B,KAAKpC,UAAU;AACvC;AAEA,SAASqC,WAAWC,KAAqB;AACvC,QAAMC,aAAaD,IAAIE,QAAQ,GAAG;AAClC,QAAMC,YAAYH,IAAIE,QAAQ,GAAG;AAEjC,QAAME,WAAWC,KAAKC,IACpBL,eAAe,KAAKM,OAAOC,oBAAoBP,YAC/CE,cAAc,KAAKI,OAAOC,oBAAoBL,SAChD;AAEA,MAAI,CAACI,OAAOE,SAASL,QAAQ,GAAG;AAC9B,WAAOJ;AAAAA,EACT;AAEA,SAAOA,IAAIU,MAAM,GAAGN,QAAQ;AAC9B;AAEA,SAASO,wBAAwBnG,QAA+B;AAC9D,QAAMoG,QAAQ;AAEd,MAAIC,OAAO;AACX,MAAIC,QAAQ;AAEZ,UAAQA,QAAQF,MAAMG,KAAKvG,MAAM,OAAO,MAAM;AAC5CqG,WAAOC,MAAM,CAAC;AAAA,EAChB;AAEA,SAAO,OAAOD,SAAS,YAAYA,KAAKrL,SAAS,IAAIqL,OAAO;AAC9D;AAEA,SAASG,qBAAqB5G,YAAoB6G,YAA4B;AAC5E,QAAMC,UAAUnB,WAAWkB,UAAU;AAErC,MAAIC,QAAQ/C,WAAW,SAAS,GAAG;AACjC,WAAOgD,cAAcD,OAAO;AAAA,EAC9B;AAEA,MAAIhJ,KAAKkJ,WAAWF,OAAO,GAAG;AAC5B,WAAOA;AAAAA,EACT;AAEA,SAAOhJ,KAAKgF,QAAQhF,KAAKmJ,QAAQjH,UAAU,GAAG8G,OAAO;AACvD;AAEA,SAAS1F,kBACPgB,KACAnH,QACAU,4BACqB;AACrB,MAAI,CAACyG,OAAO,OAAOA,QAAQ,UAAU;AACnC,WAAO,CAAA;AAAA,EACT;AAEA,QAAMjB,WAAgC,CAAA;AAEtC,aAAW,CAAC0B,UAAUa,YAAY,KAAKpC,OAAO4F,QAAQ9E,GAAG,GAAG;AAC1D,UAAMnC,eAAekH,oBAAoBtE,UAAU5H,OAAOa,OAAO;AACjE,QAAI,CAACmE,cAAc;AACjB;AAAA,IACF;AAEA,QAAIA,aAAa2E,SAAS,OAAO,KAAK3E,aAAa2E,SAAS,MAAM,GAAG;AACnE;AAAA,IACF;AAEA,QAAI1E,kBAAkBD,YAAY,GAAG;AACnC;AAAA,IACF;AAEA,QAAIE,wBAAwBF,YAAY,GAAG;AACzC;AAAA,IACF;AAEA,QAAI,CAACtE,2BAA2BsE,YAAY,GAAG;AAC7C;AAAA,IACF;AAEA,QAAIyD,gBAAgB,OAAOA,iBAAiB,UAAU;AACpDvC,eAASlB,YAAY,IAAI;AAAA,QAAE,GAAGyD;AAAAA,QAAc5F,MAAMmC;AAAAA,MAAAA;AAAAA,IACpD,OAAO;AACLkB,eAASlB,YAAY,IAAIyD;AAAAA,IAC3B;AAAA,EACF;AAEA,SAAOvC;AACT;AAEA,SAASgG,oBAAoBtE,UAAkB/G,SAAgC;AAC7E,QAAM2F,MAAMsB,OAAOF,YAAY,EAAE,EAAE0C,KAAAA;AACnC,MAAI,CAAC9D,KAAK;AACR,WAAO;AAAA,EACT;AAEA,QAAMqF,UAAUnB,WAAWlE,GAAG;AAE9B,MAAIqF,QAAQ/C,WAAW,SAAS,GAAG;AACjC,QAAI;AACF,aAAOjG,KAAKsJ,UAAUL,cAAcD,OAAO,CAAC;AAAA,IAC9C,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAEA,MAAIhJ,KAAKkJ,WAAWF,OAAO,GAAG;AAC5B,WAAOhJ,KAAKsJ,UAAUN,OAAO;AAAA,EAC/B;AAEA,MAAIA,QAAQpG,SAAS,KAAK,GAAG;AAC3B,WAAO;AAAA,EACT;AAEA,SAAO5C,KAAKsJ,UAAUtJ,KAAKgF,QAAQhH,SAASgL,OAAO,CAAC;AACtD;AAEA,SAAS5G,kBAAkB2C,UAA2B;AACpD,SAAO/E,KACJsJ,UAAUrE,OAAOF,YAAY,EAAE,CAAC,EAChCwE,MAAMvJ,KAAKwJ,GAAG,EACd5G,SAAS,cAAc;AAC5B;AAEA,SAASP,wBAAwB0C,UAA2B;AAC1D,QAAMS,aAAaxF,KAAKsJ,UAAUrE,OAAOF,YAAY,EAAE,CAAC;AACxD,QAAM0E,WAAWzJ,KAAK0C,SAAS8C,UAAU;AACzC,SAAOiE,aAAa,6BACfA,SAASxD,WAAW,0BAA0B,KAC9CwD,SAASxD,WAAW,SAAS;AACpC;AAEA,SAASyD,sBAAsB/F,KAAsB;AACnD,MAAI,CAACA,OAAO,OAAOA,QAAQ,UAAU;AACnC,WAAO;AAAA,EACT;AAEA,QAAMgG,UAAUhG,IAAIgG;AACpB,MAAI,CAACpJ,MAAMC,QAAQmJ,OAAO,KAAKA,QAAQrM,WAAW,GAAG;AACnD,WAAO;AAAA,EACT;AAEA,SAAOqG;AACT;AAEA,SAASiG,mBAAmB/G,WAAgBX,YAAyB;AACnE,QAAM2H,OAAO,OAAOhH,UAAUiH,eAAe,WACzCjH,UAAUiH,WAAWC,QAAQ,WAAW,EAAE,IAC1C;AAEJ,QAAM3K,MAAMY,KAAKmJ,QAAQjH,UAAU;AACnC,QAAM8H,eAAenH,UAAU8G,QAAQrF,IAAI,CAAChC,WAAoB;AAC9D,UAAMqB,MAAMsB,OAAO3C,UAAU,EAAE;AAC/B,UAAM0G,UAAUnB,WAAWlE,GAAG;AAE9B,QAAIqF,QAAQ/C,WAAW,SAAS,GAAG;AACjC,UAAI;AACF,eAAOjG,KAAKsJ,UAAUL,cAAcD,OAAO,CAAC;AAAA,MAC9C,QAAQ;AACN,eAAOA;AAAAA,MACT;AAAA,IACF;AAEA,UAAMiB,iBAAiBjB,QAAQe,QAAQ,iBAAiB,EAAE;AAC1D,UAAMvF,YAAYxE,KAAKC,KAAK4J,MAAMI,cAAc;AAEhD,QAAIjK,KAAKkJ,WAAW1E,SAAS,GAAG;AAC9B,aAAOxE,KAAKsJ,UAAU9E,SAAS;AAAA,IACjC;AAEA,UAAM0F,sBAAsB1F,UAAU+E,MAAM,GAAG,EAAEtJ,KAAKD,KAAKwJ,GAAG;AAE9D,QAAIpK,IAAI0H,SAAS,GAAG9G,KAAKwJ,GAAG,MAAM,KAAK,CAACU,oBAAoBjE,WAAW,KAAKjG,KAAKwJ,GAAG,EAAE,GAAG;AACvF,UAAIU,oBAAoBjE,WAAW,MAAMjG,KAAKwJ,GAAG,EAAE,KAAKU,oBAAoBjE,WAAW,MAAMjG,KAAKwJ,GAAG,EAAE,GAAG;AACxG,eAAOxJ,KAAKsJ,UAAUtJ,KAAKgF,QAAQ5F,KAAK,MAAM8K,mBAAmB,CAAC;AAAA,MACpE;AAAA,IACF;AAEA,WAAOlK,KAAKsJ,UAAUtJ,KAAKgF,QAAQ5F,KAAK8K,mBAAmB,CAAC;AAAA,EAC9D,CAAC;AAED,SAAO;AAAA,IACL,GAAGrH;AAAAA,IACH8G,SAASK;AAAAA,EAAAA;AAEb;AAEA,eAAelH,uBAAuBZ,YAAoBI,QAAqC;AAC7F,QAAMyG,aAAaN,wBAAwBnG,MAAM;AACjD,MAAI,CAACyG,YAAY;AACf,WAAO;AAAA,EACT;AAEA,QAAMC,UAAUnB,WAAWkB,UAAU;AAErC,MAAIC,QAAQ/C,WAAW,OAAO,GAAG;AAC/B,UAAMkE,aAAanB,QAAQhB,QAAQ,GAAG;AACtC,QAAImC,eAAe,IAAI;AACrB,aAAO;AAAA,IACT;AAEA,UAAMC,OAAOpB,QAAQR,MAAM,GAAG2B,UAAU;AACxC,UAAMjM,UAAU8K,QAAQR,MAAM2B,aAAa,CAAC;AAC5C,UAAMxG,MAAMyG,KAAKxH,SAAS,SAAS,IAC/ByH,OAAOtD,KAAK7I,SAAS,QAAQ,EAAEoM,SAAS,MAAM,IAC9CC,mBAAmBrM,OAAO;AAE9B,QAAI;AACF,YAAMsM,SAASd,sBAAsB9F,KAAKC,MAAMF,GAAG,CAAC;AACpD,aAAO6G,SAASZ,mBAAmBY,QAAQtI,UAAU,IAAI;AAAA,IAC3D,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAEA,MAAI;AACF,UAAMuI,UAAU3B,qBAAqB5G,YAAY8G,OAAO;AACxD,UAAMrF,MAAM,MAAMjF,GAAG6D,SAASkI,SAAS,MAAM;AAC7C,UAAMD,SAASd,sBAAsB9F,KAAKC,MAAMF,GAAG,CAAC;AACpD,WAAO6G,SAASZ,mBAAmBY,QAAQtI,UAAU,IAAI;AAAA,EAC3D,QAAQ;AACN,WAAO;AAAA,EACT;AACF;"}
@@ -17,7 +17,10 @@ class CoverageReporter {
17
17
  return;
18
18
  }
19
19
  await removeCoverageFiles(this.config);
20
- await fs.rm(this.config.coverageReportDir, { recursive: true, force: true });
20
+ await fs.rm(this.config.coverageReportDir, {
21
+ recursive: true,
22
+ force: true
23
+ });
21
24
  }
22
25
  async onEnd(result) {
23
26
  if (process.env.RB_TEST_COMBINED_COVERAGE === "1") {
@@ -1 +1 @@
1
- {"version":3,"file":"reporter.js","sources":["../../src/coverage/reporter.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\n\nimport type { FullResult } from \"@playwright/test/reporter\"\n\nimport { CoverageThresholdError, generateCoverageReport } from \"./report\"\nimport { removeCoverageFiles } from \"./files\"\nimport type { CoverageConfig } from \"./types\"\n\n\ntype FullResultWithExitCode = FullResult & { exitCode?: number }\n\nclass CoverageReporter {\n config: CoverageConfig\n\n constructor(options: { coverageConfig?: CoverageConfig } = {}) {\n this.config = options.coverageConfig as CoverageConfig\n if (!this.config) {\n throw new Error(\"CoverageReporter requires a coverageConfig option\")\n }\n }\n\n async onBegin() {\n if (process.env.RB_TEST_COMBINED_COVERAGE === \"1\") {\n return\n }\n\n if (!this.config.coverageEnabled) {\n return\n }\n\n await removeCoverageFiles(this.config)\n await fs.rm(this.config.coverageReportDir, { recursive: true, force: true })\n }\n\n async onEnd(result: FullResultWithExitCode) {\n if (process.env.RB_TEST_COMBINED_COVERAGE === \"1\") {\n return\n }\n\n if (!this.config.coverageEnabled) {\n return\n }\n\n if (result.status !== \"passed\") {\n console.warn(\"[coverage] skipping report generation because tests failed\")\n return\n }\n\n try {\n await generateCoverageReport(this.config)\n } catch (error) {\n if (error instanceof CoverageThresholdError) {\n console.error(error.message)\n setFailureExitCode(result)\n return\n }\n\n throw error\n }\n }\n}\n\nexport { CoverageReporter }\nexport default CoverageReporter\n\nfunction setFailureExitCode(result: FullResult): void {\n if (result && typeof result === \"object\") {\n const mutable = result as FullResultWithExitCode\n mutable.status = \"failed\"\n if (typeof mutable.exitCode !== \"number\" || mutable.exitCode === 0) {\n mutable.exitCode = 1\n }\n }\n\n if (!process.exitCode || process.exitCode === 0) {\n process.exitCode = 1\n }\n}\n"],"names":[],"mappings":";;;AAWA,MAAM,iBAAiB;AAAA,EACrB;AAAA,EAEA,YAAY,UAA+C,IAAI;AAC7D,SAAK,SAAS,QAAQ;AACtB,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,mDAAmD;AAAA,IACrE;AAAA,EACF;AAAA,EAEA,MAAM,UAAU;AACd,QAAI,QAAQ,IAAI,8BAA8B,KAAK;AACjD;AAAA,IACF;AAEA,QAAI,CAAC,KAAK,OAAO,iBAAiB;AAChC;AAAA,IACF;AAEA,UAAM,oBAAoB,KAAK,MAAM;AACrC,UAAM,GAAG,GAAG,KAAK,OAAO,mBAAmB,EAAE,WAAW,MAAM,OAAO,MAAM;AAAA,EAC7E;AAAA,EAEA,MAAM,MAAM,QAAgC;AAC1C,QAAI,QAAQ,IAAI,8BAA8B,KAAK;AACjD;AAAA,IACF;AAEA,QAAI,CAAC,KAAK,OAAO,iBAAiB;AAChC;AAAA,IACF;AAEA,QAAI,OAAO,WAAW,UAAU;AAC9B,cAAQ,KAAK,4DAA4D;AACzE;AAAA,IACF;AAEA,QAAI;AACF,YAAM,uBAAuB,KAAK,MAAM;AAAA,IAC1C,SAAS,OAAO;AACd,UAAI,iBAAiB,wBAAwB;AAC3C,gBAAQ,MAAM,MAAM,OAAO;AAC3B,2BAAmB,MAAM;AACzB;AAAA,MACF;AAEA,YAAM;AAAA,IACR;AAAA,EACF;AACF;AAKA,SAAS,mBAAmB,QAA0B;AACpD,MAAI,UAAU,OAAO,WAAW,UAAU;AACxC,UAAM,UAAU;AAChB,YAAQ,SAAS;AACjB,QAAI,OAAO,QAAQ,aAAa,YAAY,QAAQ,aAAa,GAAG;AAClE,cAAQ,WAAW;AAAA,IACrB;AAAA,EACF;AAEA,MAAI,CAAC,QAAQ,YAAY,QAAQ,aAAa,GAAG;AAC/C,YAAQ,WAAW;AAAA,EACrB;AACF;"}
1
+ {"version":3,"file":"reporter.js","sources":["../../src/coverage/reporter.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\n\nimport type { FullResult } from \"@playwright/test/reporter\"\n\nimport { CoverageThresholdError, generateCoverageReport } from \"./report\"\nimport { removeCoverageFiles } from \"./files\"\nimport type { CoverageConfig } from \"./types\"\n\n\ntype FullResultWithExitCode = FullResult & { exitCode?: number }\n\nclass CoverageReporter {\n config: CoverageConfig\n\n constructor(options: { coverageConfig?: CoverageConfig } = {}) {\n this.config = options.coverageConfig as CoverageConfig\n if (!this.config) {\n throw new Error(\"CoverageReporter requires a coverageConfig option\")\n }\n }\n\n async onBegin() {\n if (process.env.RB_TEST_COMBINED_COVERAGE === \"1\") {\n return\n }\n\n if (!this.config.coverageEnabled) {\n return\n }\n\n await removeCoverageFiles(this.config)\n await fs.rm(this.config.coverageReportDir, { recursive: true, force: true })\n }\n\n async onEnd(result: FullResultWithExitCode) {\n if (process.env.RB_TEST_COMBINED_COVERAGE === \"1\") {\n return\n }\n\n if (!this.config.coverageEnabled) {\n return\n }\n\n if (result.status !== \"passed\") {\n console.warn(\"[coverage] skipping report generation because tests failed\")\n return\n }\n\n try {\n await generateCoverageReport(this.config)\n } catch (error) {\n if (error instanceof CoverageThresholdError) {\n console.error(error.message)\n setFailureExitCode(result)\n return\n }\n\n throw error\n }\n }\n}\n\nexport { CoverageReporter }\nexport default CoverageReporter\n\nfunction setFailureExitCode(result: FullResult): void {\n if (result && typeof result === \"object\") {\n const mutable = result as FullResultWithExitCode\n mutable.status = \"failed\"\n if (typeof mutable.exitCode !== \"number\" || mutable.exitCode === 0) {\n mutable.exitCode = 1\n }\n }\n\n if (!process.exitCode || process.exitCode === 0) {\n process.exitCode = 1\n }\n}\n"],"names":["CoverageReporter","config","constructor","options","coverageConfig","Error","onBegin","process","env","RB_TEST_COMBINED_COVERAGE","coverageEnabled","removeCoverageFiles","fs","rm","coverageReportDir","recursive","force","onEnd","result","status","console","warn","generateCoverageReport","error","CoverageThresholdError","message","setFailureExitCode","mutable","exitCode"],"mappings":";;;AAWA,MAAMA,iBAAiB;AAAA,EACrBC;AAAAA,EAEAC,YAAYC,UAA+C,IAAI;AAC7D,SAAKF,SAASE,QAAQC;AACtB,QAAI,CAAC,KAAKH,QAAQ;AAChB,YAAM,IAAII,MAAM,mDAAmD;AAAA,IACrE;AAAA,EACF;AAAA,EAEA,MAAMC,UAAU;AACd,QAAIC,QAAQC,IAAIC,8BAA8B,KAAK;AACjD;AAAA,IACF;AAEA,QAAI,CAAC,KAAKR,OAAOS,iBAAiB;AAChC;AAAA,IACF;AAEA,UAAMC,oBAAoB,KAAKV,MAAM;AACrC,UAAMW,GAAGC,GAAG,KAAKZ,OAAOa,mBAAmB;AAAA,MAAEC,WAAW;AAAA,MAAMC,OAAO;AAAA,IAAA,CAAM;AAAA,EAC7E;AAAA,EAEA,MAAMC,MAAMC,QAAgC;AAC1C,QAAIX,QAAQC,IAAIC,8BAA8B,KAAK;AACjD;AAAA,IACF;AAEA,QAAI,CAAC,KAAKR,OAAOS,iBAAiB;AAChC;AAAA,IACF;AAEA,QAAIQ,OAAOC,WAAW,UAAU;AAC9BC,cAAQC,KAAK,4DAA4D;AACzE;AAAA,IACF;AAEA,QAAI;AACF,YAAMC,uBAAuB,KAAKrB,MAAM;AAAA,IAC1C,SAASsB,OAAO;AACd,UAAIA,iBAAiBC,wBAAwB;AAC3CJ,gBAAQG,MAAMA,MAAME,OAAO;AAC3BC,2BAAmBR,MAAM;AACzB;AAAA,MACF;AAEA,YAAMK;AAAAA,IACR;AAAA,EACF;AACF;AAKA,SAASG,mBAAmBR,QAA0B;AACpD,MAAIA,UAAU,OAAOA,WAAW,UAAU;AACxC,UAAMS,UAAUT;AAChBS,YAAQR,SAAS;AACjB,QAAI,OAAOQ,QAAQC,aAAa,YAAYD,QAAQC,aAAa,GAAG;AAClED,cAAQC,WAAW;AAAA,IACrB;AAAA,EACF;AAEA,MAAI,CAACrB,QAAQqB,YAAYrB,QAAQqB,aAAa,GAAG;AAC/CrB,YAAQqB,WAAW;AAAA,EACrB;AACF;"}
@@ -28,7 +28,10 @@ async function createCoverageTracker(page, config) {
28
28
  });
29
29
  });
30
30
  await session.send("Profiler.enable");
31
- await session.send("Profiler.startPreciseCoverage", { callCount: false, detailed: true });
31
+ await session.send("Profiler.startPreciseCoverage", {
32
+ callCount: false,
33
+ detailed: true
34
+ });
32
35
  return {
33
36
  async stop(testInfo) {
34
37
  try {
@@ -37,7 +40,9 @@ async function createCoverageTracker(page, config) {
37
40
  return;
38
41
  }
39
42
  const outputFile = resolveCoverageOutputFile(config, testInfo);
40
- await fs.mkdir(path.dirname(outputFile), { recursive: true });
43
+ await fs.mkdir(path.dirname(outputFile), {
44
+ recursive: true
45
+ });
41
46
  await fs.writeFile(outputFile, JSON.stringify(payload, null, 2), "utf8");
42
47
  } finally {
43
48
  await shutdownSession(session);
@@ -52,7 +57,9 @@ function resolveCoverageOutputFile(config, testInfo) {
52
57
  return path.join(outputDir, config.coverageFileName);
53
58
  }
54
59
  async function collectCoveragePayload(session, scriptMeta, sourceCache, testInfo, config) {
55
- const { result } = await session.send("Profiler.takePreciseCoverage");
60
+ const {
61
+ result
62
+ } = await session.send("Profiler.takePreciseCoverage");
56
63
  await session.send("Profiler.stopPreciseCoverage");
57
64
  const scripts = [];
58
65
  for (const script of result) {
@@ -85,14 +92,18 @@ async function resolveScriptSource(session, cache, scriptId) {
85
92
  }
86
93
  async function fetchScriptSource(session, scriptId) {
87
94
  try {
88
- const sourceResponse = await session.send("Debugger.getScriptSource", { scriptId });
95
+ const sourceResponse = await session.send("Debugger.getScriptSource", {
96
+ scriptId
97
+ });
89
98
  return sourceResponse?.scriptSource ?? "";
90
99
  } catch (error) {
91
100
  const message = String(error?.message ?? error);
92
101
  if (message.includes("Debugger agent is not enabled")) {
93
102
  try {
94
103
  await session.send("Debugger.enable");
95
- const sourceResponse = await session.send("Debugger.getScriptSource", { scriptId });
104
+ const sourceResponse = await session.send("Debugger.getScriptSource", {
105
+ scriptId
106
+ });
96
107
  return sourceResponse?.scriptSource ?? "";
97
108
  } catch {
98
109
  return "";
@@ -102,11 +113,7 @@ async function fetchScriptSource(session, scriptId) {
102
113
  }
103
114
  }
104
115
  async function shutdownSession(session) {
105
- await Promise.allSettled([
106
- session.send("Profiler.stopPreciseCoverage").catch(() => void 0),
107
- session.send("Profiler.disable").catch(() => void 0),
108
- session.send("Debugger.disable").catch(() => void 0)
109
- ]);
116
+ await Promise.allSettled([session.send("Profiler.stopPreciseCoverage").catch(() => void 0), session.send("Profiler.disable").catch(() => void 0), session.send("Debugger.disable").catch(() => void 0)]);
110
117
  await session.detach().catch(() => void 0);
111
118
  }
112
119
  function normalizeScriptUrl(rawUrl, config) {
@@ -150,10 +157,7 @@ function createNormalizedPath(absolutePath, rootDir) {
150
157
  function stripQuery(url) {
151
158
  const queryIndex = url.indexOf("?");
152
159
  const hashIndex = url.indexOf("#");
153
- const endIndex = Math.min(
154
- queryIndex === -1 ? Number.POSITIVE_INFINITY : queryIndex,
155
- hashIndex === -1 ? Number.POSITIVE_INFINITY : hashIndex
156
- );
160
+ const endIndex = Math.min(queryIndex === -1 ? Number.POSITIVE_INFINITY : queryIndex, hashIndex === -1 ? Number.POSITIVE_INFINITY : hashIndex);
157
161
  if (!Number.isFinite(endIndex)) {
158
162
  return url;
159
163
  }
@@ -1 +1 @@
1
- {"version":3,"file":"v8-tracker.js","sources":["../../src/coverage/v8-tracker.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\nimport path from \"node:path\"\n\nimport type { Page, TestInfo } from \"@playwright/test\"\n\nimport { isInsideAnyRoot, resolveCollectCoverageRoots } from \"./collect\"\nimport type { CoverageConfig } from \"./types\"\n\n\nconst VITE_FS_PREFIX = \"/@fs/\"\n\nfunction sanitizePathSegment(input: unknown): string {\n const value = String(input ?? \"\").trim()\n return value.replace(/[^a-zA-Z0-9._-]+/g, \"_\") || \"unknown\"\n}\n\nfunction urlPathLooksLikeFile(pathname: unknown): boolean {\n const base = path.posix.basename(String(pathname ?? \"\"))\n return base.length > 0 && base.includes(\".\")\n}\n\nexport async function createCoverageTracker(\n page: Page,\n config: CoverageConfig,\n): Promise<{ stop: (testInfo: TestInfo) => Promise<void> }> {\n const session = await page.context().newCDPSession(page)\n const scriptMeta = new Map<string, { normalized: { absolutePath: string; relativePath: string } | null; url: string }>()\n const sourceCache = new Map<string, Promise<string>>()\n const scriptRoots = resolveCollectCoverageRoots(config.collectCoverageFrom, config.rootDir)\n\n await session.send(\"Debugger.enable\")\n session.on(\"Debugger.scriptParsed\", (event: any) => {\n if (!event.url) {\n return\n }\n\n const normalized = normalizeScriptUrl(event.url, config)\n const trackable = normalized\n && !isNodeModulesPath(normalized.absolutePath)\n && isInsideAnyRoot(normalized.absolutePath, scriptRoots)\n\n scriptMeta.set(event.scriptId, {\n normalized: trackable ? normalized : null,\n url: event.url,\n })\n })\n\n await session.send(\"Profiler.enable\")\n await session.send(\"Profiler.startPreciseCoverage\", { callCount: false, detailed: true })\n\n return {\n async stop(testInfo: TestInfo) {\n try {\n const payload = await collectCoveragePayload(session, scriptMeta, sourceCache, testInfo, config)\n\n if (payload.scripts.length === 0) {\n return\n }\n\n const outputFile = resolveCoverageOutputFile(config, testInfo)\n await fs.mkdir(path.dirname(outputFile), { recursive: true })\n await fs.writeFile(outputFile, JSON.stringify(payload, null, 2), \"utf8\")\n } finally {\n await shutdownSession(session)\n }\n },\n }\n}\n\nfunction resolveCoverageOutputFile(config: CoverageConfig, testInfo: TestInfo): string {\n const projectName = sanitizePathSegment(testInfo.project?.name)\n const testId = sanitizePathSegment(testInfo.testId)\n const outputDir = path.join(config.rootDir, \"build\", \"playwright\", \"coverage\", projectName, testId)\n return path.join(outputDir, config.coverageFileName)\n}\n\nasync function collectCoveragePayload(\n session: any,\n scriptMeta: Map<string, { normalized: { absolutePath: string; relativePath: string } | null; url: string }>,\n sourceCache: Map<string, Promise<string>>,\n testInfo: TestInfo,\n config: CoverageConfig,\n): Promise<{ testId: string; scripts: any[] }> {\n const { result } = await session.send(\"Profiler.takePreciseCoverage\")\n await session.send(\"Profiler.stopPreciseCoverage\")\n\n const scripts = []\n\n for (const script of result) {\n const meta = scriptMeta.get(script.scriptId)\n if (!meta || !meta.normalized) {\n continue\n }\n\n const source = await resolveScriptSource(session, sourceCache, script.scriptId)\n scripts.push({\n absolutePath: meta.normalized.absolutePath,\n relativePath: meta.normalized.relativePath,\n source,\n functions: script.functions,\n url: meta.url,\n })\n }\n\n return {\n testId: testInfo.titlePath.join(\" › \"),\n scripts,\n }\n}\n\nasync function resolveScriptSource(session: any, cache: Map<string, Promise<string>>, scriptId: string): Promise<string> {\n const cached = cache.get(scriptId)\n if (cached) {\n return cached\n }\n\n const promise = fetchScriptSource(session, scriptId)\n cache.set(scriptId, promise)\n return promise\n}\n\nasync function fetchScriptSource(session: any, scriptId: string): Promise<string> {\n try {\n const sourceResponse = await session.send(\"Debugger.getScriptSource\", { scriptId })\n return sourceResponse?.scriptSource ?? \"\"\n } catch (error: any) {\n const message = String(error?.message ?? error)\n if (message.includes(\"Debugger agent is not enabled\")) {\n try {\n await session.send(\"Debugger.enable\")\n const sourceResponse = await session.send(\"Debugger.getScriptSource\", { scriptId })\n return sourceResponse?.scriptSource ?? \"\"\n } catch {\n return \"\"\n }\n }\n\n return \"\"\n }\n}\n\nasync function shutdownSession(session: any): Promise<void> {\n await Promise.allSettled([\n session.send(\"Profiler.stopPreciseCoverage\").catch(() => undefined),\n session.send(\"Profiler.disable\").catch(() => undefined),\n session.send(\"Debugger.disable\").catch(() => undefined),\n ])\n await session.detach().catch(() => undefined)\n}\n\nfunction normalizeScriptUrl(rawUrl: string, config: CoverageConfig): { absolutePath: string; relativePath: string } | null {\n if (!rawUrl || rawUrl.startsWith(\"node:\")) {\n return null\n }\n\n const cleaned = stripQuery(rawUrl)\n let pathname = cleaned\n\n try {\n const parsed = new URL(cleaned)\n pathname = parsed.pathname\n } catch {\n // keep as-is for relative paths\n }\n\n if (!pathname) {\n return null\n }\n\n let absolutePath\n\n const decoded = decodeURIComponent(pathname)\n if (decoded.startsWith(VITE_FS_PREFIX)) {\n const fsPath = decoded.slice(VITE_FS_PREFIX.length)\n const normalizedFsPath = fsPath.startsWith(\"/\") || /^[A-Za-z]:\\//.test(fsPath)\n ? fsPath\n : `/${fsPath}`\n absolutePath = path.normalize(normalizedFsPath)\n } else if (decoded.startsWith(\"/\")) {\n if (!urlPathLooksLikeFile(decoded)) {\n return null\n }\n absolutePath = path.resolve(process.cwd(), `.${decoded}`)\n } else {\n return null\n }\n\n return createNormalizedPath(absolutePath, config.rootDir)\n}\n\nfunction createNormalizedPath(absolutePath: string, rootDir: string): { absolutePath: string; relativePath: string } {\n const normalizedAbsolute = path.normalize(absolutePath)\n const relativePath = path.relative(rootDir, normalizedAbsolute)\n return {\n absolutePath: normalizedAbsolute,\n relativePath,\n }\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 isNodeModulesPath(filePath: string): boolean {\n return path\n .normalize(String(filePath ?? \"\"))\n .split(path.sep)\n .includes(\"node_modules\")\n}\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"v8-tracker.js","sources":["../../src/coverage/v8-tracker.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\nimport path from \"node:path\"\n\nimport type { Page, TestInfo } from \"@playwright/test\"\n\nimport { isInsideAnyRoot, resolveCollectCoverageRoots } from \"./collect\"\nimport type { CoverageConfig } from \"./types\"\n\n\nconst VITE_FS_PREFIX = \"/@fs/\"\n\nfunction sanitizePathSegment(input: unknown): string {\n const value = String(input ?? \"\").trim()\n return value.replace(/[^a-zA-Z0-9._-]+/g, \"_\") || \"unknown\"\n}\n\nfunction urlPathLooksLikeFile(pathname: unknown): boolean {\n const base = path.posix.basename(String(pathname ?? \"\"))\n return base.length > 0 && base.includes(\".\")\n}\n\nexport async function createCoverageTracker(\n page: Page,\n config: CoverageConfig,\n): Promise<{ stop: (testInfo: TestInfo) => Promise<void> }> {\n const session = await page.context().newCDPSession(page)\n const scriptMeta = new Map<string, { normalized: { absolutePath: string; relativePath: string } | null; url: string }>()\n const sourceCache = new Map<string, Promise<string>>()\n const scriptRoots = resolveCollectCoverageRoots(config.collectCoverageFrom, config.rootDir)\n\n await session.send(\"Debugger.enable\")\n session.on(\"Debugger.scriptParsed\", (event: any) => {\n if (!event.url) {\n return\n }\n\n const normalized = normalizeScriptUrl(event.url, config)\n const trackable = normalized\n && !isNodeModulesPath(normalized.absolutePath)\n && isInsideAnyRoot(normalized.absolutePath, scriptRoots)\n\n scriptMeta.set(event.scriptId, {\n normalized: trackable ? normalized : null,\n url: event.url,\n })\n })\n\n await session.send(\"Profiler.enable\")\n await session.send(\"Profiler.startPreciseCoverage\", { callCount: false, detailed: true })\n\n return {\n async stop(testInfo: TestInfo) {\n try {\n const payload = await collectCoveragePayload(session, scriptMeta, sourceCache, testInfo, config)\n\n if (payload.scripts.length === 0) {\n return\n }\n\n const outputFile = resolveCoverageOutputFile(config, testInfo)\n await fs.mkdir(path.dirname(outputFile), { recursive: true })\n await fs.writeFile(outputFile, JSON.stringify(payload, null, 2), \"utf8\")\n } finally {\n await shutdownSession(session)\n }\n },\n }\n}\n\nfunction resolveCoverageOutputFile(config: CoverageConfig, testInfo: TestInfo): string {\n const projectName = sanitizePathSegment(testInfo.project?.name)\n const testId = sanitizePathSegment(testInfo.testId)\n const outputDir = path.join(config.rootDir, \"build\", \"playwright\", \"coverage\", projectName, testId)\n return path.join(outputDir, config.coverageFileName)\n}\n\nasync function collectCoveragePayload(\n session: any,\n scriptMeta: Map<string, { normalized: { absolutePath: string; relativePath: string } | null; url: string }>,\n sourceCache: Map<string, Promise<string>>,\n testInfo: TestInfo,\n config: CoverageConfig,\n): Promise<{ testId: string; scripts: any[] }> {\n const { result } = await session.send(\"Profiler.takePreciseCoverage\")\n await session.send(\"Profiler.stopPreciseCoverage\")\n\n const scripts = []\n\n for (const script of result) {\n const meta = scriptMeta.get(script.scriptId)\n if (!meta || !meta.normalized) {\n continue\n }\n\n const source = await resolveScriptSource(session, sourceCache, script.scriptId)\n scripts.push({\n absolutePath: meta.normalized.absolutePath,\n relativePath: meta.normalized.relativePath,\n source,\n functions: script.functions,\n url: meta.url,\n })\n }\n\n return {\n testId: testInfo.titlePath.join(\" › \"),\n scripts,\n }\n}\n\nasync function resolveScriptSource(session: any, cache: Map<string, Promise<string>>, scriptId: string): Promise<string> {\n const cached = cache.get(scriptId)\n if (cached) {\n return cached\n }\n\n const promise = fetchScriptSource(session, scriptId)\n cache.set(scriptId, promise)\n return promise\n}\n\nasync function fetchScriptSource(session: any, scriptId: string): Promise<string> {\n try {\n const sourceResponse = await session.send(\"Debugger.getScriptSource\", { scriptId })\n return sourceResponse?.scriptSource ?? \"\"\n } catch (error: any) {\n const message = String(error?.message ?? error)\n if (message.includes(\"Debugger agent is not enabled\")) {\n try {\n await session.send(\"Debugger.enable\")\n const sourceResponse = await session.send(\"Debugger.getScriptSource\", { scriptId })\n return sourceResponse?.scriptSource ?? \"\"\n } catch {\n return \"\"\n }\n }\n\n return \"\"\n }\n}\n\nasync function shutdownSession(session: any): Promise<void> {\n await Promise.allSettled([\n session.send(\"Profiler.stopPreciseCoverage\").catch(() => undefined),\n session.send(\"Profiler.disable\").catch(() => undefined),\n session.send(\"Debugger.disable\").catch(() => undefined),\n ])\n await session.detach().catch(() => undefined)\n}\n\nfunction normalizeScriptUrl(rawUrl: string, config: CoverageConfig): { absolutePath: string; relativePath: string } | null {\n if (!rawUrl || rawUrl.startsWith(\"node:\")) {\n return null\n }\n\n const cleaned = stripQuery(rawUrl)\n let pathname = cleaned\n\n try {\n const parsed = new URL(cleaned)\n pathname = parsed.pathname\n } catch {\n // keep as-is for relative paths\n }\n\n if (!pathname) {\n return null\n }\n\n let absolutePath\n\n const decoded = decodeURIComponent(pathname)\n if (decoded.startsWith(VITE_FS_PREFIX)) {\n const fsPath = decoded.slice(VITE_FS_PREFIX.length)\n const normalizedFsPath = fsPath.startsWith(\"/\") || /^[A-Za-z]:\\//.test(fsPath)\n ? fsPath\n : `/${fsPath}`\n absolutePath = path.normalize(normalizedFsPath)\n } else if (decoded.startsWith(\"/\")) {\n if (!urlPathLooksLikeFile(decoded)) {\n return null\n }\n absolutePath = path.resolve(process.cwd(), `.${decoded}`)\n } else {\n return null\n }\n\n return createNormalizedPath(absolutePath, config.rootDir)\n}\n\nfunction createNormalizedPath(absolutePath: string, rootDir: string): { absolutePath: string; relativePath: string } {\n const normalizedAbsolute = path.normalize(absolutePath)\n const relativePath = path.relative(rootDir, normalizedAbsolute)\n return {\n absolutePath: normalizedAbsolute,\n relativePath,\n }\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 isNodeModulesPath(filePath: string): boolean {\n return path\n .normalize(String(filePath ?? \"\"))\n .split(path.sep)\n .includes(\"node_modules\")\n}\n"],"names":["VITE_FS_PREFIX","sanitizePathSegment","input","value","String","trim","replace","urlPathLooksLikeFile","pathname","base","path","posix","basename","length","includes","createCoverageTracker","page","config","session","context","newCDPSession","scriptMeta","Map","sourceCache","scriptRoots","resolveCollectCoverageRoots","collectCoverageFrom","rootDir","send","on","event","url","normalized","normalizeScriptUrl","trackable","isNodeModulesPath","absolutePath","isInsideAnyRoot","set","scriptId","callCount","detailed","stop","testInfo","payload","collectCoveragePayload","scripts","outputFile","resolveCoverageOutputFile","fs","mkdir","dirname","recursive","writeFile","JSON","stringify","shutdownSession","projectName","project","name","testId","outputDir","join","coverageFileName","result","script","meta","get","source","resolveScriptSource","push","relativePath","functions","titlePath","cache","cached","promise","fetchScriptSource","sourceResponse","scriptSource","error","message","Promise","allSettled","catch","undefined","detach","rawUrl","startsWith","cleaned","stripQuery","parsed","URL","decoded","decodeURIComponent","fsPath","slice","normalizedFsPath","test","normalize","resolve","process","cwd","createNormalizedPath","normalizedAbsolute","relative","queryIndex","indexOf","hashIndex","endIndex","Math","min","Number","POSITIVE_INFINITY","isFinite","filePath","split","sep"],"mappings":"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"}
package/dist/index.js CHANGED
@@ -11,12 +11,16 @@ import { generateCoverageReport } from "./coverage/report.js";
11
11
  import { createCoverageFixtures } from "./coverage/fixtures.js";
12
12
  import { createCoverageGlobalSetup } from "./coverage/global-setup.js";
13
13
  import { findCoverageFiles, removeCoverageFiles } from "./coverage/files.js";
14
- const coverageOptions = loadCoverageOptionsSync({ optional: true });
14
+ const coverageOptions = loadCoverageOptionsSync({
15
+ optional: true
16
+ });
15
17
  const coverageHarness = coverageOptions ? createCoverageHarness(coverageOptions) : null;
16
18
  const test = coverageHarness ? coverageHarness.extendTest(test$2) : test$2;
17
19
  const expect = expect$1;
18
20
  function defineConfig(userConfig = {}) {
19
- const normalized = { ...userConfig };
21
+ const normalized = {
22
+ ...userConfig
23
+ };
20
24
  const reporters = ensureReporterArray(normalized.reporter);
21
25
  if (coverageHarness?.config.coverageEnabled && process.env.RB_TEST_COMBINED_COVERAGE !== "1") {
22
26
  const coverageReporter = coverageHarness.reporterEntry();
@@ -58,7 +62,10 @@ function applySpecFileConventions(config) {
58
62
  return project;
59
63
  }
60
64
  const testMatch = project.use?.isMobile ? MOBILE_SPEC_MATCH : DESKTOP_SPEC_MATCH;
61
- return { ...project, testMatch };
65
+ return {
66
+ ...project,
67
+ testMatch
68
+ };
62
69
  });
63
70
  }
64
71
  export {
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/index.ts"],"sourcesContent":["import { test as baseTest, expect as baseExpect, defineConfig as playwrightDefineConfig, devices } from \"@playwright/test\"\n\nimport { loadCoverageOptionsSync } from \"./coverage/config-loader\"\nimport { createCoverageHarness } from \"./coverage\"\n\n\nexport * from \"@playwright/test\"\n\nexport { clearDatabase } from \"./clearDatabase\"\nexport * from \"./coverage\"\nexport * from \"./serverCoverage\"\n\nconst coverageOptions = loadCoverageOptionsSync({ optional: true })\nconst coverageHarness = coverageOptions ? createCoverageHarness(coverageOptions) : null\nexport const test: typeof baseTest = coverageHarness ? coverageHarness.extendTest(baseTest) : baseTest\nexport const expect = baseExpect\nexport { devices }\n\ntype PlaywrightConfig = Parameters<typeof playwrightDefineConfig>[0]\ntype ReporterEntry = [string, Record<string, unknown>?]\n\nexport function defineConfig(userConfig: PlaywrightConfig = {}) {\n const normalized: any = { ...userConfig }\n const reporters = ensureReporterArray(normalized.reporter)\n\n if (coverageHarness?.config.coverageEnabled && process.env.RB_TEST_COMBINED_COVERAGE !== \"1\") {\n const coverageReporter = coverageHarness.reporterEntry()\n if (!reporters.some(([name]) => name === coverageReporter[0])) {\n reporters.push(coverageReporter)\n }\n }\n\n normalized.reporter = reporters\n\n applySpecFileConventions(normalized)\n\n return playwrightDefineConfig(normalized)\n}\n\nfunction ensureReporterArray(reporter: any): ReporterEntry[] {\n if (!reporter) {\n return [[\"list\"]]\n }\n\n if (!Array.isArray(reporter)) {\n return [normalizeReporterEntry(reporter)]\n }\n\n return reporter.map((entry: any) => normalizeReporterEntry(entry))\n}\n\nfunction normalizeReporterEntry(entry: any): ReporterEntry {\n if (Array.isArray(entry)) {\n return entry as ReporterEntry\n }\n return [entry] as ReporterEntry\n}\n\nconst DESKTOP_SPEC_MATCH = [/\\.spec(?:\\.desktop)?\\.(?:[cm]?[jt]sx?)$/]\nconst MOBILE_SPEC_MATCH = [/\\.spec(?:\\.mobile)?\\.(?:[cm]?[jt]sx?)$/]\n\nfunction applySpecFileConventions(config: any): void {\n if (config.testMatch != null) {\n return\n }\n\n if (!Array.isArray(config.projects) || config.projects.length === 0) {\n config.testMatch = DESKTOP_SPEC_MATCH\n return\n }\n\n config.projects = config.projects.map((project: any) => {\n if (!project || project.testMatch != null) {\n return project\n }\n const testMatch = project.use?.isMobile ? MOBILE_SPEC_MATCH : DESKTOP_SPEC_MATCH\n return { ...project, testMatch }\n })\n}\n"],"names":["baseTest","baseExpect","playwrightDefineConfig"],"mappings":";;;;;;;;;;;;;AAYA,MAAM,kBAAkB,wBAAwB,EAAE,UAAU,MAAM;AAClE,MAAM,kBAAkB,kBAAkB,sBAAsB,eAAe,IAAI;AAC5E,MAAM,OAAwB,kBAAkB,gBAAgB,WAAWA,MAAQ,IAAIA;AACvF,MAAM,SAASC;AAMf,SAAS,aAAa,aAA+B,IAAI;AAC9D,QAAM,aAAkB,EAAE,GAAG,WAAA;AAC7B,QAAM,YAAY,oBAAoB,WAAW,QAAQ;AAEzD,MAAI,iBAAiB,OAAO,mBAAmB,QAAQ,IAAI,8BAA8B,KAAK;AAC5F,UAAM,mBAAmB,gBAAgB,cAAA;AACzC,QAAI,CAAC,UAAU,KAAK,CAAC,CAAC,IAAI,MAAM,SAAS,iBAAiB,CAAC,CAAC,GAAG;AAC7D,gBAAU,KAAK,gBAAgB;AAAA,IACjC;AAAA,EACF;AAEA,aAAW,WAAW;AAEtB,2BAAyB,UAAU;AAEnC,SAAOC,eAAuB,UAAU;AAC1C;AAEA,SAAS,oBAAoB,UAAgC;AAC3D,MAAI,CAAC,UAAU;AACb,WAAO,CAAC,CAAC,MAAM,CAAC;AAAA,EAClB;AAEA,MAAI,CAAC,MAAM,QAAQ,QAAQ,GAAG;AAC5B,WAAO,CAAC,uBAAuB,QAAQ,CAAC;AAAA,EAC1C;AAEA,SAAO,SAAS,IAAI,CAAC,UAAe,uBAAuB,KAAK,CAAC;AACnE;AAEA,SAAS,uBAAuB,OAA2B;AACzD,MAAI,MAAM,QAAQ,KAAK,GAAG;AACxB,WAAO;AAAA,EACT;AACA,SAAO,CAAC,KAAK;AACf;AAEA,MAAM,qBAAqB,CAAC,yCAAyC;AACrE,MAAM,oBAAoB,CAAC,wCAAwC;AAEnE,SAAS,yBAAyB,QAAmB;AACnD,MAAI,OAAO,aAAa,MAAM;AAC5B;AAAA,EACF;AAEA,MAAI,CAAC,MAAM,QAAQ,OAAO,QAAQ,KAAK,OAAO,SAAS,WAAW,GAAG;AACnE,WAAO,YAAY;AACnB;AAAA,EACF;AAEA,SAAO,WAAW,OAAO,SAAS,IAAI,CAAC,YAAiB;AACtD,QAAI,CAAC,WAAW,QAAQ,aAAa,MAAM;AACzC,aAAO;AAAA,IACT;AACA,UAAM,YAAY,QAAQ,KAAK,WAAW,oBAAoB;AAC9D,WAAO,EAAE,GAAG,SAAS,UAAA;AAAA,EACvB,CAAC;AACH;"}
1
+ {"version":3,"file":"index.js","sources":["../src/index.ts"],"sourcesContent":["import { test as baseTest, expect as baseExpect, defineConfig as playwrightDefineConfig, devices } from \"@playwright/test\"\n\nimport { loadCoverageOptionsSync } from \"./coverage/config-loader\"\nimport { createCoverageHarness } from \"./coverage\"\n\n\nexport * from \"@playwright/test\"\n\nexport { clearDatabase } from \"./clearDatabase\"\nexport * from \"./coverage\"\nexport * from \"./serverCoverage\"\n\nconst coverageOptions = loadCoverageOptionsSync({ optional: true })\nconst coverageHarness = coverageOptions ? createCoverageHarness(coverageOptions) : null\nexport const test: typeof baseTest = coverageHarness ? coverageHarness.extendTest(baseTest) : baseTest\nexport const expect = baseExpect\nexport { devices }\n\ntype PlaywrightConfig = Parameters<typeof playwrightDefineConfig>[0]\ntype ReporterEntry = [string, Record<string, unknown>?]\n\nexport function defineConfig(userConfig: PlaywrightConfig = {}) {\n const normalized: any = { ...userConfig }\n const reporters = ensureReporterArray(normalized.reporter)\n\n if (coverageHarness?.config.coverageEnabled && process.env.RB_TEST_COMBINED_COVERAGE !== \"1\") {\n const coverageReporter = coverageHarness.reporterEntry()\n if (!reporters.some(([name]) => name === coverageReporter[0])) {\n reporters.push(coverageReporter)\n }\n }\n\n normalized.reporter = reporters\n\n applySpecFileConventions(normalized)\n\n return playwrightDefineConfig(normalized)\n}\n\nfunction ensureReporterArray(reporter: any): ReporterEntry[] {\n if (!reporter) {\n return [[\"list\"]]\n }\n\n if (!Array.isArray(reporter)) {\n return [normalizeReporterEntry(reporter)]\n }\n\n return reporter.map((entry: any) => normalizeReporterEntry(entry))\n}\n\nfunction normalizeReporterEntry(entry: any): ReporterEntry {\n if (Array.isArray(entry)) {\n return entry as ReporterEntry\n }\n return [entry] as ReporterEntry\n}\n\nconst DESKTOP_SPEC_MATCH = [/\\.spec(?:\\.desktop)?\\.(?:[cm]?[jt]sx?)$/]\nconst MOBILE_SPEC_MATCH = [/\\.spec(?:\\.mobile)?\\.(?:[cm]?[jt]sx?)$/]\n\nfunction applySpecFileConventions(config: any): void {\n if (config.testMatch != null) {\n return\n }\n\n if (!Array.isArray(config.projects) || config.projects.length === 0) {\n config.testMatch = DESKTOP_SPEC_MATCH\n return\n }\n\n config.projects = config.projects.map((project: any) => {\n if (!project || project.testMatch != null) {\n return project\n }\n const testMatch = project.use?.isMobile ? MOBILE_SPEC_MATCH : DESKTOP_SPEC_MATCH\n return { ...project, testMatch }\n })\n}\n"],"names":["coverageOptions","loadCoverageOptionsSync","optional","coverageHarness","createCoverageHarness","test","extendTest","baseTest","expect","baseExpect","defineConfig","userConfig","normalized","reporters","ensureReporterArray","reporter","config","coverageEnabled","process","env","RB_TEST_COMBINED_COVERAGE","coverageReporter","reporterEntry","some","name","push","applySpecFileConventions","playwrightDefineConfig","Array","isArray","normalizeReporterEntry","map","entry","DESKTOP_SPEC_MATCH","MOBILE_SPEC_MATCH","testMatch","projects","length","project","use","isMobile"],"mappings":";;;;;;;;;;;;;AAYA,MAAMA,kBAAkBC,wBAAwB;AAAA,EAAEC,UAAU;AAAK,CAAC;AAClE,MAAMC,kBAAkBH,kBAAkBI,sBAAsBJ,eAAe,IAAI;AAC5E,MAAMK,OAAwBF,kBAAkBA,gBAAgBG,WAAWC,MAAQ,IAAIA;AACvF,MAAMC,SAASC;AAMf,SAASC,aAAaC,aAA+B,IAAI;AAC9D,QAAMC,aAAkB;AAAA,IAAE,GAAGD;AAAAA,EAAAA;AAC7B,QAAME,YAAYC,oBAAoBF,WAAWG,QAAQ;AAEzD,MAAIZ,iBAAiBa,OAAOC,mBAAmBC,QAAQC,IAAIC,8BAA8B,KAAK;AAC5F,UAAMC,mBAAmBlB,gBAAgBmB,cAAAA;AACzC,QAAI,CAACT,UAAUU,KAAK,CAAC,CAACC,IAAI,MAAMA,SAASH,iBAAiB,CAAC,CAAC,GAAG;AAC7DR,gBAAUY,KAAKJ,gBAAgB;AAAA,IACjC;AAAA,EACF;AAEAT,aAAWG,WAAWF;AAEtBa,2BAAyBd,UAAU;AAEnC,SAAOe,eAAuBf,UAAU;AAC1C;AAEA,SAASE,oBAAoBC,UAAgC;AAC3D,MAAI,CAACA,UAAU;AACb,WAAO,CAAC,CAAC,MAAM,CAAC;AAAA,EAClB;AAEA,MAAI,CAACa,MAAMC,QAAQd,QAAQ,GAAG;AAC5B,WAAO,CAACe,uBAAuBf,QAAQ,CAAC;AAAA,EAC1C;AAEA,SAAOA,SAASgB,IAAI,CAACC,UAAeF,uBAAuBE,KAAK,CAAC;AACnE;AAEA,SAASF,uBAAuBE,OAA2B;AACzD,MAAIJ,MAAMC,QAAQG,KAAK,GAAG;AACxB,WAAOA;AAAAA,EACT;AACA,SAAO,CAACA,KAAK;AACf;AAEA,MAAMC,qBAAqB,CAAC,yCAAyC;AACrE,MAAMC,oBAAoB,CAAC,wCAAwC;AAEnE,SAASR,yBAAyBV,QAAmB;AACnD,MAAIA,OAAOmB,aAAa,MAAM;AAC5B;AAAA,EACF;AAEA,MAAI,CAACP,MAAMC,QAAQb,OAAOoB,QAAQ,KAAKpB,OAAOoB,SAASC,WAAW,GAAG;AACnErB,WAAOmB,YAAYF;AACnB;AAAA,EACF;AAEAjB,SAAOoB,WAAWpB,OAAOoB,SAASL,IAAI,CAACO,YAAiB;AACtD,QAAI,CAACA,WAAWA,QAAQH,aAAa,MAAM;AACzC,aAAOG;AAAAA,IACT;AACA,UAAMH,YAAYG,QAAQC,KAAKC,WAAWN,oBAAoBD;AAC9D,WAAO;AAAA,MAAE,GAAGK;AAAAA,MAASH;AAAAA,IAAAA;AAAAA,EACvB,CAAC;AACH;"}
@@ -9,7 +9,9 @@ async function setupServerCoverage(config, options = {}) {
9
9
  return async () => {
10
10
  };
11
11
  }
12
- const startResponse = await fetch(`${baseURL}/api/dev/coverage/start`, { method: "POST" });
12
+ const startResponse = await fetch(`${baseURL}/api/dev/coverage/start`, {
13
+ method: "POST"
14
+ });
13
15
  if (!startResponse.ok) {
14
16
  const body = await startResponse.text().catch(() => "");
15
17
  if (!strict) {
@@ -19,7 +21,9 @@ async function setupServerCoverage(config, options = {}) {
19
21
  throw new Error(`Failed to start server coverage: ${startResponse.status} ${body}`);
20
22
  }
21
23
  return async function stopCoverage() {
22
- const stopResponse = await fetch(`${baseURL}/api/dev/coverage/stop`, { method: "POST" });
24
+ const stopResponse = await fetch(`${baseURL}/api/dev/coverage/stop`, {
25
+ method: "POST"
26
+ });
23
27
  if (!stopResponse.ok) {
24
28
  const body = await stopResponse.text().catch(() => "");
25
29
  throw new Error(`Failed to stop server coverage: ${stopResponse.status} ${body}`);
@@ -1 +1 @@
1
- {"version":3,"file":"serverCoverage.js","sources":["../src/serverCoverage.ts"],"sourcesContent":["import type { FullConfig } from \"@playwright/test\"\n\n\ntype SetupServerCoverageOptions = {\n baseURL?: string\n strict?: boolean\n}\n\nexport async function setupServerCoverage(\n config: FullConfig,\n options: SetupServerCoverageOptions = {},\n): Promise<() => Promise<void>> {\n if (process.env.RB_DISABLE_COVERAGE === \"1\") {\n return async () => {}\n }\n\n const strict = options.strict ?? true\n const baseURL = resolveBaseURL(config, options, strict)\n if (!baseURL) {\n return async () => {}\n }\n\n const startResponse = await fetch(`${baseURL}/api/dev/coverage/start`, { method: \"POST\" })\n if (!startResponse.ok) {\n const body = await startResponse.text().catch(() => \"\")\n if (!strict) {\n return async () => {}\n }\n throw new Error(`Failed to start server coverage: ${startResponse.status} ${body}`)\n }\n\n return async function stopCoverage(): Promise<void> {\n const stopResponse = await fetch(`${baseURL}/api/dev/coverage/stop`, { method: \"POST\" })\n if (!stopResponse.ok) {\n const body = await stopResponse.text().catch(() => \"\")\n throw new Error(`Failed to stop server coverage: ${stopResponse.status} ${body}`)\n }\n }\n}\n\nfunction resolveBaseURL(config: FullConfig, options: SetupServerCoverageOptions, strict: boolean): string | null {\n const raw = options.baseURL ?? (config.projects?.[0]?.use as any)?.baseURL\n ?? (process.env.PORT ? `http://localhost:${process.env.PORT}` : null)\n\n if (typeof raw !== \"string\" || !raw.trim()) {\n if (!strict) {\n return null\n }\n throw new Error(\"Cannot resolve Playwright baseURL (missing config.projects[0].use.baseURL and PORT)\")\n }\n\n return raw.replace(/\\/$/, \"\")\n}\n"],"names":[],"mappings":"AAQA,eAAsB,oBACpB,QACA,UAAsC,IACR;AAC9B,MAAI,QAAQ,IAAI,wBAAwB,KAAK;AAC3C,WAAO,YAAY;AAAA,IAAC;AAAA,EACtB;AAEA,QAAM,SAAS,QAAQ,UAAU;AACjC,QAAM,UAAU,eAAe,QAAQ,SAAS,MAAM;AACtD,MAAI,CAAC,SAAS;AACZ,WAAO,YAAY;AAAA,IAAC;AAAA,EACtB;AAEA,QAAM,gBAAgB,MAAM,MAAM,GAAG,OAAO,2BAA2B,EAAE,QAAQ,QAAQ;AACzF,MAAI,CAAC,cAAc,IAAI;AACrB,UAAM,OAAO,MAAM,cAAc,OAAO,MAAM,MAAM,EAAE;AACtD,QAAI,CAAC,QAAQ;AACX,aAAO,YAAY;AAAA,MAAC;AAAA,IACtB;AACA,UAAM,IAAI,MAAM,oCAAoC,cAAc,MAAM,IAAI,IAAI,EAAE;AAAA,EACpF;AAEA,SAAO,eAAe,eAA8B;AAClD,UAAM,eAAe,MAAM,MAAM,GAAG,OAAO,0BAA0B,EAAE,QAAQ,QAAQ;AACvF,QAAI,CAAC,aAAa,IAAI;AACpB,YAAM,OAAO,MAAM,aAAa,OAAO,MAAM,MAAM,EAAE;AACrD,YAAM,IAAI,MAAM,mCAAmC,aAAa,MAAM,IAAI,IAAI,EAAE;AAAA,IAClF;AAAA,EACF;AACF;AAEA,SAAS,eAAe,QAAoB,SAAqC,QAAgC;AAC/G,QAAM,MAAM,QAAQ,WAAY,OAAO,WAAW,CAAC,GAAG,KAAa,YAC7D,QAAQ,IAAI,OAAO,oBAAoB,QAAQ,IAAI,IAAI,KAAK;AAElE,MAAI,OAAO,QAAQ,YAAY,CAAC,IAAI,QAAQ;AAC1C,QAAI,CAAC,QAAQ;AACX,aAAO;AAAA,IACT;AACA,UAAM,IAAI,MAAM,qFAAqF;AAAA,EACvG;AAEA,SAAO,IAAI,QAAQ,OAAO,EAAE;AAC9B;"}
1
+ {"version":3,"file":"serverCoverage.js","sources":["../src/serverCoverage.ts"],"sourcesContent":["import type { FullConfig } from \"@playwright/test\"\n\n\ntype SetupServerCoverageOptions = {\n baseURL?: string\n strict?: boolean\n}\n\nexport async function setupServerCoverage(\n config: FullConfig,\n options: SetupServerCoverageOptions = {},\n): Promise<() => Promise<void>> {\n if (process.env.RB_DISABLE_COVERAGE === \"1\") {\n return async () => {}\n }\n\n const strict = options.strict ?? true\n const baseURL = resolveBaseURL(config, options, strict)\n if (!baseURL) {\n return async () => {}\n }\n\n const startResponse = await fetch(`${baseURL}/api/dev/coverage/start`, { method: \"POST\" })\n if (!startResponse.ok) {\n const body = await startResponse.text().catch(() => \"\")\n if (!strict) {\n return async () => {}\n }\n throw new Error(`Failed to start server coverage: ${startResponse.status} ${body}`)\n }\n\n return async function stopCoverage(): Promise<void> {\n const stopResponse = await fetch(`${baseURL}/api/dev/coverage/stop`, { method: \"POST\" })\n if (!stopResponse.ok) {\n const body = await stopResponse.text().catch(() => \"\")\n throw new Error(`Failed to stop server coverage: ${stopResponse.status} ${body}`)\n }\n }\n}\n\nfunction resolveBaseURL(config: FullConfig, options: SetupServerCoverageOptions, strict: boolean): string | null {\n const raw = options.baseURL ?? (config.projects?.[0]?.use as any)?.baseURL\n ?? (process.env.PORT ? `http://localhost:${process.env.PORT}` : null)\n\n if (typeof raw !== \"string\" || !raw.trim()) {\n if (!strict) {\n return null\n }\n throw new Error(\"Cannot resolve Playwright baseURL (missing config.projects[0].use.baseURL and PORT)\")\n }\n\n return raw.replace(/\\/$/, \"\")\n}\n"],"names":["setupServerCoverage","config","options","process","env","RB_DISABLE_COVERAGE","strict","baseURL","resolveBaseURL","startResponse","fetch","method","ok","body","text","catch","Error","status","stopCoverage","stopResponse","raw","projects","use","PORT","trim","replace"],"mappings":"AAQA,eAAsBA,oBACpBC,QACAC,UAAsC,IACR;AAC9B,MAAIC,QAAQC,IAAIC,wBAAwB,KAAK;AAC3C,WAAO,YAAY;AAAA,IAAC;AAAA,EACtB;AAEA,QAAMC,SAASJ,QAAQI,UAAU;AACjC,QAAMC,UAAUC,eAAeP,QAAQC,SAASI,MAAM;AACtD,MAAI,CAACC,SAAS;AACZ,WAAO,YAAY;AAAA,IAAC;AAAA,EACtB;AAEA,QAAME,gBAAgB,MAAMC,MAAM,GAAGH,OAAO,2BAA2B;AAAA,IAAEI,QAAQ;AAAA,EAAA,CAAQ;AACzF,MAAI,CAACF,cAAcG,IAAI;AACrB,UAAMC,OAAO,MAAMJ,cAAcK,OAAOC,MAAM,MAAM,EAAE;AACtD,QAAI,CAACT,QAAQ;AACX,aAAO,YAAY;AAAA,MAAC;AAAA,IACtB;AACA,UAAM,IAAIU,MAAM,oCAAoCP,cAAcQ,MAAM,IAAIJ,IAAI,EAAE;AAAA,EACpF;AAEA,SAAO,eAAeK,eAA8B;AAClD,UAAMC,eAAe,MAAMT,MAAM,GAAGH,OAAO,0BAA0B;AAAA,MAAEI,QAAQ;AAAA,IAAA,CAAQ;AACvF,QAAI,CAACQ,aAAaP,IAAI;AACpB,YAAMC,OAAO,MAAMM,aAAaL,OAAOC,MAAM,MAAM,EAAE;AACrD,YAAM,IAAIC,MAAM,mCAAmCG,aAAaF,MAAM,IAAIJ,IAAI,EAAE;AAAA,IAClF;AAAA,EACF;AACF;AAEA,SAASL,eAAeP,QAAoBC,SAAqCI,QAAgC;AAC/G,QAAMc,MAAMlB,QAAQK,WAAYN,OAAOoB,WAAW,CAAC,GAAGC,KAAaf,YAC7DJ,QAAQC,IAAImB,OAAO,oBAAoBpB,QAAQC,IAAImB,IAAI,KAAK;AAElE,MAAI,OAAOH,QAAQ,YAAY,CAACA,IAAII,QAAQ;AAC1C,QAAI,CAAClB,QAAQ;AACX,aAAO;AAAA,IACT;AACA,UAAM,IAAIU,MAAM,qFAAqF;AAAA,EACvG;AAEA,SAAOI,IAAIK,QAAQ,OAAO,EAAE;AAC9B;"}
@@ -0,0 +1,15 @@
1
+ export type SpecsMapEntry = {
2
+ spec: string;
3
+ files: string[];
4
+ failed: boolean;
5
+ };
6
+ export type SpecsMapIndex = {
7
+ version: 1;
8
+ generatedAt: string;
9
+ specs: SpecsMapEntry[];
10
+ byFile: Record<string, string[]>;
11
+ };
12
+ export declare function createSpecsByFileRecord(entries: Map<string, string[]>): Record<string, string[]>;
13
+ export declare function readSpecsMapIndex(indexPath: string): Promise<SpecsMapIndex | null>;
14
+ export declare function warnSpecsMapOutdated(indexPath: string, reason: string): void;
15
+ //# sourceMappingURL=specs-map.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"specs-map.d.ts","sourceRoot":"","sources":["../src/specs-map.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,aAAa,GAAG;IAC1B,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,EAAE,MAAM,EAAE,CAAA;IACf,MAAM,EAAE,OAAO,CAAA;CAChB,CAAA;AAED,MAAM,MAAM,aAAa,GAAG;IAC1B,OAAO,EAAE,CAAC,CAAA;IACV,WAAW,EAAE,MAAM,CAAA;IACnB,KAAK,EAAE,aAAa,EAAE,CAAA;IACtB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAA;CACjC,CAAA;AAED,wBAAgB,uBAAuB,CAAC,OAAO,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAOhG;AAED,wBAAsB,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,CAMxF;AAED,wBAAgB,oBAAoB,CAAC,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,IAAI,CAI5E"}
@@ -0,0 +1,50 @@
1
+ import fs from "node:fs/promises";
2
+ function createSpecsByFileRecord(entries) {
3
+ const out = {};
4
+ for (const file of Array.from(entries.keys()).sort()) {
5
+ const specs = entries.get(file) ?? [];
6
+ out[file] = specs.sort();
7
+ }
8
+ return out;
9
+ }
10
+ async function readSpecsMapIndex(indexPath) {
11
+ const raw = await readJson(indexPath);
12
+ if (!raw || !isValidSpecsMapIndex(raw)) {
13
+ return null;
14
+ }
15
+ return raw;
16
+ }
17
+ function warnSpecsMapOutdated(indexPath, reason) {
18
+ console.warn(`[auto] Specs map index is unavailable or outdated: ${reason}`);
19
+ console.warn(`[auto] Expected index at ${indexPath}`);
20
+ console.warn("[auto] Rebuild with: rb-test --build-specs-map");
21
+ }
22
+ function isValidSpecsMapIndex(raw) {
23
+ if (!raw || typeof raw !== "object") {
24
+ return false;
25
+ }
26
+ if (raw.version !== 1) {
27
+ return false;
28
+ }
29
+ if (!Array.isArray(raw.specs)) {
30
+ return false;
31
+ }
32
+ if (raw.byFile === void 0 || raw.byFile === null || typeof raw.byFile !== "object") {
33
+ return false;
34
+ }
35
+ return true;
36
+ }
37
+ async function readJson(filePath) {
38
+ try {
39
+ const raw = await fs.readFile(filePath, "utf8");
40
+ return JSON.parse(raw);
41
+ } catch {
42
+ return null;
43
+ }
44
+ }
45
+ export {
46
+ createSpecsByFileRecord,
47
+ readSpecsMapIndex,
48
+ warnSpecsMapOutdated
49
+ };
50
+ //# sourceMappingURL=specs-map.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"specs-map.js","sources":["../src/specs-map.ts"],"sourcesContent":["import fs from \"node:fs/promises\"\n\n\nexport type SpecsMapEntry = {\n spec: string\n files: string[]\n failed: boolean\n}\n\nexport type SpecsMapIndex = {\n version: 1\n generatedAt: string\n specs: SpecsMapEntry[]\n byFile: Record<string, string[]>\n}\n\nexport function createSpecsByFileRecord(entries: Map<string, string[]>): Record<string, string[]> {\n const out: Record<string, string[]> = {}\n for (const file of Array.from(entries.keys()).sort()) {\n const specs = entries.get(file) ?? []\n out[file] = specs.sort()\n }\n return out\n}\n\nexport async function readSpecsMapIndex(indexPath: string): Promise<SpecsMapIndex | null> {\n const raw = await readJson(indexPath)\n if (!raw || !isValidSpecsMapIndex(raw)) {\n return null\n }\n return raw\n}\n\nexport function warnSpecsMapOutdated(indexPath: string, reason: string): void {\n console.warn(`[auto] Specs map index is unavailable or outdated: ${reason}`)\n console.warn(`[auto] Expected index at ${indexPath}`)\n console.warn(\"[auto] Rebuild with: rb-test --build-specs-map\")\n}\n\nfunction isValidSpecsMapIndex(raw: any): raw is SpecsMapIndex {\n if (!raw || typeof raw !== \"object\") {\n return false\n }\n if (raw.version !== 1) {\n return false\n }\n if (!Array.isArray(raw.specs)) {\n return false\n }\n if (raw.byFile === undefined || raw.byFile === null || typeof raw.byFile !== \"object\") {\n return false\n }\n return true\n}\n\nasync function readJson(filePath: string): Promise<unknown | null> {\n try {\n const raw = await fs.readFile(filePath, \"utf8\")\n return JSON.parse(raw)\n } catch {\n return null\n }\n}\n"],"names":["createSpecsByFileRecord","entries","out","file","Array","from","keys","sort","specs","get","readSpecsMapIndex","indexPath","raw","readJson","isValidSpecsMapIndex","warnSpecsMapOutdated","reason","console","warn","version","isArray","byFile","undefined","filePath","fs","readFile","JSON","parse"],"mappings":";AAgBO,SAASA,wBAAwBC,SAA0D;AAChG,QAAMC,MAAgC,CAAA;AACtC,aAAWC,QAAQC,MAAMC,KAAKJ,QAAQK,MAAM,EAAEC,QAAQ;AACpD,UAAMC,QAAQP,QAAQQ,IAAIN,IAAI,KAAK,CAAA;AACnCD,QAAIC,IAAI,IAAIK,MAAMD,KAAAA;AAAAA,EACpB;AACA,SAAOL;AACT;AAEA,eAAsBQ,kBAAkBC,WAAkD;AACxF,QAAMC,MAAM,MAAMC,SAASF,SAAS;AACpC,MAAI,CAACC,OAAO,CAACE,qBAAqBF,GAAG,GAAG;AACtC,WAAO;AAAA,EACT;AACA,SAAOA;AACT;AAEO,SAASG,qBAAqBJ,WAAmBK,QAAsB;AAC5EC,UAAQC,KAAK,sDAAsDF,MAAM,EAAE;AAC3EC,UAAQC,KAAK,4BAA4BP,SAAS,EAAE;AACpDM,UAAQC,KAAK,gDAAgD;AAC/D;AAEA,SAASJ,qBAAqBF,KAAgC;AAC5D,MAAI,CAACA,OAAO,OAAOA,QAAQ,UAAU;AACnC,WAAO;AAAA,EACT;AACA,MAAIA,IAAIO,YAAY,GAAG;AACrB,WAAO;AAAA,EACT;AACA,MAAI,CAACf,MAAMgB,QAAQR,IAAIJ,KAAK,GAAG;AAC7B,WAAO;AAAA,EACT;AACA,MAAII,IAAIS,WAAWC,UAAaV,IAAIS,WAAW,QAAQ,OAAOT,IAAIS,WAAW,UAAU;AACrF,WAAO;AAAA,EACT;AACA,SAAO;AACT;AAEA,eAAeR,SAASU,UAA2C;AACjE,MAAI;AACF,UAAMX,MAAM,MAAMY,GAAGC,SAASF,UAAU,MAAM;AAC9C,WAAOG,KAAKC,MAAMf,GAAG;AAAA,EACvB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;"}
@@ -11,9 +11,7 @@ function loadTsconfig() {
11
11
  try {
12
12
  ts = require$1("typescript");
13
13
  } catch (_err) {
14
- throw new Error(
15
- "@rpcbase/test: TypeScript is required to resolve path aliases. Please add it as a devDependency."
16
- );
14
+ throw new Error("@rpcbase/test: TypeScript is required to resolve path aliases. Please add it as a devDependency.");
17
15
  }
18
16
  const cwd = process.cwd();
19
17
  const configPath = ts.findConfigFile(cwd, ts.sys.fileExists, "tsconfig.json") || ts.findConfigFile(cwd, ts.sys.fileExists, "tsconfig.base.json");
@@ -24,11 +22,7 @@ function loadTsconfig() {
24
22
  if (configFile.error) {
25
23
  throw new Error(`@rpcbase/test: unable to read tsconfig at ${configPath}`);
26
24
  }
27
- const parsed = ts.parseJsonConfigFileContent(
28
- configFile.config,
29
- ts.sys,
30
- path.dirname(configPath)
31
- );
25
+ const parsed = ts.parseJsonConfigFileContent(configFile.config, ts.sys, path.dirname(configPath));
32
26
  return {
33
27
  path: configPath,
34
28
  baseUrl: parsed.options?.baseUrl ? path.resolve(path.dirname(configPath), parsed.options.baseUrl) : path.dirname(configPath),
@@ -45,9 +39,7 @@ function pathsToAlias(tsconfig2) {
45
39
  for (const target of targets) {
46
40
  if (key.includes("*") && target.includes("*")) {
47
41
  const wildcardPattern = target.endsWith("*") ? "(.+)" : "([^/]+)";
48
- const find2 = new RegExp(
49
- `^${escapeRegex(key).split("\\*").join(wildcardPattern)}$`
50
- );
42
+ const find2 = new RegExp(`^${escapeRegex(key).split("\\*").join(wildcardPattern)}$`);
51
43
  let groupIndex = 0;
52
44
  const replacedTarget = target.replace(/\*/g, () => `$${++groupIndex}`);
53
45
  const replacement2 = path.resolve(baseUrl, replacedTarget);
@@ -55,7 +47,10 @@ function pathsToAlias(tsconfig2) {
55
47
  if (checkPath.includes(`${path.sep}node_modules${path.sep}`) && !fs.existsSync(checkPath)) {
56
48
  continue;
57
49
  }
58
- alias2.push({ find: find2, replacement: replacement2 });
50
+ alias2.push({
51
+ find: find2,
52
+ replacement: replacement2
53
+ });
59
54
  continue;
60
55
  }
61
56
  const find = key.replace(/\*$/, "");
@@ -64,7 +59,10 @@ function pathsToAlias(tsconfig2) {
64
59
  if (replacement.includes(`${path.sep}node_modules${path.sep}`) && !fs.existsSync(replacement)) {
65
60
  continue;
66
61
  }
67
- alias2.push({ find, replacement });
62
+ alias2.push({
63
+ find,
64
+ replacement
65
+ });
68
66
  }
69
67
  }
70
68
  return alias2;
@@ -75,7 +73,9 @@ const vitest_config = defineConfig({
75
73
  test: {
76
74
  include: ["{src,lib}/**/*.test.{js,ts,tsx}"]
77
75
  },
78
- resolve: alias.length > 0 ? { alias } : void 0
76
+ resolve: alias.length > 0 ? {
77
+ alias
78
+ } : void 0
79
79
  });
80
80
  export {
81
81
  vitest_config as default