From 184f4d5c7ce1218a3a4e3ac8229e8fd8a7ba5fea Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ari=20Perkki=C3=B6?= Date: Thu, 11 Jul 2024 09:50:18 +0300 Subject: [PATCH] fix(coverage): use project specific `vitenode` for uncovered files --- docs/guide/workspace.md | 4 +- packages/coverage-istanbul/src/provider.ts | 10 +- packages/coverage-v8/src/provider.ts | 9 +- packages/vitest/src/utils/coverage.ts | 33 +++- .../vitest.workspace.multi-transforms.ts | 78 +++++++++ .../fixtures/src/covered.custom-1 | 5 + .../fixtures/src/uncovered.custom-1 | 3 + .../fixtures/test/custom-1-syntax.test.ts | 8 + .../workspaces/custom-2/src/covered.custom-2 | 5 + .../custom-2/src/uncovered.custom-2 | 3 + .../custom-2/test/custom-2-syntax.test.ts | 8 + ...stom-file-covered-1-istanbul.snapshot.json | 83 ++++++++++ .../custom-file-covered-1-v8.snapshot.json | 150 ++++++++++++++++++ ...stom-file-covered-2-istanbul.snapshot.json | 83 ++++++++++ .../custom-file-covered-2-v8.snapshot.json | 150 ++++++++++++++++++ ...om-file-uncovered-1-istanbul.snapshot.json | 48 ++++++ .../custom-file-uncovered-1-v8.snapshot.json | 103 ++++++++++++ ...om-file-uncovered-2-istanbul.snapshot.json | 48 ++++++ .../custom-file-uncovered-2-v8.snapshot.json | 103 ++++++++++++ test/coverage-test/test/all.test.ts | 4 +- test/coverage-test/test/changed.test.ts | 1 + .../test/workspace.multi-transform.test.ts | 43 +++++ 22 files changed, 970 insertions(+), 12 deletions(-) create mode 100644 test/coverage-test/fixtures/configs/vitest.workspace.multi-transforms.ts create mode 100644 test/coverage-test/fixtures/src/covered.custom-1 create mode 100644 test/coverage-test/fixtures/src/uncovered.custom-1 create mode 100644 test/coverage-test/fixtures/test/custom-1-syntax.test.ts create mode 100644 test/coverage-test/fixtures/workspaces/custom-2/src/covered.custom-2 create mode 100644 test/coverage-test/fixtures/workspaces/custom-2/src/uncovered.custom-2 create mode 100644 test/coverage-test/fixtures/workspaces/custom-2/test/custom-2-syntax.test.ts create mode 100644 test/coverage-test/test/__snapshots__/custom-file-covered-1-istanbul.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-covered-1-v8.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-covered-2-istanbul.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-covered-2-v8.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-uncovered-1-istanbul.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-uncovered-1-v8.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-uncovered-2-istanbul.snapshot.json create mode 100644 test/coverage-test/test/__snapshots__/custom-file-uncovered-2-v8.snapshot.json create mode 100644 test/coverage-test/test/workspace.multi-transform.test.ts diff --git a/docs/guide/workspace.md b/docs/guide/workspace.md index 70818b605ca1c..52e1134649d93 100644 --- a/docs/guide/workspace.md +++ b/docs/guide/workspace.md @@ -185,6 +185,4 @@ All configuration options that are not supported inside a project config have !coveredFiles.includes(file)) .sort() - const cacheKey = new Date().getTime() const coverageMap = libCoverage.createCoverageMap({}) + // Make sure file is not served from cache so that instrumenter loads up requested file coverage + const cacheKey = new Date().getTime() + + const transform = this.createUncoveredFileTransformer(this.ctx) + // Note that these cannot be run parallel as synchronous instrumenter.lastFileCoverage // returns the coverage of the last transformed file for (const [index, filename] of uncoveredFiles.entries()) { debug('Uncovered file %s %d/%d', filename, index, uncoveredFiles.length) - // Make sure file is not served from cache so that instrumenter loads up requested file coverage - await this.ctx.vitenode.transformRequest(`${filename}?v=${cacheKey}`) + await transform(`${filename}?v=${cacheKey}`) + const lastCoverage = this.instrumenter.lastFileCoverage() coverageMap.addFileCoverage(lastCoverage) } diff --git a/packages/coverage-v8/src/provider.ts b/packages/coverage-v8/src/provider.ts index aa01dced6bfe7..803fbccb6db8c 100644 --- a/packages/coverage-v8/src/provider.ts +++ b/packages/coverage-v8/src/provider.ts @@ -363,6 +363,7 @@ export class V8CoverageProvider const transformResults = normalizeTransformResults( this.ctx.vitenode.fetchCache, ) + const transform = this.createUncoveredFileTransformer(this.ctx) const allFiles = await this.testExclude.glob(this.ctx.config.root) let includedFiles = allFiles.map(file => @@ -396,6 +397,7 @@ export class V8CoverageProvider const { originalSource, source } = await this.getSources( filename.href, transformResults, + transform, ) // Ignore empty files, e.g. files that contain only typescript types and no runtime code @@ -441,6 +443,7 @@ export class V8CoverageProvider private async getSources( url: string, transformResults: TransformResults, + transform: ReturnType, functions: Profiler.FunctionCoverage[] = [], ): Promise<{ source: string @@ -458,9 +461,7 @@ export class V8CoverageProvider if (!transformResult) { isExecuted = false - transformResult = await this.ctx.vitenode - .transformRequest(filePath) - .catch(() => null) + transformResult = await transform(filePath).catch(() => null) } const map = transformResult?.map as EncodedSourceMap | undefined @@ -515,6 +516,7 @@ export class V8CoverageProvider ? viteNode.fetchCaches[transformMode] : viteNode.fetchCache const transformResults = normalizeTransformResults(fetchCache) + const transform = this.createUncoveredFileTransformer(this.ctx) const scriptCoverages = coverage.result.filter(result => this.testExclude.shouldInstrument(fileURLToPath(result.url)), @@ -536,6 +538,7 @@ export class V8CoverageProvider const sources = await this.getSources( url, transformResults, + transform, functions, ) diff --git a/packages/vitest/src/utils/coverage.ts b/packages/vitest/src/utils/coverage.ts index e5755c3303731..814c27d17a85e 100644 --- a/packages/vitest/src/utils/coverage.ts +++ b/packages/vitest/src/utils/coverage.ts @@ -1,7 +1,7 @@ import { relative } from 'pathe' import mm from 'micromatch' import type { CoverageMap } from 'istanbul-lib-coverage' -import type { BaseCoverageOptions, ResolvedCoverageOptions } from '../types' +import type { BaseCoverageOptions, ResolvedCoverageOptions, Vitest } from '../types' type Threshold = 'lines' | 'functions' | 'statements' | 'branches' @@ -293,6 +293,37 @@ export class BaseCoverageProvider { return chunks }, []) } + + createUncoveredFileTransformer(ctx: Vitest) { + const servers = [ + ...ctx.projects.map(project => ({ + root: project.config.root, + vitenode: project.vitenode, + })), + // Check core last as it will match all files anyway + { root: ctx.config.root, vitenode: ctx.vitenode }, + ] + + return async function transformFile(filename: string) { + let lastError + + for (const { root, vitenode } of servers) { + if (!filename.startsWith(root)) { + continue + } + + try { + return await vitenode.transformRequest(filename) + } + catch (error) { + lastError = error + } + } + + // All vite-node servers failed to transform the file + throw lastError + } + } } /** diff --git a/test/coverage-test/fixtures/configs/vitest.workspace.multi-transforms.ts b/test/coverage-test/fixtures/configs/vitest.workspace.multi-transforms.ts new file mode 100644 index 0000000000000..ac704efcc2b4a --- /dev/null +++ b/test/coverage-test/fixtures/configs/vitest.workspace.multi-transforms.ts @@ -0,0 +1,78 @@ +import { Plugin, defineWorkspace } from "vitest/config"; +import MagicString from "magic-string"; +import { readFileSync } from "fs"; + +export default defineWorkspace([ + // Project that uses its own "root" and custom transform plugin + { + test: { + name: "custom-with-root", + root: "fixtures/workspaces/custom-2", + }, + plugins: [customFilePlugin("2")], + }, + + // Project that cannot transform "*.custom-x" files + { + test: { + name: "normal", + include: ["fixtures/test/math.test.ts"], + }, + }, + + // Project that uses default "root" and has custom transform plugin + { + test: { + name: "custom", + include: ["fixtures/test/custom-1-syntax.test.ts"], + }, + plugins: [customFilePlugin("1")], + }, +]); + +/** + * Plugin for transforming `.custom-1` and/or `.custom-2` files to Javascript + */ +function customFilePlugin(postfix: "1" | "2"): Plugin { + function transform(code: MagicString) { + code.replaceAll( + "", + ` +function covered() { + return "Custom-${postfix} file loaded!" +} + `.trim() + ); + + code.replaceAll( + "", + ` +function uncovered() { + return "This should be uncovered!" +} + `.trim() + ); + + code.replaceAll("", "export default covered()"); + code.replaceAll("", "export default uncovered()"); + } + + return { + name: `custom-${postfix}-file-plugin`, + transform(_, id) { + const filename = id.split("?")[0]; + + if (filename.endsWith(`.custom-${postfix}`)) { + const content = readFileSync(filename, "utf8"); + + const s = new MagicString(content); + transform(s); + + return { + code: s.toString(), + map: s.generateMap({ hires: "boundary" }), + }; + } + }, + }; +} diff --git a/test/coverage-test/fixtures/src/covered.custom-1 b/test/coverage-test/fixtures/src/covered.custom-1 new file mode 100644 index 0000000000000..095b52aa44c8f --- /dev/null +++ b/test/coverage-test/fixtures/src/covered.custom-1 @@ -0,0 +1,5 @@ + + + + + diff --git a/test/coverage-test/fixtures/src/uncovered.custom-1 b/test/coverage-test/fixtures/src/uncovered.custom-1 new file mode 100644 index 0000000000000..f4e16be2968b7 --- /dev/null +++ b/test/coverage-test/fixtures/src/uncovered.custom-1 @@ -0,0 +1,3 @@ + + + diff --git a/test/coverage-test/fixtures/test/custom-1-syntax.test.ts b/test/coverage-test/fixtures/test/custom-1-syntax.test.ts new file mode 100644 index 0000000000000..6da0aea4b5531 --- /dev/null +++ b/test/coverage-test/fixtures/test/custom-1-syntax.test.ts @@ -0,0 +1,8 @@ +import { expect, test } from 'vitest' + +// @ts-expect-error -- untyped +import output from '../src/covered.custom-1' + +test('custom file loads fine', () => { + expect(output).toMatch('Custom-1 file loaded!') +}) diff --git a/test/coverage-test/fixtures/workspaces/custom-2/src/covered.custom-2 b/test/coverage-test/fixtures/workspaces/custom-2/src/covered.custom-2 new file mode 100644 index 0000000000000..095b52aa44c8f --- /dev/null +++ b/test/coverage-test/fixtures/workspaces/custom-2/src/covered.custom-2 @@ -0,0 +1,5 @@ + + + + + diff --git a/test/coverage-test/fixtures/workspaces/custom-2/src/uncovered.custom-2 b/test/coverage-test/fixtures/workspaces/custom-2/src/uncovered.custom-2 new file mode 100644 index 0000000000000..f4e16be2968b7 --- /dev/null +++ b/test/coverage-test/fixtures/workspaces/custom-2/src/uncovered.custom-2 @@ -0,0 +1,3 @@ + + + diff --git a/test/coverage-test/fixtures/workspaces/custom-2/test/custom-2-syntax.test.ts b/test/coverage-test/fixtures/workspaces/custom-2/test/custom-2-syntax.test.ts new file mode 100644 index 0000000000000..ffc0d6c55b6ec --- /dev/null +++ b/test/coverage-test/fixtures/workspaces/custom-2/test/custom-2-syntax.test.ts @@ -0,0 +1,8 @@ +import { expect, test } from 'vitest' + +// @ts-expect-error -- untyped +import output from '../src/covered.custom-2' + +test('custom-2 file loads fine', () => { + expect(output).toMatch('Custom-2 file loaded!') +}) diff --git a/test/coverage-test/test/__snapshots__/custom-file-covered-1-istanbul.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-covered-1-istanbul.snapshot.json new file mode 100644 index 0000000000000..8ebdab27587b5 --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-covered-1-istanbul.snapshot.json @@ -0,0 +1,83 @@ +{ + "path": "/fixtures/src/covered.custom-1", + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "1": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + } + }, + "fnMap": { + "0": { + "name": "covered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + } + }, + "1": { + "name": "uncovered", + "decl": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "loc": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + } + } + }, + "branchMap": {}, + "s": { + "0": 1, + "1": 0 + }, + "f": { + "0": 1, + "1": 0 + }, + "b": {} +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-covered-1-v8.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-covered-1-v8.snapshot.json new file mode 100644 index 0000000000000..35a17c7894b8b --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-covered-1-v8.snapshot.json @@ -0,0 +1,150 @@ +{ + "path": "/fixtures/src/covered.custom-1", + "all": false, + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "1": { + "start": { + "line": 2, + "column": 0 + }, + "end": { + "line": 2, + "column": 0 + } + }, + "2": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "3": { + "start": { + "line": 4, + "column": 0 + }, + "end": { + "line": 4, + "column": 0 + } + }, + "4": { + "start": { + "line": 5, + "column": 0 + }, + "end": { + "line": 5, + "column": 24 + } + } + }, + "s": { + "0": 1, + "1": 1, + "2": 0, + "3": 1, + "4": 1 + }, + "branchMap": { + "0": { + "type": "branch", + "line": 1, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "locations": [ + { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + } + ] + } + }, + "b": { + "0": [ + 1 + ] + }, + "fnMap": { + "0": { + "name": "covered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "line": 1 + }, + "1": { + "name": "uncovered", + "decl": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "loc": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "line": 3 + } + }, + "f": { + "0": 1, + "1": 0 + } +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-covered-2-istanbul.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-covered-2-istanbul.snapshot.json new file mode 100644 index 0000000000000..996a4a66c39ea --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-covered-2-istanbul.snapshot.json @@ -0,0 +1,83 @@ +{ + "path": "/fixtures/workspaces/custom-2/src/covered.custom-2", + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "1": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + } + }, + "fnMap": { + "0": { + "name": "covered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + } + }, + "1": { + "name": "uncovered", + "decl": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "loc": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + } + } + }, + "branchMap": {}, + "s": { + "0": 1, + "1": 0 + }, + "f": { + "0": 1, + "1": 0 + }, + "b": {} +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-covered-2-v8.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-covered-2-v8.snapshot.json new file mode 100644 index 0000000000000..e5380af0ec313 --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-covered-2-v8.snapshot.json @@ -0,0 +1,150 @@ +{ + "path": "/fixtures/workspaces/custom-2/src/covered.custom-2", + "all": false, + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "1": { + "start": { + "line": 2, + "column": 0 + }, + "end": { + "line": 2, + "column": 0 + } + }, + "2": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "3": { + "start": { + "line": 4, + "column": 0 + }, + "end": { + "line": 4, + "column": 0 + } + }, + "4": { + "start": { + "line": 5, + "column": 0 + }, + "end": { + "line": 5, + "column": 24 + } + } + }, + "s": { + "0": 1, + "1": 1, + "2": 0, + "3": 1, + "4": 1 + }, + "branchMap": { + "0": { + "type": "branch", + "line": 1, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "locations": [ + { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + } + ] + } + }, + "b": { + "0": [ + 1 + ] + }, + "fnMap": { + "0": { + "name": "covered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 18 + } + }, + "line": 1 + }, + "1": { + "name": "uncovered", + "decl": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "loc": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 20 + } + }, + "line": 3 + } + }, + "f": { + "0": 1, + "1": 0 + } +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-istanbul.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-istanbul.snapshot.json new file mode 100644 index 0000000000000..af9efc13611ed --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-istanbul.snapshot.json @@ -0,0 +1,48 @@ +{ + "path": "/fixtures/src/uncovered.custom-1", + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + } + }, + "fnMap": { + "0": { + "name": "uncovered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + } + } + }, + "branchMap": {}, + "s": { + "0": 0 + }, + "f": { + "0": 0 + }, + "b": {} +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-v8.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-v8.snapshot.json new file mode 100644 index 0000000000000..bcd396b7e8173 --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-uncovered-1-v8.snapshot.json @@ -0,0 +1,103 @@ +{ + "path": "/fixtures/src/uncovered.custom-1", + "all": true, + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + }, + "1": { + "start": { + "line": 2, + "column": 0 + }, + "end": { + "line": 2, + "column": 0 + } + }, + "2": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 26 + } + } + }, + "s": { + "0": 0, + "1": 0, + "2": 0 + }, + "branchMap": { + "0": { + "type": "branch", + "line": 1, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "locations": [ + { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + } + ] + } + }, + "b": { + "0": [ + 0 + ] + }, + "fnMap": { + "0": { + "name": "(empty-report)", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "line": 1 + } + }, + "f": { + "0": 0 + } +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-istanbul.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-istanbul.snapshot.json new file mode 100644 index 0000000000000..eb4119bb3faae --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-istanbul.snapshot.json @@ -0,0 +1,48 @@ +{ + "path": "/fixtures/workspaces/custom-2/src/uncovered.custom-2", + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + } + }, + "fnMap": { + "0": { + "name": "uncovered", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + } + } + }, + "branchMap": {}, + "s": { + "0": 0 + }, + "f": { + "0": 0 + }, + "b": {} +} \ No newline at end of file diff --git a/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-v8.snapshot.json b/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-v8.snapshot.json new file mode 100644 index 0000000000000..a26b2caac637c --- /dev/null +++ b/test/coverage-test/test/__snapshots__/custom-file-uncovered-2-v8.snapshot.json @@ -0,0 +1,103 @@ +{ + "path": "/fixtures/workspaces/custom-2/src/uncovered.custom-2", + "all": true, + "statementMap": { + "0": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 1, + "column": 20 + } + }, + "1": { + "start": { + "line": 2, + "column": 0 + }, + "end": { + "line": 2, + "column": 0 + } + }, + "2": { + "start": { + "line": 3, + "column": 0 + }, + "end": { + "line": 3, + "column": 26 + } + } + }, + "s": { + "0": 0, + "1": 0, + "2": 0 + }, + "branchMap": { + "0": { + "type": "branch", + "line": 1, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "locations": [ + { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + } + ] + } + }, + "b": { + "0": [ + 0 + ] + }, + "fnMap": { + "0": { + "name": "(empty-report)", + "decl": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "loc": { + "start": { + "line": 1, + "column": 0 + }, + "end": { + "line": 3, + "column": -2 + } + }, + "line": 1 + } + }, + "f": { + "0": 0 + } +} \ No newline at end of file diff --git a/test/coverage-test/test/all.test.ts b/test/coverage-test/test/all.test.ts index 063072e1f3dbf..97e464ed80af5 100644 --- a/test/coverage-test/test/all.test.ts +++ b/test/coverage-test/test/all.test.ts @@ -4,7 +4,7 @@ import { readCoverageMap, runVitest, test } from '../utils' test('{ all: true } includes uncovered files', async () => { await runVitest({ include: ['fixtures/test/**'], - exclude: ['**/virtual-files-**'], + exclude: ['**/virtual-files-**', '**/custom-1-syntax**'], coverage: { include: ['fixtures/src/**'], all: true, @@ -25,7 +25,7 @@ test('{ all: true } includes uncovered files', async () => { test('{ all: false } excludes uncovered files', async () => { await runVitest({ include: ['fixtures/test/**'], - exclude: ['**/virtual-files-**'], + exclude: ['**/virtual-files-**', '**/custom-1-syntax**'], coverage: { include: ['fixtures/src/**'], all: false, diff --git a/test/coverage-test/test/changed.test.ts b/test/coverage-test/test/changed.test.ts index 33ed43d6d139f..d80916764b11c 100644 --- a/test/coverage-test/test/changed.test.ts +++ b/test/coverage-test/test/changed.test.ts @@ -29,6 +29,7 @@ afterAll(() => { test('{ changed: "HEAD" }', async () => { await runVitest({ include: ['fixtures/test/**'], + exclude: ['**/custom-1-syntax**'], changed: 'HEAD', coverage: { include: ['fixtures/src/**'], diff --git a/test/coverage-test/test/workspace.multi-transform.test.ts b/test/coverage-test/test/workspace.multi-transform.test.ts new file mode 100644 index 0000000000000..254c1d6950340 --- /dev/null +++ b/test/coverage-test/test/workspace.multi-transform.test.ts @@ -0,0 +1,43 @@ +import { expect } from 'vitest' +import { isV8Provider, readCoverageMap, runVitest, test } from '../utils' + +test('{ all: true } includes uncovered files that require custom transform', async () => { + await runVitest({ + workspace: 'fixtures/configs/vitest.workspace.multi-transforms.ts', + coverage: { + all: true, + extension: ['.ts', '.custom-1', '.custom-2'], + reporter: 'json', + include: ['**/*.custom-1', '**/*.custom-2', '**/math.ts'], + }, + }) + + const coverageMap = await readCoverageMap() + const files = coverageMap.files() + + // All files from workspace should be picked + expect(files).toMatchInlineSnapshot(` + [ + "/fixtures/src/covered.custom-1", + "/fixtures/src/math.ts", + "/fixtures/src/uncovered.custom-1", + "/fixtures/workspaces/custom-2/src/covered.custom-2", + "/fixtures/workspaces/custom-2/src/uncovered.custom-2", + ] + `) + + const covered1 = coverageMap.fileCoverageFor('/fixtures/src/covered.custom-1') + const uncovered1 = coverageMap.fileCoverageFor('/fixtures/src/uncovered.custom-1') + const covered2 = coverageMap.fileCoverageFor('/fixtures/workspaces/custom-2/src/covered.custom-2') + const uncovered2 = coverageMap.fileCoverageFor('/fixtures/workspaces/custom-2/src/uncovered.custom-2') + + // Coverage maps indicate whether source maps are correct. Check html-report if these change + expect(JSON.stringify(covered1, null, 2)).toMatchFileSnapshot(snapshotName('covered-1')) + expect(JSON.stringify(uncovered1, null, 2)).toMatchFileSnapshot(snapshotName('uncovered-1')) + expect(JSON.stringify(covered2, null, 2)).toMatchFileSnapshot(snapshotName('covered-2')) + expect(JSON.stringify(uncovered2, null, 2)).toMatchFileSnapshot(snapshotName('uncovered-2')) +}) + +function snapshotName(label: string) { + return `__snapshots__/custom-file-${label}-${isV8Provider() ? 'v8' : 'istanbul'}.snapshot.json` +}