import { readFileSync } from 'node:fs' import { resolve } from 'node:path' import { fileURLToPath } from 'node:url' import { normalize } from 'pathe' import libCoverage from 'istanbul-lib-coverage' import type { FileCoverageData } from 'istanbul-lib-coverage' import type { TestFunction, UserConfig } from 'vitest' import { describe as vitestDescribe, test as vitestTest } from 'vitest' import * as testUtils from '../test-utils' export function test(name: string, fn: TestFunction, skip = false) { if (process.env.COVERAGE_TEST !== 'true') { return vitestTest.skipIf(skip)(name, fn) } } export function describe(name: string, fn: () => void) { if (process.env.COVERAGE_TEST !== 'true') { return vitestDescribe(name, () => fn()) } } export function coverageTest(name: string, fn: TestFunction) { if (process.env.COVERAGE_TEST === 'true') { return vitestTest(name, fn) } } export async function runVitest(config: UserConfig, options = { throwOnError: true }) { const provider = process.env.COVERAGE_PROVIDER as any const result = await testUtils.runVitest({ config: 'fixtures/configs/vitest.config.ts', pool: 'threads', ...config, env: { COVERAGE_TEST: 'true', ...config.env, }, coverage: { enabled: true, reporter: [], ...config.coverage, provider, customProviderModule: provider === 'custom' ? 'fixtures/custom-provider' : undefined, }, browser: { enabled: process.env.COVERAGE_BROWSER === 'true', headless: true, name: 'chromium', provider: 'playwright', }, }) if (options.throwOnError) { if (result.stderr !== '') { throw new Error(result.stderr) } } return result } /** * Read JSON coverage report from file system. * Normalizes paths to keep contents consistent between OS's */ export async function readCoverageJson(name = './coverage/coverage-final.json') { const jsonReport = JSON.parse(readFileSync(name, 'utf8')) as Record const normalizedReport: typeof jsonReport = {} for (const [filename, coverage] of Object.entries(jsonReport)) { coverage.path = normalizeFilename(coverage.path) normalizedReport[normalizeFilename(filename)] = coverage } return normalizedReport } /** * Read coverage report from file system as Istanbul's `CoverageMap` */ export async function readCoverageMap(name = './coverage/coverage-final.json') { const coverageJson = await readCoverageJson(name) return libCoverage.createCoverageMap(coverageJson) } export function normalizeFilename(filename: string) { return normalize(filename) .replace(normalize(process.cwd()), '') .replace(normalize(resolve(process.cwd(), '../../')), '') } export function isV8Provider() { return process.env.COVERAGE_PROVIDER === 'v8' } export function normalizeURL(importMetaURL: string) { return normalize(fileURLToPath(importMetaURL)) }