@rpcbase/test 0.357.0 → 0.358.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/clearDatabase.js +12 -10
- package/dist/clearDatabase.js.map +1 -1
- package/dist/cli.js +556 -439
- package/dist/cli.js.map +1 -1
- package/dist/coverage/collect.js +101 -63
- package/dist/coverage/collect.js.map +1 -1
- package/dist/coverage/config-loader.js +230 -180
- package/dist/coverage/config-loader.js.map +1 -1
- package/dist/coverage/config.js +100 -76
- package/dist/coverage/config.js.map +1 -1
- package/dist/coverage/console-text-report.js +220 -175
- package/dist/coverage/console-text-report.js.map +1 -1
- package/dist/coverage/files.js +58 -45
- package/dist/coverage/files.js.map +1 -1
- package/dist/coverage/fixtures.js +38 -27
- package/dist/coverage/fixtures.js.map +1 -1
- package/dist/coverage/global-setup.js +18 -15
- package/dist/coverage/global-setup.js.map +1 -1
- package/dist/coverage/index.js +55 -38
- package/dist/coverage/index.js.map +1 -1
- package/dist/coverage/report.js +466 -341
- package/dist/coverage/report.js.map +1 -1
- package/dist/coverage/reporter.js +60 -47
- package/dist/coverage/reporter.js.map +1 -1
- package/dist/coverage/v8-tracker.js +147 -115
- package/dist/coverage/v8-tracker.js.map +1 -1
- package/dist/index.js +75 -46
- package/dist/index.js.map +1 -1
- package/dist/runners/playwright.js +563 -438
- package/dist/runners/playwright.js.map +1 -1
- package/dist/runners/process.d.ts.map +1 -1
- package/dist/runners/process.js +183 -125
- package/dist/runners/process.js.map +1 -1
- package/dist/runners/vitest.js +171 -124
- package/dist/runners/vitest.js.map +1 -1
- package/dist/serverCoverage.js +42 -28
- package/dist/serverCoverage.js.map +1 -1
- package/dist/vitest.config.d.ts +1 -1
- package/dist/vitest.config.js +74 -62
- package/dist/vitest.config.js.map +1 -1
- package/package.json +6 -6
package/dist/cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"cli.js","names":["spawnSync","fs","fsPromises","path","createRequire","fg","createCoverageConfig","createCollectCoverageMatcher","loadCoverageOptions","removeCoverageFiles","collectCoveredFiles","CoverageThresholdError","generateCoverageReport","CoverageConfig","runPlaywright","resolveNodeCoverageDir","runVitest","require","import","meta","url","shouldForceTty","process","stdout","isTTY","env","FORCE_COLOR","VITEST_COVERAGE_CANDIDATES","IS_AIDER","undefined","console","warn","CoverageState","config","enabled","CliArgs","buildSpecsMap","auto","showMapping","list","passthroughArgs","RB_CLI_OPTIONS","const","isRbCliOption","arg","normalized","toLowerCase","some","option","startsWith","parseBooleanCliArg","rawArgs","optionPrefix","optionWithValuePrefix","value","slice","length","parseCliArgs","yargs","args","help","version","strict","exitProcess","parserConfiguration","options","Record","name","parseSync","parsed","type","default","Array","isArray","_","map","entry","String","filter","runTests","Promise","argv","filteredArgs","Error","playwrightCoverage","loadPlaywrightCoverageConfig","vitestCoverage","loadVitestCoverageConfig","combinedCoverage","resolveCombinedCoverage","buildSpecsMapFromCoverage","userArgs","resolveAutoPlaywrightArgs","listOnly","listPlaywrightSpecFiles","rootDir","cwd","shouldGenerateCoverageReport","cleanCoverageArtifacts","testError","disableCoverage","log","playwrightArgs","error","finalizeCoverage","then","exit","catch","stack","key","workspaceRoot","findWorkspaceRoot","specSourceFiles","findSpecSourceFiles","filesMapDir","join","testResultsRoot","rm","recursive","force","mkdir","specSourceFile","specProjectPath","relative","specWorkspacePath","toPosixPath","testFile","resolvePlaywrightSpecFile","failed","runError","coveredFiles","impactedFiles","filePath","relativePath","sort","outputFile","dirname","writeFile","JSON","stringify","spec","files","GitChange","kind","oldPath","newPath","mapFiles","findFilesMapJson","gitChanges","getGitChanges","renameMap","Map","change","specRootAbs","matchesCollectCoverageFrom","collectCoverageFrom","directSpecChanges","Set","sourceChanges","oldAbs","newAbs","isSpecSourceFile","existsSync","add","oldMatches","newMatches","push","abs","size","parsedMaps","file","json","readJson","specsByImpactedFile","resolvedSpec","resolveRenamedPath","get","set","unmappedSourceChanges","has","selectedSpecs","triggersBySpec","specs","forEach","current","oldSpecs","newSpecs","allSpecs","missingSpecs","specsToRun","from","triggers","trigger","playwrightFiles","specAbs","isSubpath","totalSpecFiles","playwrightFile","patterns","matches","absolute","onlyFiles","projectRoot","result","encoding","status","stderr","tokens","split","Boolean","changes","i","record","pathPart","isRenameOrCopyStatus","next","includes","original","visited","candidate","root","isAbsolute","absolutePath","specRootAbsolute","test","buildSpecRoot","isBuildSpecProject","builtCandidate","normalizeBuiltSpecPath","builtAbsolute","resolve","endsWith","replace","input","sep","dir","pkgPath","parse","readFileSync","workspaces","parent","optional","candidates","coverageEnabled","coverageReportDir","message","raw","readFile"],"sources":["../src/cli.ts"],"sourcesContent":["import { spawnSync } from \"node:child_process\"\nimport fs from \"node:fs\"\nimport fsPromises from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { createRequire } from \"node:module\"\n\nimport fg from \"fast-glob\"\n\nimport { createCoverageConfig } from \"./coverage/config\"\nimport { createCollectCoverageMatcher } from \"./coverage/collect\"\nimport { loadCoverageOptions } from \"./coverage/config-loader\"\nimport { removeCoverageFiles } from \"./coverage/files\"\nimport { collectCoveredFiles, CoverageThresholdError, generateCoverageReport } from \"./coverage/report\"\nimport type { CoverageConfig } from \"./coverage/types\"\nimport { runPlaywright } from \"./runners/playwright\"\nimport { resolveNodeCoverageDir, runVitest } from \"./runners/vitest\"\n\n\nconst require = createRequire(import.meta.url)\n\nconst shouldForceTty =\n !process.stdout.isTTY\n && process.env.FORCE_COLOR === \"true\"\n\nif (shouldForceTty) {\n require(\"./register-tty.cjs\")\n}\n\n\nconst VITEST_COVERAGE_CANDIDATES = [\"src/coverage.json\"]\n\nif (process.env.IS_AIDER !== undefined && process.env.IS_AIDER !== \"yes\") {\n console.warn(\"Warning: IS_AIDER is set to a value other than 'yes'.\")\n}\n\ntype CoverageState = { config: CoverageConfig; enabled: boolean } | null\n\ntype CliArgs = {\n buildSpecsMap: boolean\n auto: boolean\n showMapping: boolean\n list: boolean\n passthroughArgs: string[]\n}\n\nconst RB_CLI_OPTIONS = [\"build-specs-map\", \"auto\", \"full\", \"show-mapping\", \"list\"] as const\n\nfunction isRbCliOption(arg: string): boolean {\n const normalized = arg.toLowerCase()\n return RB_CLI_OPTIONS.some((option) => normalized === `--${option}` || normalized.startsWith(`--${option}=`))\n}\n\nfunction parseBooleanCliArg(rawArgs: string[], option: string): boolean {\n const optionPrefix = `--${option}`\n const optionWithValuePrefix = `${optionPrefix}=`\n return rawArgs.some((arg) => {\n if (arg === optionPrefix) {\n return true\n }\n if (!arg.startsWith(optionWithValuePrefix)) {\n return false\n }\n const value = arg.slice(optionWithValuePrefix.length).toLowerCase()\n if (value === \"false\" || value === \"0\") {\n return false\n }\n return value.length > 0\n })\n}\n\nfunction parseCliArgs(rawArgs: string[]): CliArgs {\n const yargs = require(\"yargs/yargs\") as (args: string[]) => {\n help: (enabled: boolean) => any\n version: (enabled: boolean) => any\n strict: (enabled: boolean) => any\n exitProcess: (enabled: boolean) => any\n parserConfiguration: (options: Record<string, unknown>) => any\n option: (name: string, options: Record<string, unknown>) => any\n parseSync: () => Record<string, unknown>\n }\n\n const parsed = yargs(rawArgs)\n .help(false)\n .version(false)\n .strict(false)\n .exitProcess(false)\n .parserConfiguration({\n \"unknown-options-as-args\": true,\n \"populate--\": true,\n \"strip-dashed\": true,\n \"strip-aliased\": true,\n })\n .option(\"build-specs-map\", { type: \"boolean\", default: false })\n .option(\"auto\", { type: \"boolean\", default: false })\n .option(\"full\", { type: \"boolean\", default: false })\n .option(\"show-mapping\", { type: \"boolean\", default: false })\n .option(\"list\", { type: \"boolean\", default: false })\n .parseSync()\n\n const passthroughArgs = [\n ...(Array.isArray(parsed._) ? parsed._ : []),\n ...(Array.isArray(parsed[\"--\"]) ? parsed[\"--\"] : []),\n ].map((entry) => String(entry)).filter((arg) => !isRbCliOption(arg))\n\n return {\n buildSpecsMap: parseBooleanCliArg(rawArgs, \"build-specs-map\"),\n auto: parseBooleanCliArg(rawArgs, \"auto\") && !parseBooleanCliArg(rawArgs, \"full\"),\n showMapping: parseBooleanCliArg(rawArgs, \"show-mapping\"),\n list: parseBooleanCliArg(rawArgs, \"list\"),\n passthroughArgs,\n }\n}\n\nasync function runTests(): Promise<void> {\n const args = parseCliArgs(process.argv.slice(2))\n const buildSpecsMap = args.buildSpecsMap\n const auto = args.auto && !buildSpecsMap\n const showMapping = args.showMapping\n const list = args.list\n const filteredArgs = args.passthroughArgs\n\n if (showMapping && !auto) {\n throw new Error(\"[rb-test] --show-mapping requires --auto\")\n }\n\n const playwrightCoverage = await loadPlaywrightCoverageConfig()\n const vitestCoverage = await loadVitestCoverageConfig()\n const combinedCoverage = resolveCombinedCoverage(playwrightCoverage, vitestCoverage)\n\n if (buildSpecsMap) {\n await buildSpecsMapFromCoverage({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n combinedCoverage,\n })\n return\n }\n\n if (list) {\n if (auto) {\n await resolveAutoPlaywrightArgs({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping,\n listOnly: true,\n })\n } else {\n await listPlaywrightSpecFiles(combinedCoverage?.config.rootDir ?? process.cwd())\n }\n return\n }\n\n const shouldGenerateCoverageReport = combinedCoverage?.enabled && !auto\n\n if (shouldGenerateCoverageReport) {\n await cleanCoverageArtifacts(combinedCoverage.config)\n }\n\n let testError = null\n\n try {\n await runVitest(vitestCoverage, combinedCoverage?.config ?? null, filteredArgs, { disableCoverage: auto })\n console.log(\"\\nRunning Playwright Tests...\")\n const playwrightArgs = auto\n ? await resolveAutoPlaywrightArgs({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping,\n })\n : filteredArgs\n if (playwrightArgs) {\n await runPlaywright(playwrightArgs, { disableCoverage: auto })\n }\n } catch (error: any) {\n testError = error\n }\n\n if (shouldGenerateCoverageReport) {\n if (testError) {\n console.warn(\"[coverage] skipping report generation because tests failed\")\n } else {\n try {\n await finalizeCoverage(combinedCoverage.config)\n } catch (error) {\n testError = error\n }\n }\n }\n\n if (testError) {\n throw testError\n }\n}\n\nrunTests()\n .then(() => process.exit(0))\n .catch((error) => {\n if (!(error instanceof CoverageThresholdError)) {\n console.error(error?.stack ?? String(error))\n }\n process.exit(1)\n })\n\nasync function buildSpecsMapFromCoverage({\n userArgs,\n combinedCoverage,\n}: {\n userArgs: string[]\n combinedCoverage: CoverageState\n [key: string]: unknown\n}): Promise<void> {\n if (!combinedCoverage?.enabled) {\n throw new Error(\"[specs-map] Coverage must be enabled to build the specs map.\")\n }\n\n const config = combinedCoverage.config\n const workspaceRoot = findWorkspaceRoot(process.cwd())\n\n const specSourceFiles = await findSpecSourceFiles(config.rootDir)\n if (specSourceFiles.length === 0) {\n throw new Error(\"[specs-map] No spec files found under spec/**/*.spec{,.desktop,.mobile}.ts\")\n }\n\n const filesMapDir = path.join(config.testResultsRoot, \"files-map\")\n await fsPromises.rm(filesMapDir, { recursive: true, force: true })\n await fsPromises.mkdir(filesMapDir, { recursive: true })\n\n for (const specSourceFile of specSourceFiles) {\n const specProjectPath = path.relative(config.rootDir, specSourceFile)\n const specWorkspacePath = toPosixPath(path.relative(workspaceRoot, specSourceFile))\n const testFile = resolvePlaywrightSpecFile(specProjectPath)\n\n console.log(`\\n[specs-map] Running ${specWorkspacePath}`)\n\n await removeCoverageFiles(config)\n\n let error = null\n let failed = false\n try {\n await runPlaywright([...userArgs, testFile])\n } catch (runError: any) {\n error = runError\n failed = true\n console.error(`[specs-map] Failed: ${specWorkspacePath}`)\n console.error(runError?.stack ?? String(runError))\n }\n\n const coveredFiles = await collectCoveredFiles(config)\n const impactedFiles = coveredFiles\n .map((filePath) => toPosixPath(path.relative(workspaceRoot, filePath)))\n .filter((relativePath) => relativePath && !relativePath.startsWith(\"../\") && relativePath !== \"..\")\n .sort()\n\n const outputFile = path.join(filesMapDir, `${specProjectPath}.json`)\n await fsPromises.mkdir(path.dirname(outputFile), { recursive: true })\n await fsPromises.writeFile(\n outputFile,\n JSON.stringify(\n {\n spec: specWorkspacePath,\n files: impactedFiles,\n failed,\n },\n null,\n 2,\n ),\n \"utf8\",\n )\n\n if (failed) {\n throw error\n }\n }\n}\n\ntype GitChange =\n | { kind: \"rename\"; oldPath: string; newPath: string }\n | { kind: \"path\"; path: string }\n\nasync function resolveAutoPlaywrightArgs({\n userArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping = false,\n listOnly = false,\n}: {\n userArgs: string[]\n playwrightCoverage: CoverageState\n vitestCoverage: CoverageState\n showMapping?: boolean\n listOnly?: boolean\n [key: string]: unknown\n}): Promise<string[] | null> {\n const config = playwrightCoverage?.config ?? vitestCoverage?.config ?? null\n if (!config) {\n console.warn(\"[auto] Coverage config not found; running full Playwright suite.\")\n return userArgs\n }\n\n const filesMapDir = path.join(config.testResultsRoot, \"files-map\")\n const mapFiles = await findFilesMapJson(filesMapDir)\n if (mapFiles.length === 0) {\n console.warn(\"[auto] Specs map not found; running full Playwright suite.\")\n return userArgs\n }\n\n const workspaceRoot = findWorkspaceRoot(process.cwd())\n const gitChanges = getGitChanges(workspaceRoot)\n const renameMap = new Map<string, string>(\n gitChanges\n .filter((change): change is { kind: \"rename\"; oldPath: string; newPath: string } => change.kind === \"rename\")\n .map((change) => [change.oldPath, change.newPath]),\n )\n\n const specRootAbs = path.join(config.rootDir, \"spec\")\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n const directSpecChanges = new Set<string>()\n const sourceChanges: GitChange[] = []\n\n for (const change of gitChanges) {\n if (change.kind === \"rename\") {\n const oldAbs = path.join(workspaceRoot, change.oldPath)\n const newAbs = path.join(workspaceRoot, change.newPath)\n\n if (isSpecSourceFile(newAbs, specRootAbs) && fs.existsSync(newAbs)) {\n directSpecChanges.add(change.newPath)\n }\n\n const oldMatches = matchesCollectCoverageFrom(oldAbs)\n const newMatches = matchesCollectCoverageFrom(newAbs)\n if (oldMatches || newMatches) {\n sourceChanges.push(change)\n }\n continue\n }\n\n const abs = path.join(workspaceRoot, change.path)\n\n if (isSpecSourceFile(abs, specRootAbs) && fs.existsSync(abs)) {\n directSpecChanges.add(change.path)\n }\n\n if (matchesCollectCoverageFrom(abs)) {\n sourceChanges.push(change)\n }\n }\n\n if (directSpecChanges.size === 0 && sourceChanges.length === 0) {\n if (listOnly) {\n console.log(\"[auto] No relevant git changes.\")\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.warn(\"[auto] No relevant git changes; running full Playwright suite.\")\n return userArgs\n }\n\n const parsedMaps: Array<{ spec: string; files: string[] }> = []\n for (const file of mapFiles) {\n const json = await readJson(file)\n if (!json) {\n continue\n }\n\n if (json.failed === true) {\n console.warn(\"[auto] Specs map contains failed entries; running full Playwright suite.\")\n return userArgs\n }\n\n const spec = typeof json?.spec === \"string\" ? json.spec : null\n if (!spec) {\n continue\n }\n\n const files = Array.isArray(json?.files)\n ? (json.files as unknown[]).filter((entry): entry is string => typeof entry === \"string\")\n : []\n parsedMaps.push({ spec, files })\n }\n\n if (parsedMaps.length === 0) {\n console.warn(\"[auto] Specs map is empty; running full Playwright suite.\")\n return userArgs\n }\n\n const specsByImpactedFile = new Map<string, string[]>()\n\n for (const entry of parsedMaps) {\n const resolvedSpec = resolveRenamedPath(entry.spec, renameMap)\n for (const file of entry.files) {\n const list = specsByImpactedFile.get(file) ?? []\n list.push(resolvedSpec)\n specsByImpactedFile.set(file, list)\n }\n }\n\n const unmappedSourceChanges = sourceChanges.filter((change) => {\n if (change.kind === \"path\") {\n return !specsByImpactedFile.has(change.path)\n }\n return !specsByImpactedFile.has(change.oldPath) && !specsByImpactedFile.has(change.newPath)\n })\n\n if (unmappedSourceChanges.length > 0) {\n console.warn(\"[auto] Unmapped source changes detected:\")\n for (const change of unmappedSourceChanges) {\n if (change.kind === \"path\") {\n console.warn(` - ${change.path}`)\n } else {\n console.warn(` - ${change.oldPath} -> ${change.newPath}`)\n }\n }\n }\n\n const selectedSpecs = new Set<string>(directSpecChanges)\n const triggersBySpec = new Map<string, Set<string>>()\n\n for (const spec of directSpecChanges) {\n if (showMapping) {\n triggersBySpec.set(spec, new Set([spec]))\n }\n }\n\n for (const change of sourceChanges) {\n if (change.kind === \"path\") {\n const specs = specsByImpactedFile.get(change.path) ?? []\n specs.forEach((spec) => selectedSpecs.add(spec))\n if (showMapping) {\n for (const spec of specs) {\n const current = triggersBySpec.get(spec) ?? new Set()\n current.add(change.path)\n triggersBySpec.set(spec, current)\n }\n }\n continue\n }\n\n const oldSpecs = specsByImpactedFile.get(change.oldPath) ?? []\n oldSpecs.forEach((spec) => selectedSpecs.add(spec))\n const newSpecs = specsByImpactedFile.get(change.newPath) ?? []\n newSpecs.forEach((spec) => selectedSpecs.add(spec))\n if (showMapping) {\n const key = `${change.oldPath} -> ${change.newPath}`\n const allSpecs = new Set([...oldSpecs, ...newSpecs])\n for (const spec of allSpecs) {\n const current = triggersBySpec.get(spec) ?? new Set()\n current.add(key)\n triggersBySpec.set(spec, current)\n }\n }\n }\n\n const missingSpecs: string[] = []\n const specsToRun = Array.from(selectedSpecs)\n .filter((spec) => {\n const abs = path.join(workspaceRoot, spec)\n if (fs.existsSync(abs)) {\n return true\n }\n missingSpecs.push(spec)\n return false\n })\n .sort()\n if (missingSpecs.length > 0) {\n console.warn(`[auto] Ignoring ${missingSpecs.length} missing spec file(s):`)\n for (const spec of missingSpecs.sort()) {\n console.warn(` - ${spec}`)\n }\n }\n\n if (specsToRun.length === 0) {\n if (listOnly) {\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.log(\"[auto] No impacted specs.\")\n return null\n }\n\n if (showMapping) {\n console.log(\"[auto] Mapping:\")\n for (const spec of specsToRun) {\n const triggers = Array.from(triggersBySpec.get(spec) ?? []).sort()\n if (triggers.length === 0) {\n continue\n }\n console.log(` - ${spec}`)\n for (const trigger of triggers) {\n console.log(` <- ${trigger}`)\n }\n }\n }\n\n const playwrightFiles = specsToRun\n .map((specWorkspacePath) => path.join(workspaceRoot, specWorkspacePath))\n .filter((specAbs) => isSubpath(specAbs, config.rootDir))\n .map((specAbs) => {\n const specProjectPath = path.relative(config.rootDir, specAbs)\n return resolvePlaywrightSpecFile(specProjectPath)\n })\n\n if (playwrightFiles.length === 0) {\n if (listOnly) {\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.log(\"[auto] No impacted specs.\")\n return null\n }\n\n const totalSpecFiles = (await findSpecSourceFiles(config.rootDir)).length\n\n if (listOnly) {\n console.log(`[auto] Selected ${playwrightFiles.length}/${totalSpecFiles} spec file(s):`)\n for (const playwrightFile of playwrightFiles) {\n console.log(` - ${playwrightFile}`)\n }\n return null\n }\n\n console.log(`[auto] Running ${playwrightFiles.length}/${totalSpecFiles} spec file(s).`)\n return [...userArgs, ...playwrightFiles]\n}\n\nasync function findFilesMapJson(filesMapDir: string): Promise<string[]> {\n const patterns = [\"spec/**/*.spec{,.desktop,.mobile}.ts.json\", \"spec/**/*.spec{,.desktop,.mobile}.tsx.json\"]\n const matches = await fg(patterns, { cwd: filesMapDir, absolute: true, onlyFiles: true }).catch(() => [])\n return matches.sort()\n}\n\nasync function listPlaywrightSpecFiles(projectRoot: string): Promise<void> {\n const specSourceFiles = await findSpecSourceFiles(projectRoot)\n if (specSourceFiles.length === 0) {\n console.log(\"[list] No spec files found.\")\n return\n }\n\n const playwrightFiles = specSourceFiles\n .map((specSourceFile) => path.relative(projectRoot, specSourceFile))\n .map((specProjectPath) => resolvePlaywrightSpecFile(specProjectPath))\n\n console.log(`[list] Selected ${playwrightFiles.length} spec file(s):`)\n for (const playwrightFile of playwrightFiles) {\n console.log(` - ${playwrightFile}`)\n }\n}\n\nfunction getGitChanges(workspaceRoot: string): GitChange[] {\n const result = spawnSync(\"git\", [\"status\", \"--porcelain=1\", \"-z\"], {\n cwd: workspaceRoot,\n encoding: \"utf8\",\n })\n\n if (result.status !== 0) {\n throw new Error(`[auto] Failed to read git status: ${result.stderr || \"unknown error\"}`)\n }\n\n const tokens = String(result.stdout ?? \"\").split(\"\\0\").filter(Boolean)\n const changes: GitChange[] = []\n\n for (let i = 0; i < tokens.length; i += 1) {\n const record = tokens[i]\n if (record.length < 4) {\n continue\n }\n\n const status = record.slice(0, 2)\n const pathPart = toPosixPath(record.slice(3))\n\n if (isRenameOrCopyStatus(status)) {\n const next = tokens[i + 1]\n if (typeof next !== \"string\") {\n continue\n }\n changes.push({\n kind: \"rename\",\n oldPath: pathPart,\n newPath: toPosixPath(next),\n })\n i += 1\n continue\n }\n\n changes.push({\n kind: \"path\",\n path: pathPart,\n })\n }\n\n return changes\n}\n\nfunction isRenameOrCopyStatus(status: string): boolean {\n return status.includes(\"R\") || status.includes(\"C\")\n}\n\nfunction resolveRenamedPath(original: string, renameMap: Map<string, string>): string {\n let current = original\n const visited = new Set<string>()\n\n while (!visited.has(current)) {\n const next = renameMap.get(current)\n if (!next) {\n break\n }\n visited.add(current)\n current = next\n }\n\n return current\n}\n\nfunction isSubpath(candidate: string, root: string): boolean {\n const relative = path.relative(root, candidate)\n return relative === \"\" || (!relative.startsWith(\"..\") && !path.isAbsolute(relative))\n}\n\nfunction isSpecSourceFile(absolutePath: string, specRootAbsolute: string): boolean {\n if (!isSubpath(absolutePath, specRootAbsolute)) {\n return false\n }\n\n return /\\.spec(?:\\.(?:desktop|mobile))?\\.tsx?$/.test(absolutePath)\n}\n\nasync function findSpecSourceFiles(projectRoot: string): Promise<string[]> {\n const patterns = [\"spec/**/*.spec{,.desktop,.mobile}.ts\", \"spec/**/*.spec{,.desktop,.mobile}.tsx\"]\n const matches = await fg(patterns, { cwd: projectRoot, absolute: true, onlyFiles: true })\n return matches.sort()\n}\n\nfunction resolvePlaywrightSpecFile(specProjectPath: string): string {\n const buildSpecRoot = path.join(process.cwd(), \"build\", \"spec\")\n const isBuildSpecProject = fs.existsSync(buildSpecRoot)\n\n if (!isBuildSpecProject) {\n return specProjectPath\n }\n\n const builtCandidate = normalizeBuiltSpecPath(path.join(\"build\", specProjectPath))\n const builtAbsolute = path.resolve(process.cwd(), builtCandidate)\n if (!fs.existsSync(builtAbsolute)) {\n throw new Error(`[specs-map] Missing built spec file: ${builtCandidate}`)\n }\n\n return builtCandidate\n}\n\nfunction normalizeBuiltSpecPath(filePath: string): string {\n if (filePath.endsWith(\".ts\") || filePath.endsWith(\".tsx\")) {\n return `${filePath.replace(/\\.tsx?$/, \"\")}.js`\n }\n return filePath\n}\n\nfunction toPosixPath(input: unknown): string {\n return String(input ?? \"\").split(path.sep).join(\"/\")\n}\n\nfunction findWorkspaceRoot(projectRoot: string): string {\n let dir = path.resolve(projectRoot)\n\n while (true) {\n const pkgPath = path.join(dir, \"package.json\")\n try {\n if (fs.existsSync(pkgPath)) {\n const parsed = JSON.parse(fs.readFileSync(pkgPath, \"utf8\"))\n if (parsed && typeof parsed === \"object\" && parsed.workspaces) {\n return dir\n }\n }\n } catch {\n //\n }\n\n const parent = path.dirname(dir)\n if (parent === dir) {\n return path.resolve(projectRoot)\n }\n dir = parent\n }\n}\n\nasync function loadVitestCoverageConfig(): Promise<CoverageState> {\n const options = await loadCoverageOptions({\n optional: true,\n candidates: VITEST_COVERAGE_CANDIDATES,\n })\n if (!options) {\n return null\n }\n\n const config = createCoverageConfig(options)\n\n return {\n config,\n enabled: config.coverageEnabled,\n }\n}\n\nasync function loadPlaywrightCoverageConfig(): Promise<CoverageState> {\n const options = await loadCoverageOptions({ optional: true })\n if (!options) {\n return null\n }\n\n const config = createCoverageConfig(options)\n\n return {\n config,\n enabled: config.coverageEnabled,\n }\n}\n\nfunction resolveCombinedCoverage(playwrightCoverage: CoverageState, vitestCoverage: CoverageState): CoverageState {\n if (playwrightCoverage?.enabled) {\n return playwrightCoverage\n }\n\n if (vitestCoverage?.enabled) {\n return vitestCoverage\n }\n\n return null\n}\n\nasync function cleanCoverageArtifacts(config: CoverageConfig): Promise<void> {\n await removeCoverageFiles(config)\n await fsPromises.rm(config.coverageReportDir, { recursive: true, force: true })\n await fsPromises.rm(resolveNodeCoverageDir(config), { recursive: true, force: true })\n}\n\nasync function finalizeCoverage(config: CoverageConfig): Promise<void> {\n try {\n await generateCoverageReport(config)\n } catch (error: any) {\n if (error instanceof CoverageThresholdError) {\n console.error(error.message)\n }\n throw error\n }\n}\n\nasync function readJson(filePath: string): Promise<any | null> {\n try {\n const raw = await fsPromises.readFile(filePath, \"utf8\")\n return JSON.parse(raw)\n } catch {\n return null\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"cli.js","sources":["../src/cli.ts"],"sourcesContent":["import { spawnSync } from \"node:child_process\"\nimport fs from \"node:fs\"\nimport fsPromises from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { createRequire } from \"node:module\"\n\nimport fg from \"fast-glob\"\n\nimport { createCoverageConfig } from \"./coverage/config\"\nimport { createCollectCoverageMatcher } from \"./coverage/collect\"\nimport { loadCoverageOptions } from \"./coverage/config-loader\"\nimport { removeCoverageFiles } from \"./coverage/files\"\nimport { collectCoveredFiles, CoverageThresholdError, generateCoverageReport } from \"./coverage/report\"\nimport type { CoverageConfig } from \"./coverage/types\"\nimport { runPlaywright } from \"./runners/playwright\"\nimport { resolveNodeCoverageDir, runVitest } from \"./runners/vitest\"\n\n\nconst require = createRequire(import.meta.url)\n\nconst shouldForceTty =\n !process.stdout.isTTY\n && process.env.FORCE_COLOR === \"true\"\n\nif (shouldForceTty) {\n require(\"./register-tty.cjs\")\n}\n\n\nconst VITEST_COVERAGE_CANDIDATES = [\"src/coverage.json\"]\n\nif (process.env.IS_AIDER !== undefined && process.env.IS_AIDER !== \"yes\") {\n console.warn(\"Warning: IS_AIDER is set to a value other than 'yes'.\")\n}\n\ntype CoverageState = { config: CoverageConfig; enabled: boolean } | null\n\ntype CliArgs = {\n buildSpecsMap: boolean\n auto: boolean\n showMapping: boolean\n list: boolean\n passthroughArgs: string[]\n}\n\nconst RB_CLI_OPTIONS = [\"build-specs-map\", \"auto\", \"full\", \"show-mapping\", \"list\"] as const\n\nfunction isRbCliOption(arg: string): boolean {\n const normalized = arg.toLowerCase()\n return RB_CLI_OPTIONS.some((option) => normalized === `--${option}` || normalized.startsWith(`--${option}=`))\n}\n\nfunction parseBooleanCliArg(rawArgs: string[], option: string): boolean {\n const optionPrefix = `--${option}`\n const optionWithValuePrefix = `${optionPrefix}=`\n return rawArgs.some((arg) => {\n if (arg === optionPrefix) {\n return true\n }\n if (!arg.startsWith(optionWithValuePrefix)) {\n return false\n }\n const value = arg.slice(optionWithValuePrefix.length).toLowerCase()\n if (value === \"false\" || value === \"0\") {\n return false\n }\n return value.length > 0\n })\n}\n\nfunction parseCliArgs(rawArgs: string[]): CliArgs {\n const yargs = require(\"yargs/yargs\") as (args: string[]) => {\n help: (enabled: boolean) => any\n version: (enabled: boolean) => any\n strict: (enabled: boolean) => any\n exitProcess: (enabled: boolean) => any\n parserConfiguration: (options: Record<string, unknown>) => any\n option: (name: string, options: Record<string, unknown>) => any\n parseSync: () => Record<string, unknown>\n }\n\n const parsed = yargs(rawArgs)\n .help(false)\n .version(false)\n .strict(false)\n .exitProcess(false)\n .parserConfiguration({\n \"unknown-options-as-args\": true,\n \"populate--\": true,\n \"strip-dashed\": true,\n \"strip-aliased\": true,\n })\n .option(\"build-specs-map\", { type: \"boolean\", default: false })\n .option(\"auto\", { type: \"boolean\", default: false })\n .option(\"full\", { type: \"boolean\", default: false })\n .option(\"show-mapping\", { type: \"boolean\", default: false })\n .option(\"list\", { type: \"boolean\", default: false })\n .parseSync()\n\n const passthroughArgs = [\n ...(Array.isArray(parsed._) ? parsed._ : []),\n ...(Array.isArray(parsed[\"--\"]) ? parsed[\"--\"] : []),\n ].map((entry) => String(entry)).filter((arg) => !isRbCliOption(arg))\n\n return {\n buildSpecsMap: parseBooleanCliArg(rawArgs, \"build-specs-map\"),\n auto: parseBooleanCliArg(rawArgs, \"auto\") && !parseBooleanCliArg(rawArgs, \"full\"),\n showMapping: parseBooleanCliArg(rawArgs, \"show-mapping\"),\n list: parseBooleanCliArg(rawArgs, \"list\"),\n passthroughArgs,\n }\n}\n\nasync function runTests(): Promise<void> {\n const args = parseCliArgs(process.argv.slice(2))\n const buildSpecsMap = args.buildSpecsMap\n const auto = args.auto && !buildSpecsMap\n const showMapping = args.showMapping\n const list = args.list\n const filteredArgs = args.passthroughArgs\n\n if (showMapping && !auto) {\n throw new Error(\"[rb-test] --show-mapping requires --auto\")\n }\n\n const playwrightCoverage = await loadPlaywrightCoverageConfig()\n const vitestCoverage = await loadVitestCoverageConfig()\n const combinedCoverage = resolveCombinedCoverage(playwrightCoverage, vitestCoverage)\n\n if (buildSpecsMap) {\n await buildSpecsMapFromCoverage({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n combinedCoverage,\n })\n return\n }\n\n if (list) {\n if (auto) {\n await resolveAutoPlaywrightArgs({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping,\n listOnly: true,\n })\n } else {\n await listPlaywrightSpecFiles(combinedCoverage?.config.rootDir ?? process.cwd())\n }\n return\n }\n\n const shouldGenerateCoverageReport = combinedCoverage?.enabled && !auto\n\n if (shouldGenerateCoverageReport) {\n await cleanCoverageArtifacts(combinedCoverage.config)\n }\n\n let testError = null\n\n try {\n await runVitest(vitestCoverage, combinedCoverage?.config ?? null, filteredArgs, { disableCoverage: auto })\n console.log(\"\\nRunning Playwright Tests...\")\n const playwrightArgs = auto\n ? await resolveAutoPlaywrightArgs({\n userArgs: filteredArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping,\n })\n : filteredArgs\n if (playwrightArgs) {\n await runPlaywright(playwrightArgs, { disableCoverage: auto })\n }\n } catch (error: any) {\n testError = error\n }\n\n if (shouldGenerateCoverageReport) {\n if (testError) {\n console.warn(\"[coverage] skipping report generation because tests failed\")\n } else {\n try {\n await finalizeCoverage(combinedCoverage.config)\n } catch (error) {\n testError = error\n }\n }\n }\n\n if (testError) {\n throw testError\n }\n}\n\nrunTests()\n .then(() => process.exit(0))\n .catch((error) => {\n if (!(error instanceof CoverageThresholdError)) {\n console.error(error?.stack ?? String(error))\n }\n process.exit(1)\n })\n\nasync function buildSpecsMapFromCoverage({\n userArgs,\n combinedCoverage,\n}: {\n userArgs: string[]\n combinedCoverage: CoverageState\n [key: string]: unknown\n}): Promise<void> {\n if (!combinedCoverage?.enabled) {\n throw new Error(\"[specs-map] Coverage must be enabled to build the specs map.\")\n }\n\n const config = combinedCoverage.config\n const workspaceRoot = findWorkspaceRoot(process.cwd())\n\n const specSourceFiles = await findSpecSourceFiles(config.rootDir)\n if (specSourceFiles.length === 0) {\n throw new Error(\"[specs-map] No spec files found under spec/**/*.spec{,.desktop,.mobile}.ts\")\n }\n\n const filesMapDir = path.join(config.testResultsRoot, \"files-map\")\n await fsPromises.rm(filesMapDir, { recursive: true, force: true })\n await fsPromises.mkdir(filesMapDir, { recursive: true })\n\n for (const specSourceFile of specSourceFiles) {\n const specProjectPath = path.relative(config.rootDir, specSourceFile)\n const specWorkspacePath = toPosixPath(path.relative(workspaceRoot, specSourceFile))\n const testFile = resolvePlaywrightSpecFile(specProjectPath)\n\n console.log(`\\n[specs-map] Running ${specWorkspacePath}`)\n\n await removeCoverageFiles(config)\n\n let error = null\n let failed = false\n try {\n await runPlaywright([...userArgs, testFile])\n } catch (runError: any) {\n error = runError\n failed = true\n console.error(`[specs-map] Failed: ${specWorkspacePath}`)\n console.error(runError?.stack ?? String(runError))\n }\n\n const coveredFiles = await collectCoveredFiles(config)\n const impactedFiles = coveredFiles\n .map((filePath) => toPosixPath(path.relative(workspaceRoot, filePath)))\n .filter((relativePath) => relativePath && !relativePath.startsWith(\"../\") && relativePath !== \"..\")\n .sort()\n\n const outputFile = path.join(filesMapDir, `${specProjectPath}.json`)\n await fsPromises.mkdir(path.dirname(outputFile), { recursive: true })\n await fsPromises.writeFile(\n outputFile,\n JSON.stringify(\n {\n spec: specWorkspacePath,\n files: impactedFiles,\n failed,\n },\n null,\n 2,\n ),\n \"utf8\",\n )\n\n if (failed) {\n throw error\n }\n }\n}\n\ntype GitChange =\n | { kind: \"rename\"; oldPath: string; newPath: string }\n | { kind: \"path\"; path: string }\n\nasync function resolveAutoPlaywrightArgs({\n userArgs,\n playwrightCoverage,\n vitestCoverage,\n showMapping = false,\n listOnly = false,\n}: {\n userArgs: string[]\n playwrightCoverage: CoverageState\n vitestCoverage: CoverageState\n showMapping?: boolean\n listOnly?: boolean\n [key: string]: unknown\n}): Promise<string[] | null> {\n const config = playwrightCoverage?.config ?? vitestCoverage?.config ?? null\n if (!config) {\n console.warn(\"[auto] Coverage config not found; running full Playwright suite.\")\n return userArgs\n }\n\n const filesMapDir = path.join(config.testResultsRoot, \"files-map\")\n const mapFiles = await findFilesMapJson(filesMapDir)\n if (mapFiles.length === 0) {\n console.warn(\"[auto] Specs map not found; running full Playwright suite.\")\n return userArgs\n }\n\n const workspaceRoot = findWorkspaceRoot(process.cwd())\n const gitChanges = getGitChanges(workspaceRoot)\n const renameMap = new Map<string, string>(\n gitChanges\n .filter((change): change is { kind: \"rename\"; oldPath: string; newPath: string } => change.kind === \"rename\")\n .map((change) => [change.oldPath, change.newPath]),\n )\n\n const specRootAbs = path.join(config.rootDir, \"spec\")\n const matchesCollectCoverageFrom = createCollectCoverageMatcher(config.collectCoverageFrom, config.rootDir)\n\n const directSpecChanges = new Set<string>()\n const sourceChanges: GitChange[] = []\n\n for (const change of gitChanges) {\n if (change.kind === \"rename\") {\n const oldAbs = path.join(workspaceRoot, change.oldPath)\n const newAbs = path.join(workspaceRoot, change.newPath)\n\n if (isSpecSourceFile(newAbs, specRootAbs) && fs.existsSync(newAbs)) {\n directSpecChanges.add(change.newPath)\n }\n\n const oldMatches = matchesCollectCoverageFrom(oldAbs)\n const newMatches = matchesCollectCoverageFrom(newAbs)\n if (oldMatches || newMatches) {\n sourceChanges.push(change)\n }\n continue\n }\n\n const abs = path.join(workspaceRoot, change.path)\n\n if (isSpecSourceFile(abs, specRootAbs) && fs.existsSync(abs)) {\n directSpecChanges.add(change.path)\n }\n\n if (matchesCollectCoverageFrom(abs)) {\n sourceChanges.push(change)\n }\n }\n\n if (directSpecChanges.size === 0 && sourceChanges.length === 0) {\n if (listOnly) {\n console.log(\"[auto] No relevant git changes.\")\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.warn(\"[auto] No relevant git changes; running full Playwright suite.\")\n return userArgs\n }\n\n const parsedMaps: Array<{ spec: string; files: string[] }> = []\n for (const file of mapFiles) {\n const json = await readJson(file)\n if (!json) {\n continue\n }\n\n if (json.failed === true) {\n console.warn(\"[auto] Specs map contains failed entries; running full Playwright suite.\")\n return userArgs\n }\n\n const spec = typeof json?.spec === \"string\" ? json.spec : null\n if (!spec) {\n continue\n }\n\n const files = Array.isArray(json?.files)\n ? (json.files as unknown[]).filter((entry): entry is string => typeof entry === \"string\")\n : []\n parsedMaps.push({ spec, files })\n }\n\n if (parsedMaps.length === 0) {\n console.warn(\"[auto] Specs map is empty; running full Playwright suite.\")\n return userArgs\n }\n\n const specsByImpactedFile = new Map<string, string[]>()\n\n for (const entry of parsedMaps) {\n const resolvedSpec = resolveRenamedPath(entry.spec, renameMap)\n for (const file of entry.files) {\n const list = specsByImpactedFile.get(file) ?? []\n list.push(resolvedSpec)\n specsByImpactedFile.set(file, list)\n }\n }\n\n const unmappedSourceChanges = sourceChanges.filter((change) => {\n if (change.kind === \"path\") {\n return !specsByImpactedFile.has(change.path)\n }\n return !specsByImpactedFile.has(change.oldPath) && !specsByImpactedFile.has(change.newPath)\n })\n\n if (unmappedSourceChanges.length > 0) {\n console.warn(\"[auto] Unmapped source changes detected:\")\n for (const change of unmappedSourceChanges) {\n if (change.kind === \"path\") {\n console.warn(` - ${change.path}`)\n } else {\n console.warn(` - ${change.oldPath} -> ${change.newPath}`)\n }\n }\n }\n\n const selectedSpecs = new Set<string>(directSpecChanges)\n const triggersBySpec = new Map<string, Set<string>>()\n\n for (const spec of directSpecChanges) {\n if (showMapping) {\n triggersBySpec.set(spec, new Set([spec]))\n }\n }\n\n for (const change of sourceChanges) {\n if (change.kind === \"path\") {\n const specs = specsByImpactedFile.get(change.path) ?? []\n specs.forEach((spec) => selectedSpecs.add(spec))\n if (showMapping) {\n for (const spec of specs) {\n const current = triggersBySpec.get(spec) ?? new Set()\n current.add(change.path)\n triggersBySpec.set(spec, current)\n }\n }\n continue\n }\n\n const oldSpecs = specsByImpactedFile.get(change.oldPath) ?? []\n oldSpecs.forEach((spec) => selectedSpecs.add(spec))\n const newSpecs = specsByImpactedFile.get(change.newPath) ?? []\n newSpecs.forEach((spec) => selectedSpecs.add(spec))\n if (showMapping) {\n const key = `${change.oldPath} -> ${change.newPath}`\n const allSpecs = new Set([...oldSpecs, ...newSpecs])\n for (const spec of allSpecs) {\n const current = triggersBySpec.get(spec) ?? new Set()\n current.add(key)\n triggersBySpec.set(spec, current)\n }\n }\n }\n\n const missingSpecs: string[] = []\n const specsToRun = Array.from(selectedSpecs)\n .filter((spec) => {\n const abs = path.join(workspaceRoot, spec)\n if (fs.existsSync(abs)) {\n return true\n }\n missingSpecs.push(spec)\n return false\n })\n .sort()\n if (missingSpecs.length > 0) {\n console.warn(`[auto] Ignoring ${missingSpecs.length} missing spec file(s):`)\n for (const spec of missingSpecs.sort()) {\n console.warn(` - ${spec}`)\n }\n }\n\n if (specsToRun.length === 0) {\n if (listOnly) {\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.log(\"[auto] No impacted specs.\")\n return null\n }\n\n if (showMapping) {\n console.log(\"[auto] Mapping:\")\n for (const spec of specsToRun) {\n const triggers = Array.from(triggersBySpec.get(spec) ?? []).sort()\n if (triggers.length === 0) {\n continue\n }\n console.log(` - ${spec}`)\n for (const trigger of triggers) {\n console.log(` <- ${trigger}`)\n }\n }\n }\n\n const playwrightFiles = specsToRun\n .map((specWorkspacePath) => path.join(workspaceRoot, specWorkspacePath))\n .filter((specAbs) => isSubpath(specAbs, config.rootDir))\n .map((specAbs) => {\n const specProjectPath = path.relative(config.rootDir, specAbs)\n return resolvePlaywrightSpecFile(specProjectPath)\n })\n\n if (playwrightFiles.length === 0) {\n if (listOnly) {\n console.log(\"[list] No matched spec files.\")\n return null\n }\n console.log(\"[auto] No impacted specs.\")\n return null\n }\n\n const totalSpecFiles = (await findSpecSourceFiles(config.rootDir)).length\n\n if (listOnly) {\n console.log(`[auto] Selected ${playwrightFiles.length}/${totalSpecFiles} spec file(s):`)\n for (const playwrightFile of playwrightFiles) {\n console.log(` - ${playwrightFile}`)\n }\n return null\n }\n\n console.log(`[auto] Running ${playwrightFiles.length}/${totalSpecFiles} spec file(s).`)\n return [...userArgs, ...playwrightFiles]\n}\n\nasync function findFilesMapJson(filesMapDir: string): Promise<string[]> {\n const patterns = [\"spec/**/*.spec{,.desktop,.mobile}.ts.json\", \"spec/**/*.spec{,.desktop,.mobile}.tsx.json\"]\n const matches = await fg(patterns, { cwd: filesMapDir, absolute: true, onlyFiles: true }).catch(() => [])\n return matches.sort()\n}\n\nasync function listPlaywrightSpecFiles(projectRoot: string): Promise<void> {\n const specSourceFiles = await findSpecSourceFiles(projectRoot)\n if (specSourceFiles.length === 0) {\n console.log(\"[list] No spec files found.\")\n return\n }\n\n const playwrightFiles = specSourceFiles\n .map((specSourceFile) => path.relative(projectRoot, specSourceFile))\n .map((specProjectPath) => resolvePlaywrightSpecFile(specProjectPath))\n\n console.log(`[list] Selected ${playwrightFiles.length} spec file(s):`)\n for (const playwrightFile of playwrightFiles) {\n console.log(` - ${playwrightFile}`)\n }\n}\n\nfunction getGitChanges(workspaceRoot: string): GitChange[] {\n const result = spawnSync(\"git\", [\"status\", \"--porcelain=1\", \"-z\"], {\n cwd: workspaceRoot,\n encoding: \"utf8\",\n })\n\n if (result.status !== 0) {\n throw new Error(`[auto] Failed to read git status: ${result.stderr || \"unknown error\"}`)\n }\n\n const tokens = String(result.stdout ?? \"\").split(\"\\0\").filter(Boolean)\n const changes: GitChange[] = []\n\n for (let i = 0; i < tokens.length; i += 1) {\n const record = tokens[i]\n if (record.length < 4) {\n continue\n }\n\n const status = record.slice(0, 2)\n const pathPart = toPosixPath(record.slice(3))\n\n if (isRenameOrCopyStatus(status)) {\n const next = tokens[i + 1]\n if (typeof next !== \"string\") {\n continue\n }\n changes.push({\n kind: \"rename\",\n oldPath: pathPart,\n newPath: toPosixPath(next),\n })\n i += 1\n continue\n }\n\n changes.push({\n kind: \"path\",\n path: pathPart,\n })\n }\n\n return changes\n}\n\nfunction isRenameOrCopyStatus(status: string): boolean {\n return status.includes(\"R\") || status.includes(\"C\")\n}\n\nfunction resolveRenamedPath(original: string, renameMap: Map<string, string>): string {\n let current = original\n const visited = new Set<string>()\n\n while (!visited.has(current)) {\n const next = renameMap.get(current)\n if (!next) {\n break\n }\n visited.add(current)\n current = next\n }\n\n return current\n}\n\nfunction isSubpath(candidate: string, root: string): boolean {\n const relative = path.relative(root, candidate)\n return relative === \"\" || (!relative.startsWith(\"..\") && !path.isAbsolute(relative))\n}\n\nfunction isSpecSourceFile(absolutePath: string, specRootAbsolute: string): boolean {\n if (!isSubpath(absolutePath, specRootAbsolute)) {\n return false\n }\n\n return /\\.spec(?:\\.(?:desktop|mobile))?\\.tsx?$/.test(absolutePath)\n}\n\nasync function findSpecSourceFiles(projectRoot: string): Promise<string[]> {\n const patterns = [\"spec/**/*.spec{,.desktop,.mobile}.ts\", \"spec/**/*.spec{,.desktop,.mobile}.tsx\"]\n const matches = await fg(patterns, { cwd: projectRoot, absolute: true, onlyFiles: true })\n return matches.sort()\n}\n\nfunction resolvePlaywrightSpecFile(specProjectPath: string): string {\n const buildSpecRoot = path.join(process.cwd(), \"build\", \"spec\")\n const isBuildSpecProject = fs.existsSync(buildSpecRoot)\n\n if (!isBuildSpecProject) {\n return specProjectPath\n }\n\n const builtCandidate = normalizeBuiltSpecPath(path.join(\"build\", specProjectPath))\n const builtAbsolute = path.resolve(process.cwd(), builtCandidate)\n if (!fs.existsSync(builtAbsolute)) {\n throw new Error(`[specs-map] Missing built spec file: ${builtCandidate}`)\n }\n\n return builtCandidate\n}\n\nfunction normalizeBuiltSpecPath(filePath: string): string {\n if (filePath.endsWith(\".ts\") || filePath.endsWith(\".tsx\")) {\n return `${filePath.replace(/\\.tsx?$/, \"\")}.js`\n }\n return filePath\n}\n\nfunction toPosixPath(input: unknown): string {\n return String(input ?? \"\").split(path.sep).join(\"/\")\n}\n\nfunction findWorkspaceRoot(projectRoot: string): string {\n let dir = path.resolve(projectRoot)\n\n while (true) {\n const pkgPath = path.join(dir, \"package.json\")\n try {\n if (fs.existsSync(pkgPath)) {\n const parsed = JSON.parse(fs.readFileSync(pkgPath, \"utf8\"))\n if (parsed && typeof parsed === \"object\" && parsed.workspaces) {\n return dir\n }\n }\n } catch {\n //\n }\n\n const parent = path.dirname(dir)\n if (parent === dir) {\n return path.resolve(projectRoot)\n }\n dir = parent\n }\n}\n\nasync function loadVitestCoverageConfig(): Promise<CoverageState> {\n const options = await loadCoverageOptions({\n optional: true,\n candidates: VITEST_COVERAGE_CANDIDATES,\n })\n if (!options) {\n return null\n }\n\n const config = createCoverageConfig(options)\n\n return {\n config,\n enabled: config.coverageEnabled,\n }\n}\n\nasync function loadPlaywrightCoverageConfig(): Promise<CoverageState> {\n const options = await loadCoverageOptions({ optional: true })\n if (!options) {\n return null\n }\n\n const config = createCoverageConfig(options)\n\n return {\n config,\n enabled: config.coverageEnabled,\n }\n}\n\nfunction resolveCombinedCoverage(playwrightCoverage: CoverageState, vitestCoverage: CoverageState): CoverageState {\n if (playwrightCoverage?.enabled) {\n return playwrightCoverage\n }\n\n if (vitestCoverage?.enabled) {\n return vitestCoverage\n }\n\n return null\n}\n\nasync function cleanCoverageArtifacts(config: CoverageConfig): Promise<void> {\n await removeCoverageFiles(config)\n await fsPromises.rm(config.coverageReportDir, { recursive: true, force: true })\n await fsPromises.rm(resolveNodeCoverageDir(config), { recursive: true, force: true })\n}\n\nasync function finalizeCoverage(config: CoverageConfig): Promise<void> {\n try {\n await generateCoverageReport(config)\n } catch (error: any) {\n if (error instanceof CoverageThresholdError) {\n console.error(error.message)\n }\n throw error\n }\n}\n\nasync function readJson(filePath: string): Promise<any | null> {\n try {\n const raw = await fsPromises.readFile(filePath, \"utf8\")\n return JSON.parse(raw)\n } catch {\n return null\n }\n}\n"],"names":["require","createRequire","import","url","shouldForceTty","process","stdout","isTTY","env","FORCE_COLOR","VITEST_COVERAGE_CANDIDATES","IS_AIDER","undefined","console","warn","RB_CLI_OPTIONS","isRbCliOption","arg","normalized","toLowerCase","some","option","startsWith","parseBooleanCliArg","rawArgs","optionPrefix","optionWithValuePrefix","value","slice","length","parseCliArgs","yargs","parsed","help","version","strict","exitProcess","parserConfiguration","type","default","parseSync","passthroughArgs","Array","isArray","_","map","entry","String","filter","buildSpecsMap","auto","showMapping","list","runTests","args","argv","filteredArgs","Error","playwrightCoverage","loadPlaywrightCoverageConfig","vitestCoverage","loadVitestCoverageConfig","combinedCoverage","resolveCombinedCoverage","buildSpecsMapFromCoverage","userArgs","resolveAutoPlaywrightArgs","listOnly","listPlaywrightSpecFiles","config","rootDir","cwd","shouldGenerateCoverageReport","enabled","cleanCoverageArtifacts","testError","runVitest","disableCoverage","log","playwrightArgs","runPlaywright","error","finalizeCoverage","then","exit","catch","CoverageThresholdError","stack","workspaceRoot","findWorkspaceRoot","specSourceFiles","findSpecSourceFiles","filesMapDir","path","join","testResultsRoot","fsPromises","rm","recursive","force","mkdir","specSourceFile","specProjectPath","relative","specWorkspacePath","toPosixPath","testFile","resolvePlaywrightSpecFile","removeCoverageFiles","failed","runError","coveredFiles","collectCoveredFiles","impactedFiles","filePath","relativePath","sort","outputFile","dirname","writeFile","JSON","stringify","spec","files","mapFiles","findFilesMapJson","gitChanges","getGitChanges","renameMap","Map","change","kind","oldPath","newPath","specRootAbs","matchesCollectCoverageFrom","createCollectCoverageMatcher","collectCoverageFrom","directSpecChanges","Set","sourceChanges","oldAbs","newAbs","isSpecSourceFile","fs","existsSync","add","oldMatches","newMatches","push","abs","size","parsedMaps","file","json","readJson","specsByImpactedFile","resolvedSpec","resolveRenamedPath","get","set","unmappedSourceChanges","has","selectedSpecs","triggersBySpec","specs","forEach","current","oldSpecs","newSpecs","key","allSpecs","missingSpecs","specsToRun","from","triggers","trigger","playwrightFiles","specAbs","isSubpath","totalSpecFiles","playwrightFile","patterns","matches","fg","absolute","onlyFiles","projectRoot","result","spawnSync","encoding","status","stderr","tokens","split","Boolean","changes","i","record","pathPart","isRenameOrCopyStatus","next","includes","original","visited","candidate","root","isAbsolute","absolutePath","specRootAbsolute","test","buildSpecRoot","isBuildSpecProject","builtCandidate","normalizeBuiltSpecPath","builtAbsolute","resolve","endsWith","replace","input","sep","dir","pkgPath","parse","readFileSync","workspaces","parent","options","loadCoverageOptions","optional","candidates","createCoverageConfig","coverageEnabled","coverageReportDir","resolveNodeCoverageDir","generateCoverageReport","message","raw","readFile"],"mappings":"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"}
|
package/dist/coverage/collect.js
CHANGED
|
@@ -1,77 +1,115 @@
|
|
|
1
1
|
import path from "node:path";
|
|
2
2
|
import picomatch from "picomatch";
|
|
3
|
-
//#region src/coverage/collect.ts
|
|
4
3
|
function createCollectCoverageMatcher(patterns, rootDir) {
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
4
|
+
const normalizedRoot = path.resolve(String(rootDir ?? ""));
|
|
5
|
+
const includeMatchers = [];
|
|
6
|
+
const excludeMatchers = [];
|
|
7
|
+
if (Array.isArray(patterns)) {
|
|
8
|
+
for (const pattern of patterns) {
|
|
9
|
+
const raw = String(pattern ?? "").trim();
|
|
10
|
+
if (!raw) {
|
|
11
|
+
continue;
|
|
12
|
+
}
|
|
13
|
+
const isExclude = raw.startsWith("!");
|
|
14
|
+
const body = toPosix(isExclude ? raw.slice(1) : raw);
|
|
15
|
+
if (!body) {
|
|
16
|
+
continue;
|
|
17
|
+
}
|
|
18
|
+
const matcher = picomatch(body, {
|
|
19
|
+
dot: true
|
|
20
|
+
});
|
|
21
|
+
if (isExclude) {
|
|
22
|
+
excludeMatchers.push(matcher);
|
|
23
|
+
} else {
|
|
24
|
+
includeMatchers.push(matcher);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
if (includeMatchers.length === 0) {
|
|
29
|
+
return () => false;
|
|
30
|
+
}
|
|
31
|
+
return (absolutePath) => {
|
|
32
|
+
const normalizedAbsolute = path.resolve(String(absolutePath ?? ""));
|
|
33
|
+
if (!normalizedAbsolute) {
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
const relativePosix = toPosix(path.relative(normalizedRoot, normalizedAbsolute));
|
|
37
|
+
const absolutePosix = toPosix(normalizedAbsolute);
|
|
38
|
+
const candidates = /* @__PURE__ */ new Set([absolutePosix, relativePosix]);
|
|
39
|
+
if (relativePosix) {
|
|
40
|
+
candidates.add(`./${relativePosix}`);
|
|
41
|
+
}
|
|
42
|
+
const candidateList = Array.from(candidates);
|
|
43
|
+
const included = includeMatchers.some((matcher) => candidateList.some((candidate) => matcher(candidate)));
|
|
44
|
+
if (!included) {
|
|
45
|
+
return false;
|
|
46
|
+
}
|
|
47
|
+
return !excludeMatchers.some((matcher) => candidateList.some((candidate) => matcher(candidate)));
|
|
48
|
+
};
|
|
30
49
|
}
|
|
31
50
|
function toPosix(input) {
|
|
32
|
-
|
|
51
|
+
return String(input ?? "").split(path.sep).join("/");
|
|
33
52
|
}
|
|
34
53
|
function isInsideAnyRoot(absolutePath, roots) {
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
54
|
+
const normalizedAbsolute = path.resolve(String(absolutePath ?? ""));
|
|
55
|
+
return (Array.isArray(roots) ? roots : []).some((root) => {
|
|
56
|
+
const normalizedRoot = path.resolve(String(root ?? ""));
|
|
57
|
+
const relative = path.relative(normalizedRoot, normalizedAbsolute);
|
|
58
|
+
return relative === "" || !relative.startsWith("..") && !path.isAbsolute(relative);
|
|
59
|
+
});
|
|
41
60
|
}
|
|
42
61
|
function resolveCollectCoverageRoots(patterns, rootDir) {
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
62
|
+
const resolvedRootDir = path.resolve(String(rootDir ?? ""));
|
|
63
|
+
const roots = /* @__PURE__ */ new Set([resolvedRootDir]);
|
|
64
|
+
if (!Array.isArray(patterns)) {
|
|
65
|
+
return Array.from(roots);
|
|
66
|
+
}
|
|
67
|
+
for (const pattern of patterns) {
|
|
68
|
+
const raw = String(pattern ?? "").trim();
|
|
69
|
+
if (!raw) {
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
const cleaned = raw.startsWith("!") ? raw.slice(1) : raw;
|
|
73
|
+
if (!cleaned) {
|
|
74
|
+
continue;
|
|
75
|
+
}
|
|
76
|
+
const prefix = staticGlobPrefix(cleaned);
|
|
77
|
+
if (!prefix) {
|
|
78
|
+
continue;
|
|
79
|
+
}
|
|
80
|
+
roots.add(path.resolve(resolvedRootDir, prefix));
|
|
81
|
+
}
|
|
82
|
+
return Array.from(roots);
|
|
56
83
|
}
|
|
57
84
|
function staticGlobPrefix(pattern) {
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
85
|
+
const normalized = String(pattern ?? "").trim();
|
|
86
|
+
if (!normalized) {
|
|
87
|
+
return null;
|
|
88
|
+
}
|
|
89
|
+
const firstGlobIndex = findFirstGlobIndex(normalized);
|
|
90
|
+
const prefix = firstGlobIndex === -1 ? normalized : normalized.slice(0, firstGlobIndex);
|
|
91
|
+
if (!prefix) {
|
|
92
|
+
return null;
|
|
93
|
+
}
|
|
94
|
+
if (prefix.endsWith("/") || prefix.endsWith(path.sep)) {
|
|
95
|
+
return prefix;
|
|
96
|
+
}
|
|
97
|
+
const posix = prefix.split(path.sep).join("/");
|
|
98
|
+
return path.posix.dirname(posix);
|
|
66
99
|
}
|
|
67
100
|
function findFirstGlobIndex(value) {
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
101
|
+
for (let i = 0; i < value.length; i += 1) {
|
|
102
|
+
const char = value[i];
|
|
103
|
+
if (char === "*" || char === "?" || char === "[" || char === "{") {
|
|
104
|
+
return i;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return -1;
|
|
73
108
|
}
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
109
|
+
export {
|
|
110
|
+
createCollectCoverageMatcher,
|
|
111
|
+
isInsideAnyRoot,
|
|
112
|
+
resolveCollectCoverageRoots,
|
|
113
|
+
toPosix
|
|
114
|
+
};
|
|
115
|
+
//# sourceMappingURL=collect.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"collect.js","
|
|
1
|
+
{"version":3,"file":"collect.js","sources":["../../src/coverage/collect.ts"],"sourcesContent":["import path from \"node:path\"\n\nimport picomatch from \"picomatch\"\n\n\nexport function createCollectCoverageMatcher(patterns: unknown, rootDir: unknown): (absolutePath: string) => boolean {\n const normalizedRoot = path.resolve(String(rootDir ?? \"\"))\n\n const includeMatchers: Array<(candidate: string) => boolean> = []\n const excludeMatchers: Array<(candidate: string) => boolean> = []\n\n if (Array.isArray(patterns)) {\n for (const pattern of patterns) {\n const raw = String(pattern ?? \"\").trim()\n if (!raw) {\n continue\n }\n\n const isExclude = raw.startsWith(\"!\")\n const body = toPosix(isExclude ? raw.slice(1) : raw)\n if (!body) {\n continue\n }\n\n const matcher = picomatch(body, { dot: true })\n if (isExclude) {\n excludeMatchers.push(matcher)\n } else {\n includeMatchers.push(matcher)\n }\n }\n }\n\n if (includeMatchers.length === 0) {\n return () => false\n }\n\n return (absolutePath) => {\n const normalizedAbsolute = path.resolve(String(absolutePath ?? \"\"))\n if (!normalizedAbsolute) {\n return false\n }\n\n const relativePosix = toPosix(path.relative(normalizedRoot, normalizedAbsolute))\n const absolutePosix = toPosix(normalizedAbsolute)\n const candidates = new Set([absolutePosix, relativePosix])\n\n if (relativePosix) {\n candidates.add(`./${relativePosix}`)\n }\n\n const candidateList = Array.from(candidates)\n const included = includeMatchers.some((matcher) => candidateList.some((candidate) => matcher(candidate)))\n if (!included) {\n return false\n }\n\n return !excludeMatchers.some((matcher) => candidateList.some((candidate) => matcher(candidate)))\n }\n}\n\nexport function toPosix(input: unknown): string {\n return String(input ?? \"\").split(path.sep).join(\"/\")\n}\n\nexport function isInsideAnyRoot(absolutePath: string, roots: unknown): boolean {\n const normalizedAbsolute = path.resolve(String(absolutePath ?? \"\"))\n return (Array.isArray(roots) ? roots : []).some((root) => {\n const normalizedRoot = path.resolve(String(root ?? \"\"))\n const relative = path.relative(normalizedRoot, normalizedAbsolute)\n return relative === \"\" || (!relative.startsWith(\"..\") && !path.isAbsolute(relative))\n })\n}\n\nexport function resolveCollectCoverageRoots(patterns: unknown, rootDir: unknown): string[] {\n const resolvedRootDir = path.resolve(String(rootDir ?? \"\"))\n const roots = new Set([resolvedRootDir])\n\n if (!Array.isArray(patterns)) {\n return Array.from(roots)\n }\n\n for (const pattern of patterns) {\n const raw = String(pattern ?? \"\").trim()\n if (!raw) {\n continue\n }\n\n const cleaned = raw.startsWith(\"!\") ? raw.slice(1) : raw\n if (!cleaned) {\n continue\n }\n\n const prefix = staticGlobPrefix(cleaned)\n if (!prefix) {\n continue\n }\n\n roots.add(path.resolve(resolvedRootDir, prefix))\n }\n\n return Array.from(roots)\n}\n\nfunction staticGlobPrefix(pattern: unknown): string | null {\n const normalized = String(pattern ?? \"\").trim()\n if (!normalized) {\n return null\n }\n\n const firstGlobIndex = findFirstGlobIndex(normalized)\n const prefix = firstGlobIndex === -1 ? normalized : normalized.slice(0, firstGlobIndex)\n\n if (!prefix) {\n return null\n }\n\n if (prefix.endsWith(\"/\") || prefix.endsWith(path.sep)) {\n return prefix\n }\n\n const posix = prefix.split(path.sep).join(\"/\")\n return path.posix.dirname(posix)\n}\n\nfunction findFirstGlobIndex(value: string): number {\n for (let i = 0; i < value.length; i += 1) {\n const char = value[i]\n if (char === \"*\" || char === \"?\" || char === \"[\" || char === \"{\") {\n return i\n }\n }\n return -1\n}\n"],"names":["createCollectCoverageMatcher","patterns","rootDir","normalizedRoot","path","resolve","String","includeMatchers","excludeMatchers","Array","isArray","pattern","raw","trim","isExclude","startsWith","body","toPosix","slice","matcher","picomatch","dot","push","length","absolutePath","normalizedAbsolute","relativePosix","relative","absolutePosix","candidates","Set","add","candidateList","from","included","some","candidate","input","split","sep","join","isInsideAnyRoot","roots","root","isAbsolute","resolveCollectCoverageRoots","resolvedRootDir","cleaned","prefix","staticGlobPrefix","normalized","firstGlobIndex","findFirstGlobIndex","endsWith","posix","dirname","value","i","char"],"mappings":"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"}
|