diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 044929ef5..1d5886739 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -173,8 +173,6 @@ jobs: cache: npm - name: Install dependencies run: npm ci - - name: Collect unit test coverage (temporary due to coverage plugin) - run: npx nx run-many -t unit-test --coverage - name: Collect Code PushUp report run: npx nx run-collect - name: Upload Code PushUp report to portal diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index bed1f33c7..aecc69130 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -90,8 +90,6 @@ jobs: run: | echo "//registry.npmjs.org/:_authToken=${NPM_TOKEN}" >> .npmrc npx nx run-many -t=deploy --exclude=plugin-lighthouse,nx-plugin - - name: Collect unit test coverage (temporary due to coverage plugin) - run: npx nx run-many -t unit-test --coverage - name: Collect and upload Code PushUp report run: npx nx run-autorun env: diff --git a/code-pushup.config.ts b/code-pushup.config.ts index d5c1ccdf5..8cd040db7 100644 --- a/code-pushup.config.ts +++ b/code-pushup.config.ts @@ -11,7 +11,9 @@ import { packageJsonPerformanceGroupRef, packageJsonPlugin, } from './dist/examples/plugins'; -import coveragePlugin from './dist/packages/plugin-coverage'; +import coveragePlugin, { + getNxCoveragePaths, +} from './dist/packages/plugin-coverage'; import eslintPlugin, { eslintConfigFromNxProjects, } from './dist/packages/plugin-eslint'; @@ -49,33 +51,20 @@ const config: CoreConfig = { plugins: [ await eslintPlugin(await eslintConfigFromNxProjects()), - coveragePlugin({ - reports: [ - { - resultsPath: 'coverage/cli/unit-tests/lcov.info', - pathToProject: 'packages/cli', - }, - { - resultsPath: 'coverage/core/unit-tests/lcov.info', - pathToProject: 'packages/core', - }, - { - resultsPath: 'coverage/models/unit-tests/lcov.info', - pathToProject: 'packages/models', - }, - { - resultsPath: 'coverage/utils/unit-tests/lcov.info', - pathToProject: 'packages/utils', - }, - { - resultsPath: 'coverage/plugin-eslint/unit-tests/lcov.info', - pathToProject: 'packages/plugin-eslint', - }, - { - resultsPath: 'coverage/plugin-coverage/unit-tests/lcov.info', - pathToProject: 'packages/plugin-coverage', - }, - ], + await coveragePlugin({ + coverageToolCommand: { + command: 'npx', + args: [ + 'nx', + 'run-many', + '-t', + 'unit-test', + 'integration-test', + '--coverage', + '--skipNxCache', + ], + }, + reports: await getNxCoveragePaths(['unit-test', 'integration-test']), }), fileSizePlugin({ directory: './dist/examples/react-todos-app', @@ -114,21 +103,9 @@ const config: CoreConfig = { title: 'Code coverage', refs: [ { - type: 'audit', - plugin: 'coverage', - slug: 'function-coverage', - weight: 1, - }, - { - type: 'audit', - plugin: 'coverage', - slug: 'branch-coverage', - weight: 1, - }, - { - type: 'audit', + type: 'group', plugin: 'coverage', - slug: 'line-coverage', + slug: 'coverage', weight: 1, }, ], diff --git a/e2e/cli-e2e/mocks/fixtures/code-pushup.config.coverage.ts b/e2e/cli-e2e/mocks/fixtures/code-pushup.config.coverage.ts index 99cf2f19c..1b211f2ba 100644 --- a/e2e/cli-e2e/mocks/fixtures/code-pushup.config.coverage.ts +++ b/e2e/cli-e2e/mocks/fixtures/code-pushup.config.coverage.ts @@ -15,29 +15,16 @@ export default { title: 'Code coverage', refs: [ { - type: 'audit', + type: 'group', plugin: 'coverage', - slug: 'function-coverage', - weight: 1, - }, - { - type: 'audit', - plugin: 'coverage', - slug: 'branch-coverage', - weight: 1, - }, - { - type: 'audit', - plugin: 'coverage', - slug: 'line-coverage', + slug: 'coverage', weight: 1, }, ], }, ], plugins: [ - coveragePlugin({ - coverageTypes: ['branch', 'function', 'line'], + await coveragePlugin({ reports: [ { resultsPath: join('e2e', 'cli-e2e', 'mocks', 'fixtures', 'lcov.info'), diff --git a/e2e/cli-e2e/tests/__snapshots__/collect.e2e.test.ts.snap b/e2e/cli-e2e/tests/__snapshots__/collect.e2e.test.ts.snap index 5a1c04e01..78376699e 100644 --- a/e2e/cli-e2e/tests/__snapshots__/collect.e2e.test.ts.snap +++ b/e2e/cli-e2e/tests/__snapshots__/collect.e2e.test.ts.snap @@ -7,20 +7,8 @@ exports[`CLI collect > should run Code coverage plugin and create report.json 1` "refs": [ { "plugin": "coverage", - "slug": "function-coverage", - "type": "audit", - "weight": 1, - }, - { - "plugin": "coverage", - "slug": "branch-coverage", - "type": "audit", - "weight": 1, - }, - { - "plugin": "coverage", - "slug": "line-coverage", - "type": "audit", + "slug": "coverage", + "type": "group", "weight": 1, }, ], @@ -33,36 +21,48 @@ exports[`CLI collect > should run Code coverage plugin and create report.json 1` { "audits": [ { - "description": "Branch coverage percentage on the project.", + "description": "Measures how many functions were called in at least one test.", "details": { "issues": [ { - "message": "2nd branch is not taken in any test case.", + "message": "Function formatReportScore is not called in any test case.", "severity": "error", "source": { "file": "packages/cli/src/lib/partly-covered/utils.ts", "position": { - "startLine": 6, + "startLine": 2, }, }, }, { - "message": "2nd branch is not taken in any test case.", + "message": "Function sortReport is not called in any test case.", "severity": "error", "source": { - "file": "packages/cli/src/lib/partly-covered/utils.ts", + "file": "packages/cli/src/lib/not-covered/sorting.ts", "position": { - "startLine": 10, + "startLine": 1, }, }, }, + ], + }, + "displayValue": "60 %", + "score": 0.6, + "slug": "function-coverage", + "title": "Function coverage", + "value": 60, + }, + { + "description": "Measures how many branches were executed after conditional statements in at least one test.", + "details": { + "issues": [ { - "message": "1st branch is not taken in any test case.", + "message": "2nd branch is not taken in any test case.", "severity": "error", "source": { - "file": "packages/cli/src/lib/not-covered/sorting.ts", + "file": "packages/cli/src/lib/partly-covered/utils.ts", "position": { - "startLine": 7, + "startLine": 6, }, }, }, @@ -70,54 +70,42 @@ exports[`CLI collect > should run Code coverage plugin and create report.json 1` "message": "2nd branch is not taken in any test case.", "severity": "error", "source": { - "file": "packages/cli/src/lib/not-covered/sorting.ts", + "file": "packages/cli/src/lib/partly-covered/utils.ts", "position": { - "startLine": 7, + "startLine": 10, }, }, }, - ], - }, - "displayValue": "76 %", - "score": 0.7647, - "slug": "branch-coverage", - "title": "Branch coverage", - "value": 76, - }, - { - "description": "Function coverage percentage on the project.", - "details": { - "issues": [ { - "message": "Function formatReportScore is not called in any test case.", + "message": "1st branch is not taken in any test case.", "severity": "error", "source": { - "file": "packages/cli/src/lib/partly-covered/utils.ts", + "file": "packages/cli/src/lib/not-covered/sorting.ts", "position": { - "startLine": 2, + "startLine": 7, }, }, }, { - "message": "Function sortReport is not called in any test case.", + "message": "2nd branch is not taken in any test case.", "severity": "error", "source": { "file": "packages/cli/src/lib/not-covered/sorting.ts", "position": { - "startLine": 1, + "startLine": 7, }, }, }, ], }, - "displayValue": "60 %", - "score": 0.6, - "slug": "function-coverage", - "title": "Function coverage", - "value": 60, + "displayValue": "76 %", + "score": 0.7647, + "slug": "branch-coverage", + "title": "Branch coverage", + "value": 76, }, { - "description": "Line coverage percentage on the project.", + "description": "Measures how many lines of code were executed in at least one test.", "details": { "issues": [ { @@ -153,6 +141,27 @@ exports[`CLI collect > should run Code coverage plugin and create report.json 1` ], "description": "Official Code PushUp code coverage plugin.", "docsUrl": "https://www.npmjs.com/package/@code-pushup/coverage-plugin/", + "groups": [ + { + "description": "Group containing all defined coverage types as audits.", + "refs": [ + { + "slug": "function-coverage", + "weight": 6, + }, + { + "slug": "branch-coverage", + "weight": 3, + }, + { + "slug": "line-coverage", + "weight": 1, + }, + ], + "slug": "coverage", + "title": "Code coverage metrics", + }, + ], "icon": "folder-coverage-open", "packageName": "@code-pushup/coverage-plugin", "slug": "coverage", diff --git a/examples/plugins/vite.config.integration.ts b/examples/plugins/vite.config.integration.ts index 8561409cd..c5cfba46b 100644 --- a/examples/plugins/vite.config.integration.ts +++ b/examples/plugins/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/examples/plugins/vite.config.unit.ts b/examples/plugins/vite.config.unit.ts index 1f8ed984b..699d83cc8 100644 --- a/examples/plugins/vite.config.unit.ts +++ b/examples/plugins/vite.config.unit.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.unit.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/cli/vite.config.integration.ts b/packages/cli/vite.config.integration.ts index 65ed19a91..feae2c448 100644 --- a/packages/cli/vite.config.integration.ts +++ b/packages/cli/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/core/vite.config.integration.ts b/packages/core/vite.config.integration.ts index bc7172cc4..94446d812 100644 --- a/packages/core/vite.config.integration.ts +++ b/packages/core/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/models/vite.config.integration.ts b/packages/models/vite.config.integration.ts index 65ed19a91..feae2c448 100644 --- a/packages/models/vite.config.integration.ts +++ b/packages/models/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/nx-plugin/vite.config.integration.ts b/packages/nx-plugin/vite.config.integration.ts index ff29ab4fa..ea9b02407 100644 --- a/packages/nx-plugin/vite.config.integration.ts +++ b/packages/nx-plugin/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], setupFiles: [ diff --git a/packages/nx-plugin/vite.config.unit.ts b/packages/nx-plugin/vite.config.unit.ts index 91f8d2cb0..2b8aba973 100644 --- a/packages/nx-plugin/vite.config.unit.ts +++ b/packages/nx-plugin/vite.config.unit.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.unit.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], setupFiles: [ diff --git a/packages/plugin-coverage/README.md b/packages/plugin-coverage/README.md index 372f27b83..aede2bb20 100644 --- a/packages/plugin-coverage/README.md +++ b/packages/plugin-coverage/README.md @@ -10,6 +10,9 @@ Measured coverage types are mapped to Code PushUp audits in the following way - the score is value converted to 0-1 range - missing coverage is mapped to issues in the audit details (uncalled functions, uncovered branches or lines) +> [!IMPORTANT] +> In order to successfully run your coverage tool and gather coverage results directly within the plugin, all your tests need to pass! + ## Getting started 1. If you haven't already, install [@code-pushup/cli](../cli/README.md) and create a configuration file. @@ -43,7 +46,7 @@ Measured coverage types are mapped to Code PushUp audits in the following way }; ``` -4. (Optional) Reference audits which you wish to include in custom categories (use `npx code-pushup print-config` to list audits and groups). +4. (Optional) Reference individual audits or the provided plugin group which you wish to include in custom categories (use `npx code-pushup print-config` to list audits and groups). 💡 Assign weights based on what influence each coverage type should have on the overall category score (assign weight 0 to only include as extra info, without influencing category score). @@ -56,21 +59,9 @@ Measured coverage types are mapped to Code PushUp audits in the following way title: 'Code coverage', refs: [ { - type: 'audit', - plugin: 'coverage', - slug: 'function-coverage', - weight: 2, - }, - { - type: 'audit', + type: 'group', plugin: 'coverage', - slug: 'branch-coverage', - weight: 1, - }, - { - type: 'audit', - plugin: 'coverage', - slug: 'line-coverage', + slug: 'coverage', weight: 1, }, // ... @@ -87,6 +78,12 @@ Measured coverage types are mapped to Code PushUp audits in the following way Code coverage is a metric that indicates what percentage of source code is executed by unit tests. It can give insights into test effectiveness and uncover parts of source code that would otherwise go untested. +- **Statement coverage**: Measures how many statements are executed in at least one test. +- **Line coverage**: Measures how many lines are executed in at least one test. Unlike statement coverage, any partially executed line counts towards line coverage. +- **Condition coverage**: Measures all condition values (`true`/`false`) evaluated for a conditional statement in at least one test. +- **Branch coverage**: Measures how many branches are executed as a result of conditional statements (`if`/`else` and other) in at least one test. In case of short-circuit logic, only executed paths are counted in. Unlike condition coverage, it does not ensure all combinations of condition values are tested. +- **Function coverage**: Measures how many functions are called in at least one test. Argument values, usage of optional arguments or default values is irrelevant for this metric. + > [!IMPORTANT] > Please note that code coverage is not the same as test coverage. Test coverage measures the amount of acceptance criteria covered by tests and is hard to formally verify. This means that code coverage cannot guarantee that the designed software caters to the business requirements. @@ -123,9 +120,62 @@ The plugin accepts the following parameters: - `coverageTypes`: An array of types of coverage that you wish to track. Supported values: `function`, `branch`, `line`. Defaults to all available types. - `reports`: Array of information about files with code coverage results - paths to results, path to project root the results belong to. LCOV format is supported for now. + - If you have an Nx monorepo, you can adjust our helper function `getNxCoveragePaths` to get the path information automatically. - (optional) `coverageToolCommand`: If you wish to run your coverage tool to generate the results first, you may define it here. - (optional) `perfectScoreThreshold`: If your coverage goal is not 100%, you may define it here in range 0-1. Any score above the defined threshold will be given the perfect score. The value will stay unaffected. +### Audits and group + +This plugin provides a group for convenient declaration in your config. When defined this way, all measured coverage type audits have the same weight. + +```ts + // ... + categories: [ + { + slug: 'code-coverage', + title: 'Code coverage', + refs: [ + { + type: 'group', + plugin: 'coverage', + slug: 'coverage', + weight: 1, + }, + // ... + ], + }, + // ... + ], +``` + +Each coverage type still has its own audit. So when you want to include a subset of coverage types or assign different weights to them, you can do so in the following way: + +```ts + // ... + categories: [ + { + slug: 'code-coverage', + title: 'Code coverage', + refs: [ + { + type: 'audit', + plugin: 'coverage', + slug: 'function-coverage', + weight: 2, + }, + { + type: 'audit', + plugin: 'coverage', + slug: 'branch-coverage', + weight: 1, + }, + // ... + ], + }, + // ... + ], +``` + ### Audit output An audit is an aggregation of all results for one coverage type passed to the plugin. @@ -157,3 +207,14 @@ For instance, the following can be an audit output for line coverage. } } ``` + +### Providing coverage results in Nx monorepo + +As a part of the plugin, there is a `getNxCoveragePaths` helper for setting up paths to coverage results if you are using Nx. The helper accepts all relevant targets (e.g. `test` or `unit-test`) and searches for a coverage path option. +Jest and Vitest configuration options are currently supported: + +- For `@nx/jest` executor it looks for the `coverageDirectory` option. +- For `@nx/vite` executor it looks for the `reportsDirectory` option. + +> [!IMPORTANT] +> Please note that you need to set up the coverage directory option in your `project.json` target options. Test configuration files are not searched. diff --git a/packages/plugin-coverage/package.json b/packages/plugin-coverage/package.json index 7aacac7c0..bc153606b 100644 --- a/packages/plugin-coverage/package.json +++ b/packages/plugin-coverage/package.json @@ -5,6 +5,15 @@ "@code-pushup/models": "*", "@code-pushup/utils": "*", "parse-lcov": "^1.0.4", + "chalk": "^5.3.0", "zod": "^3.22.4" + }, + "peerDependencies": { + "@nx/devkit": "^17.0.0" + }, + "peerDependenciesMeta": { + "@nx/devkit": { + "optional": true + } } } diff --git a/packages/plugin-coverage/project.json b/packages/plugin-coverage/project.json index 2b325a66f..ed08d37de 100644 --- a/packages/plugin-coverage/project.json +++ b/packages/plugin-coverage/project.json @@ -11,6 +11,7 @@ "outputPath": "dist/packages/plugin-coverage", "main": "packages/plugin-coverage/src/index.ts", "tsConfig": "packages/plugin-coverage/tsconfig.lib.json", + "additionalEntryPoints": ["packages/plugin-coverage/src/bin.ts"], "assets": ["packages/plugin-coverage/*.md"], "esbuildConfig": "esbuild.config.js" } diff --git a/packages/plugin-coverage/src/bin.ts b/packages/plugin-coverage/src/bin.ts new file mode 100644 index 000000000..9121d88b2 --- /dev/null +++ b/packages/plugin-coverage/src/bin.ts @@ -0,0 +1,3 @@ +import { executeRunner } from './lib/runner'; + +await executeRunner().catch(console.error); diff --git a/packages/plugin-coverage/src/index.ts b/packages/plugin-coverage/src/index.ts index 89fb4def3..fadde8e0d 100644 --- a/packages/plugin-coverage/src/index.ts +++ b/packages/plugin-coverage/src/index.ts @@ -2,3 +2,4 @@ import { coveragePlugin } from './lib/coverage-plugin'; export default coveragePlugin; export type { CoveragePluginConfig } from './lib/config'; +export { getNxCoveragePaths } from './lib/nx/coverage-paths'; diff --git a/packages/plugin-coverage/src/lib/config.ts b/packages/plugin-coverage/src/lib/config.ts index bd7a6f4bf..ef2c5d336 100644 --- a/packages/plugin-coverage/src/lib/config.ts +++ b/packages/plugin-coverage/src/lib/config.ts @@ -3,7 +3,7 @@ import { z } from 'zod'; export const coverageTypeSchema = z.enum(['function', 'branch', 'line']); export type CoverageType = z.infer; -export const coverageReportSchema = z.object({ +export const coverageResultSchema = z.object({ resultsPath: z.string().includes('lcov'), pathToProject: z .string({ @@ -12,7 +12,7 @@ export const coverageReportSchema = z.object({ }) .optional(), }); -export type CoverageReport = z.infer; +export type CoverageResult = z.infer; export const coveragePluginConfigSchema = z.object({ coverageToolCommand: z @@ -34,7 +34,7 @@ export const coveragePluginConfigSchema = z.object({ .min(1) .default(['function', 'branch', 'line']), reports: z - .array(coverageReportSchema, { + .array(coverageResultSchema, { description: 'Path to all code coverage report files. Only LCOV format is supported for now.', }) @@ -49,3 +49,6 @@ export const coveragePluginConfigSchema = z.object({ .optional(), }); export type CoveragePluginConfig = z.input; +export type FinalCoveragePluginConfig = z.infer< + typeof coveragePluginConfigSchema +>; diff --git a/packages/plugin-coverage/src/lib/coverage-plugin.integration.test.ts b/packages/plugin-coverage/src/lib/coverage-plugin.integration.test.ts deleted file mode 100644 index 78c36b436..000000000 --- a/packages/plugin-coverage/src/lib/coverage-plugin.integration.test.ts +++ /dev/null @@ -1,85 +0,0 @@ -import { join } from 'node:path'; -import { describe, expect, it } from 'vitest'; -import { CoveragePluginConfig } from './config'; -import { coveragePlugin } from './coverage-plugin'; - -describe('coveragePlugin', () => { - const LCOV_PATH = join( - 'packages', - 'plugin-coverage', - 'mocks', - 'single-record-lcov.info', - ); - - it('should initialise a Code coverage plugin', () => { - expect( - coveragePlugin({ - coverageTypes: ['function'], - reports: [{ resultsPath: LCOV_PATH }], - }), - ).toStrictEqual( - expect.objectContaining({ - slug: 'coverage', - title: 'Code coverage', - audits: expect.any(Array), - }), - ); - }); - - it('should generate audits from coverage types', () => { - expect( - coveragePlugin({ - coverageTypes: ['function', 'branch'], - reports: [{ resultsPath: LCOV_PATH }], - }), - ).toStrictEqual( - expect.objectContaining({ - audits: [ - { - slug: 'function-coverage', - title: 'Function coverage', - description: expect.stringContaining('Function coverage'), - }, - expect.objectContaining({ slug: 'branch-coverage' }), - ], - }), - ); - }); - - it('should assign RunnerConfig when a command is passed', () => { - expect( - coveragePlugin({ - coverageTypes: ['line'], - reports: [{ resultsPath: LCOV_PATH }], - coverageToolCommand: { - command: 'npm run-many', - args: ['-t', 'test', '--coverage'], - }, - } satisfies CoveragePluginConfig), - ).toStrictEqual( - expect.objectContaining({ - slug: 'coverage', - runner: { - command: 'npm run-many', - args: ['-t', 'test', '--coverage'], - outputFile: expect.stringContaining('runner-output.json'), - outputTransform: expect.any(Function), - }, - }), - ); - }); - - it('should assign a RunnerFunction when only reports are passed', () => { - expect( - coveragePlugin({ - coverageTypes: ['line'], - reports: [{ resultsPath: LCOV_PATH }], - } satisfies CoveragePluginConfig), - ).toStrictEqual( - expect.objectContaining({ - slug: 'coverage', - runner: expect.any(Function), - }), - ); - }); -}); diff --git a/packages/plugin-coverage/src/lib/coverage-plugin.ts b/packages/plugin-coverage/src/lib/coverage-plugin.ts index 989fe5687..d6bad0003 100644 --- a/packages/plugin-coverage/src/lib/coverage-plugin.ts +++ b/packages/plugin-coverage/src/lib/coverage-plugin.ts @@ -1,20 +1,15 @@ -import { join } from 'node:path'; -import type { - Audit, - PluginConfig, - RunnerConfig, - RunnerFunction, -} from '@code-pushup/models'; -import { capitalize, pluginWorkDir } from '@code-pushup/utils'; +import { dirname, join } from 'node:path'; +import { fileURLToPath } from 'node:url'; +import type { Audit, Group, PluginConfig } from '@code-pushup/models'; +import { capitalize } from '@code-pushup/utils'; import { name, version } from '../../package.json'; -import { CoveragePluginConfig, coveragePluginConfigSchema } from './config'; -import { lcovResultsToAuditOutputs } from './runner/lcov/runner'; -import { applyMaxScoreAboveThreshold } from './utils'; - -export const RUNNER_OUTPUT_PATH = join( - pluginWorkDir('coverage'), - 'runner-output.json', -); +import { + CoveragePluginConfig, + CoverageType, + coveragePluginConfigSchema, +} from './config'; +import { createRunnerConfig } from './runner'; +import { coverageDescription, coverageTypeWeightMapper } from './utils'; /** * Instantiates Code PushUp code coverage plugin for core config. @@ -34,37 +29,36 @@ export const RUNNER_OUTPUT_PATH = join( * * @returns Plugin configuration. */ -export function coveragePlugin(config: CoveragePluginConfig): PluginConfig { - const { reports, perfectScoreThreshold, coverageTypes, coverageToolCommand } = - coveragePluginConfigSchema.parse(config); +export async function coveragePlugin( + config: CoveragePluginConfig, +): Promise { + const coverageConfig = coveragePluginConfigSchema.parse(config); - const audits = coverageTypes.map( + const audits = coverageConfig.coverageTypes.map( (type): Audit => ({ slug: `${type}-coverage`, title: `${capitalize(type)} coverage`, - description: `${capitalize(type)} coverage percentage on the project.`, + description: coverageDescription[type], }), ); - const getAuditOutputs = async () => - perfectScoreThreshold - ? applyMaxScoreAboveThreshold( - await lcovResultsToAuditOutputs(reports, coverageTypes), - perfectScoreThreshold, - ) - : await lcovResultsToAuditOutputs(reports, coverageTypes); + const group: Group = { + slug: 'coverage', + title: 'Code coverage metrics', + description: 'Group containing all defined coverage types as audits.', + refs: audits.map(audit => ({ + ...audit, + weight: + coverageTypeWeightMapper[ + audit.slug.slice(0, audit.slug.indexOf('-')) as CoverageType + ], + })), + }; - // if coverage results are provided, only convert them to AuditOutputs - // if not, run coverage command and then run result conversion - const runner: RunnerConfig | RunnerFunction = - coverageToolCommand == null - ? getAuditOutputs - : { - command: coverageToolCommand.command, - args: coverageToolCommand.args, - outputFile: RUNNER_OUTPUT_PATH, - outputTransform: getAuditOutputs, - }; + const runnerScriptPath = join( + fileURLToPath(dirname(import.meta.url)), + 'bin.js', + ); return { slug: 'coverage', @@ -75,6 +69,7 @@ export function coveragePlugin(config: CoveragePluginConfig): PluginConfig { packageName: name, version, audits, - runner, + groups: [group], + runner: await createRunnerConfig(runnerScriptPath, coverageConfig), }; } diff --git a/packages/plugin-coverage/src/lib/coverage-plugin.unit.test.ts b/packages/plugin-coverage/src/lib/coverage-plugin.unit.test.ts new file mode 100644 index 000000000..a3879c198 --- /dev/null +++ b/packages/plugin-coverage/src/lib/coverage-plugin.unit.test.ts @@ -0,0 +1,84 @@ +import { join } from 'node:path'; +import { describe, expect, it } from 'vitest'; +import { RunnerConfig } from '@code-pushup/models'; +import { coveragePlugin } from './coverage-plugin'; + +vi.mock('./runner/index.ts', () => ({ + createRunnerConfig: vi.fn().mockReturnValue({ + command: 'node', + outputFile: 'runner-output.json', + } satisfies RunnerConfig), +})); + +describe('coveragePlugin', () => { + const LCOV_PATH = join( + 'packages', + 'plugin-coverage', + 'mocks', + 'single-record-lcov.info', + ); + + it('should initialise a Code coverage plugin', async () => { + await expect( + coveragePlugin({ + coverageTypes: ['function'], + reports: [{ resultsPath: LCOV_PATH }], + }), + ).resolves.toStrictEqual( + expect.objectContaining({ + slug: 'coverage', + title: 'Code coverage', + audits: expect.any(Array), + groups: expect.any(Array), + runner: expect.any(Object), + }), + ); + }); + + it('should generate audits from coverage types', async () => { + await expect( + coveragePlugin({ + coverageTypes: ['function', 'branch'], + reports: [{ resultsPath: LCOV_PATH }], + }), + ).resolves.toStrictEqual( + expect.objectContaining({ + audits: [ + { + slug: 'function-coverage', + title: 'Function coverage', + description: expect.stringContaining( + 'how many functions were called', + ), + }, + expect.objectContaining({ slug: 'branch-coverage' }), + ], + }), + ); + }); + + it('should provide a group from defined coverage types', async () => { + await expect( + coveragePlugin({ + coverageTypes: ['branch', 'line'], + reports: [{ resultsPath: LCOV_PATH }], + }), + ).resolves.toStrictEqual( + expect.objectContaining({ + audits: [ + expect.objectContaining({ slug: 'branch-coverage' }), + expect.objectContaining({ slug: 'line-coverage' }), + ], + groups: [ + expect.objectContaining({ + slug: 'coverage', + refs: [ + expect.objectContaining({ slug: 'branch-coverage' }), + expect.objectContaining({ slug: 'line-coverage' }), + ], + }), + ], + }), + ); + }); +}); diff --git a/packages/plugin-coverage/src/lib/nx/coverage-paths.ts b/packages/plugin-coverage/src/lib/nx/coverage-paths.ts new file mode 100644 index 000000000..171487eea --- /dev/null +++ b/packages/plugin-coverage/src/lib/nx/coverage-paths.ts @@ -0,0 +1,84 @@ +import type { ProjectGraphProjectNode, TargetConfiguration } from '@nx/devkit'; +import chalk from 'chalk'; +import { join } from 'node:path'; +import { CoverageResult } from '../config'; + +/** + * @param targets nx targets to be used for measuring coverage, test by default + * @returns An array of coverage result information for the coverage plugin. + */ +export async function getNxCoveragePaths( + targets: string[] = ['test'], +): Promise { + console.info( + chalk.bold('💡 Gathering coverage from the following nx projects:'), + ); + const { createProjectGraphAsync } = await import('@nx/devkit'); + const { nodes } = await createProjectGraphAsync({ exitOnError: false }); + + const coverageResults = targets.map(target => { + const relevantNodes = Object.values(nodes).filter(graph => + hasNxTarget(graph, target), + ); + + return relevantNodes.map(({ name, data }) => { + const targetConfig = data.targets?.[target] as TargetConfiguration; + const coveragePath = getCoveragePathForTarget(target, targetConfig, name); + const rootToReportsDir = join(data.root, coveragePath); + + console.info(`- ${name}: ${target}`); + + return { + pathToProject: data.root, + resultsPath: join(rootToReportsDir, 'lcov.info'), + }; + }); + }); + + console.info('\n'); + return coverageResults.flat(); +} + +function hasNxTarget( + project: ProjectGraphProjectNode, + target: string, +): boolean { + return project.data.targets != null && target in project.data.targets; +} + +function getCoveragePathForTarget( + target: string, + targetConfig: TargetConfiguration, + projectName: string, +): string { + if (targetConfig.executor?.includes('@nx/vite')) { + const { reportsDirectory } = targetConfig.options as { + reportsDirectory?: string; + }; + + if (reportsDirectory == null) { + throw new Error( + `Coverage configuration not found for target ${target} in ${projectName}. Define your Vitest coverage directory in the reportsDirectory option.`, + ); + } + + return reportsDirectory; + } + + if (targetConfig.executor?.includes('@nx/jest')) { + const { coverageDirectory } = targetConfig.options as { + coverageDirectory?: string; + }; + + if (coverageDirectory == null) { + throw new Error( + `Coverage configuration not found for target ${target} in ${projectName}. Define your Jest coverage directory in the coverageDirectory option.`, + ); + } + return coverageDirectory; + } + + throw new Error( + `Unsupported executor ${targetConfig.executor}. @nx/vite and @nx/jest are currently supported.`, + ); +} diff --git a/packages/plugin-coverage/src/lib/runner/constants.ts b/packages/plugin-coverage/src/lib/runner/constants.ts new file mode 100644 index 000000000..45490e44a --- /dev/null +++ b/packages/plugin-coverage/src/lib/runner/constants.ts @@ -0,0 +1,10 @@ +import { join } from 'node:path'; +import { pluginWorkDir } from '@code-pushup/utils'; + +export const WORKDIR = pluginWorkDir('coverage'); +export const RUNNER_OUTPUT_PATH = join(WORKDIR, 'runner-output.json'); +export const PLUGIN_CONFIG_PATH = join( + process.cwd(), + WORKDIR, + 'plugin-config.json', +); diff --git a/packages/plugin-coverage/src/lib/runner/index.ts b/packages/plugin-coverage/src/lib/runner/index.ts new file mode 100644 index 000000000..8ae74517c --- /dev/null +++ b/packages/plugin-coverage/src/lib/runner/index.ts @@ -0,0 +1,66 @@ +import chalk from 'chalk'; +import { writeFile } from 'node:fs/promises'; +import { dirname } from 'node:path'; +import type { AuditOutputs, RunnerConfig } from '@code-pushup/models'; +import { + ProcessError, + ensureDirectoryExists, + executeProcess, + readJsonFile, +} from '@code-pushup/utils'; +import { FinalCoveragePluginConfig } from '../config'; +import { applyMaxScoreAboveThreshold } from '../utils'; +import { PLUGIN_CONFIG_PATH, RUNNER_OUTPUT_PATH, WORKDIR } from './constants'; +import { lcovResultsToAuditOutputs } from './lcov/lcov-runner'; + +export async function executeRunner(): Promise { + const { reports, coverageToolCommand, coverageTypes } = + await readJsonFile(PLUGIN_CONFIG_PATH); + + // Run coverage tool if provided + if (coverageToolCommand != null) { + const { command, args } = coverageToolCommand; + + try { + await executeProcess({ command, args }); + } catch (error) { + if (error instanceof ProcessError) { + console.info(chalk.bold('stdout from failed process:')); + console.info(error.stdout); + console.error(chalk.bold('stderr from failed process:')); + console.error(error.stderr); + } + + throw new Error( + 'Coverage plugin: Running coverage tool failed. Make sure all your provided tests are passing.', + ); + } + } + + // Caculate coverage from LCOV results + const auditOutputs = await lcovResultsToAuditOutputs(reports, coverageTypes); + + await ensureDirectoryExists(dirname(RUNNER_OUTPUT_PATH)); + await writeFile(RUNNER_OUTPUT_PATH, JSON.stringify(auditOutputs)); +} + +export async function createRunnerConfig( + scriptPath: string, + config: FinalCoveragePluginConfig, +): Promise { + // Create JSON config for executeRunner + await ensureDirectoryExists(WORKDIR); + await writeFile(PLUGIN_CONFIG_PATH, JSON.stringify(config)); + + const threshold = config.perfectScoreThreshold; + + return { + command: 'node', + args: [scriptPath], + outputFile: RUNNER_OUTPUT_PATH, + ...(threshold != null && { + outputTransform: outputs => + applyMaxScoreAboveThreshold(outputs as AuditOutputs, threshold), + }), + }; +} diff --git a/packages/plugin-coverage/src/lib/runner/lcov/__snapshots__/runner.integration.test.ts.snap b/packages/plugin-coverage/src/lib/runner/lcov/__snapshots__/lcov-runner.integration.test.ts.snap similarity index 100% rename from packages/plugin-coverage/src/lib/runner/lcov/__snapshots__/runner.integration.test.ts.snap rename to packages/plugin-coverage/src/lib/runner/lcov/__snapshots__/lcov-runner.integration.test.ts.snap diff --git a/packages/plugin-coverage/src/lib/runner/lcov/runner.integration.test.ts b/packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.integration.test.ts similarity index 94% rename from packages/plugin-coverage/src/lib/runner/lcov/runner.integration.test.ts rename to packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.integration.test.ts index 626867bfa..58bea1618 100644 --- a/packages/plugin-coverage/src/lib/runner/lcov/runner.integration.test.ts +++ b/packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.integration.test.ts @@ -1,7 +1,7 @@ import { dirname, join } from 'node:path'; import { fileURLToPath } from 'node:url'; import { describe, it } from 'vitest'; -import { lcovResultsToAuditOutputs } from './runner'; +import { lcovResultsToAuditOutputs } from './lcov-runner'; describe('lcovResultsToAuditOutputs', () => { it('should correctly convert lcov results to AuditOutputs', async () => { diff --git a/packages/plugin-coverage/src/lib/runner/lcov/runner.ts b/packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.ts similarity index 69% rename from packages/plugin-coverage/src/lib/runner/lcov/runner.ts rename to packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.ts index 1888c265b..7f4023ac8 100644 --- a/packages/plugin-coverage/src/lib/runner/lcov/runner.ts +++ b/packages/plugin-coverage/src/lib/runner/lcov/lcov-runner.ts @@ -2,7 +2,7 @@ import { join } from 'node:path'; import type { LCOVRecord } from 'parse-lcov'; import { AuditOutputs } from '@code-pushup/models'; import { exists, readTextFile, toUnixNewlines } from '@code-pushup/utils'; -import { CoverageReport, CoverageType } from '../../config'; +import { CoverageResult, CoverageType } from '../../config'; import { parseLcov } from './parse-lcov'; import { lcovCoverageToAuditOutput, @@ -15,40 +15,20 @@ import { LCOVStat, LCOVStats } from './types'; /** * - * @param reports report files + * @param results Paths to LCOV results * @param coverageTypes types of coverage to be considered * @returns Audit outputs with complete coverage data. */ export async function lcovResultsToAuditOutputs( - reports: CoverageReport[], + results: CoverageResult[], coverageTypes: CoverageType[], ): Promise { - const parsedReports = await Promise.all( - reports.map(async report => { - const reportContent = await readTextFile(report.resultsPath); - const parsedRecords = parseLcov(toUnixNewlines(reportContent)); - return parsedRecords.map(record => ({ - ...record, - file: - report.pathToProject == null - ? record.file - : join(report.pathToProject, record.file), - })); - }), - ); - if (parsedReports.length !== reports.length) { - throw new Error('Some provided LCOV reports were not valid.'); - } - - const flatReports = parsedReports.flat(); - - if (flatReports.length === 0) { - throw new Error('All provided reports are empty.'); - } + // Parse lcov files + const lcovResults = await parseLcovFiles(results); - // Accumulate code coverage from all coverage result files - const totalCoverageStats = getTotalCoverageFromLcovReports( - flatReports, + // Calculate code coverage from all coverage results + const totalCoverageStats = getTotalCoverageFromLcovRecords( + lcovResults, coverageTypes, ); @@ -63,13 +43,47 @@ export async function lcovResultsToAuditOutputs( .filter(exists); } +/** + * + * @param results Paths to LCOV results + * @returns Array of parsed LCOVRecords. + */ +async function parseLcovFiles( + results: CoverageResult[], +): Promise { + const parsedResults = await Promise.all( + results.map(async result => { + const lcovFileContent = await readTextFile(result.resultsPath); + const parsedRecords = parseLcov(toUnixNewlines(lcovFileContent)); + return parsedRecords.map(record => ({ + ...record, + file: + result.pathToProject == null + ? record.file + : join(result.pathToProject, record.file), + })); + }), + ); + if (parsedResults.length !== results.length) { + throw new Error('Some provided LCOV results were not valid.'); + } + + const flatResults = parsedResults.flat(); + + if (flatResults.length === 0) { + throw new Error('All provided results are empty.'); + } + + return flatResults; +} + /** * * @param records This function aggregates coverage stats from all coverage files * @param coverageTypes Types of coverage to be gathered * @returns Complete coverage stats for all defined types of coverage. */ -function getTotalCoverageFromLcovReports( +function getTotalCoverageFromLcovRecords( records: LCOVRecord[], coverageTypes: CoverageType[], ): LCOVStats { diff --git a/packages/plugin-coverage/src/lib/runner/runner.integration.test.ts b/packages/plugin-coverage/src/lib/runner/runner.integration.test.ts new file mode 100644 index 000000000..aa02ed30e --- /dev/null +++ b/packages/plugin-coverage/src/lib/runner/runner.integration.test.ts @@ -0,0 +1,82 @@ +import { writeFile } from 'node:fs/promises'; +import { dirname, join } from 'node:path'; +import { fileURLToPath } from 'node:url'; +import { describe, it } from 'vitest'; +import { AuditOutput, AuditOutputs, RunnerConfig } from '@code-pushup/models'; +import { readJsonFile, removeDirectoryIfExists } from '@code-pushup/utils'; +import { createRunnerConfig, executeRunner } from '.'; +import { FinalCoveragePluginConfig } from '../config'; +import { PLUGIN_CONFIG_PATH, RUNNER_OUTPUT_PATH, WORKDIR } from './constants'; + +describe('createRunnerConfig', () => { + it('should create a valid runner config', async () => { + const runnerConfig = await createRunnerConfig('executeRunner.ts', { + reports: [{ resultsPath: 'coverage/lcov.info' }], + coverageTypes: ['branch'], + perfectScoreThreshold: 85, + }); + expect(runnerConfig).toStrictEqual({ + command: 'node', + args: ['executeRunner.ts'], + outputTransform: expect.any(Function), + outputFile: expect.stringContaining('runner-output.json'), + } satisfies RunnerConfig); + }); + + it('should provide plugin config to runner in JSON file', async () => { + await removeDirectoryIfExists(WORKDIR); + + const pluginConfig: FinalCoveragePluginConfig = { + coverageTypes: ['line'], + reports: [{ resultsPath: 'coverage/lcov.info' }], + coverageToolCommand: { command: 'npm', args: ['run', 'test'] }, + perfectScoreThreshold: 85, + }; + + await createRunnerConfig('executeRunner.ts', pluginConfig); + + const config = await readJsonFile( + PLUGIN_CONFIG_PATH, + ); + expect(config).toStrictEqual(pluginConfig); + }); +}); + +describe('executeRunner', () => { + it('should successfully execute runner', async () => { + const config: FinalCoveragePluginConfig = { + reports: [ + { + resultsPath: join( + fileURLToPath(dirname(import.meta.url)), + '..', + '..', + '..', + 'mocks', + 'single-record-lcov.info', + ), + }, + ], + coverageTypes: ['line'], + }; + + await writeFile(PLUGIN_CONFIG_PATH, JSON.stringify(config)); + await executeRunner(); + + const results = await readJsonFile(RUNNER_OUTPUT_PATH); + expect(results).toStrictEqual([ + expect.objectContaining({ + slug: 'line-coverage', + score: 0.7, + value: 70, + details: { + issues: [ + expect.objectContaining({ + message: 'Lines 7-9 are not covered in any test case.', + }), + ], + }, + } satisfies AuditOutput), + ]); + }); +}); diff --git a/packages/plugin-coverage/src/lib/utils.ts b/packages/plugin-coverage/src/lib/utils.ts index b9df7c1c0..194b39532 100644 --- a/packages/plugin-coverage/src/lib/utils.ts +++ b/packages/plugin-coverage/src/lib/utils.ts @@ -1,4 +1,12 @@ import type { AuditOutputs } from '@code-pushup/models'; +import { CoverageType } from './config'; + +export const coverageDescription: Record = { + branch: + 'Measures how many branches were executed after conditional statements in at least one test.', + line: 'Measures how many lines of code were executed in at least one test.', + function: 'Measures how many functions were called in at least one test.', +}; /** * Since more code coverage does not necessarily mean better score, this optional override allows for defining custom coverage goals. @@ -14,3 +22,11 @@ export function applyMaxScoreAboveThreshold( output.score >= threshold ? { ...output, score: 1 } : output, ); } + +/* eslint-disable no-magic-numbers */ +export const coverageTypeWeightMapper: Record = { + function: 6, + branch: 3, + line: 1, +}; +/* eslint-enable no-magic-numbers */ diff --git a/packages/plugin-coverage/vite.config.integration.ts b/packages/plugin-coverage/vite.config.integration.ts index 92dda7e4a..f04fcf1a1 100644 --- a/packages/plugin-coverage/vite.config.integration.ts +++ b/packages/plugin-coverage/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/plugin-eslint/vite.config.integration.ts b/packages/plugin-eslint/vite.config.integration.ts index 06de4549f..63e112d65 100644 --- a/packages/plugin-eslint/vite.config.integration.ts +++ b/packages/plugin-eslint/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/plugin-lighthouse/vite.config.integration.ts b/packages/plugin-lighthouse/vite.config.integration.ts index 394bf5446..1a543d7c0 100644 --- a/packages/plugin-lighthouse/vite.config.integration.ts +++ b/packages/plugin-lighthouse/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/packages/utils/src/index.ts b/packages/utils/src/index.ts index 932cde2d4..cec3a39b8 100644 --- a/packages/utils/src/index.ts +++ b/packages/utils/src/index.ts @@ -20,6 +20,7 @@ export { pluginWorkDir, readJsonFile, readTextFile, + removeDirectoryIfExists, } from './lib/file-system'; export { formatBytes, diff --git a/packages/utils/src/lib/file-system.ts b/packages/utils/src/lib/file-system.ts index 3df771856..256768d49 100644 --- a/packages/utils/src/lib/file-system.ts +++ b/packages/utils/src/lib/file-system.ts @@ -1,6 +1,6 @@ import { type Options, bundleRequire } from 'bundle-require'; import chalk from 'chalk'; -import { mkdir, readFile, readdir, stat } from 'node:fs/promises'; +import { mkdir, readFile, readdir, rm, stat } from 'node:fs/promises'; import { join } from 'node:path'; import { formatBytes } from './formatting'; import { logMultipleResults } from './log-results'; @@ -45,6 +45,12 @@ export async function ensureDirectoryExists(baseDir: string) { } } +export async function removeDirectoryIfExists(dir: string) { + if (await directoryExists(dir)) { + await rm(dir, { recursive: true, force: true }); + } +} + export type FileResult = readonly [string] | readonly [string, number]; export type MultipleFileResults = PromiseSettledResult[]; diff --git a/packages/utils/vite.config.integration.ts b/packages/utils/vite.config.integration.ts index 65ed19a91..feae2c448 100644 --- a/packages/utils/vite.config.integration.ts +++ b/packages/utils/vite.config.integration.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.integration.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'], diff --git a/project.json b/project.json index db378e489..fe487ca3f 100644 --- a/project.json +++ b/project.json @@ -50,6 +50,7 @@ "projects": [ "cli", "plugin-eslint", + "plugin-coverage", "examples-plugins", "react-todos-app" ], @@ -64,6 +65,7 @@ "projects": [ "cli", "plugin-eslint", + "plugin-coverage", "examples-plugins", "react-todos-app" ], diff --git a/testing-utils/vite.config.unit.ts b/testing-utils/vite.config.unit.ts index 6bb75b280..ea7a64394 100644 --- a/testing-utils/vite.config.unit.ts +++ b/testing-utils/vite.config.unit.ts @@ -10,6 +10,9 @@ export default defineConfig({ cache: { dir: '../node_modules/.vitest', }, + coverage: { + reporter: ['lcov'], + }, environment: 'node', include: ['src/**/*.unit.test.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], globalSetup: ['global-setup.ts'],