diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 7488baba0..ff3f3e093 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -12,6 +12,31 @@ on: - 'LICENSE' jobs: + coverage: + name: 'Coverage' + runs-on: ubuntu-latest + needs: + # bun excluded because it uses built-in test tool which does not support coverage in v8/istanbul format. + # refer https://github.com/oven-sh/bun/issues/4015 + - main + - fastly + - node + - wrangler + - lambda + - lambda-edge + steps: + - uses: actions/checkout@v4 + - uses: actions/download-artifact@v4 + with: + pattern: coverage-* + merge-multiple: true + path: ./coverage + - uses: codecov/codecov-action@v4 + with: + fail_ci_if_error: true + directory: ./coverage + token: ${{ secrets.CODECOV_TOKEN }} + main: name: 'Main' runs-on: ubuntu-latest @@ -28,6 +53,10 @@ jobs: - run: bun run lint - run: bun run build - run: bun run test + - uses: actions/upload-artifact@v4 + with: + name: coverage-main + path: coverage/ jsr-dry-run: name: "Checking if it's valid for JSR" @@ -47,9 +76,13 @@ jobs: - uses: denoland/setup-deno@v1 with: deno-version: v1.x - - run: env NAME=Deno deno test --allow-read --allow-env --allow-write -c runtime_tests/deno/deno.json runtime_tests/deno - - run: deno test -c runtime_tests/deno-jsx/deno.precompile.json runtime_tests/deno-jsx - - run: deno test -c runtime_tests/deno-jsx/deno.react-jsx.json runtime_tests/deno-jsx + - run: env NAME=Deno deno test --coverage=coverage/raw/deno-runtime --allow-read --allow-env --allow-write -c runtime_tests/deno/deno.json runtime_tests/deno + - run: deno test -c runtime_tests/deno-jsx/deno.precompile.json --coverage=coverage/raw/deno-precompile-jsx runtime_tests/deno-jsx + - run: deno test -c runtime_tests/deno-jsx/deno.react-jsx.json --coverage=coverage/raw/deno-react-jsx runtime_tests/deno-jsx + - uses: actions/upload-artifact@v4 + with: + name: coverage-deno + path: coverage/ bun: name: 'Bun' @@ -70,6 +103,10 @@ jobs: - run: bun install - run: bun run build - run: bun run test:fastly + - uses: actions/upload-artifact@v4 + with: + name: coverage-fastly + path: coverage/ node: name: 'Node.js v${{ matrix.node }}' @@ -86,6 +123,11 @@ jobs: - run: bun install - run: bun run build - run: bun run test:node + - uses: actions/upload-artifact@v4 + if: matrix.node == '22.x' + with: + name: coverage-node + path: coverage/ wrangler: name: 'Cloudflare Workers' @@ -96,6 +138,10 @@ jobs: - run: bun install - run: bun run build - run: bun run test:wrangler + - uses: actions/upload-artifact@v4 + with: + name: coverage-wrangler + path: coverage/ lambda: name: 'AWS Lambda' @@ -106,6 +152,10 @@ jobs: - run: bun install - run: bun run build - run: bun run test:lambda + - uses: actions/upload-artifact@v4 + with: + name: coverage-lambda + path: coverage/ lambda-edge: name: 'Lambda@Edge' @@ -116,3 +166,7 @@ jobs: - run: bun install --frozen-lockfile - run: bun run build - run: bun run test:lambda-edge + - uses: actions/upload-artifact@v4 + with: + name: coverage-lambda-edge + path: coverage/ diff --git a/.vitest.config/jsx-runtime-default.ts b/.vitest.config/jsx-runtime-default.ts index 9885a19ba..a325bc047 100644 --- a/.vitest.config/jsx-runtime-default.ts +++ b/.vitest.config/jsx-runtime-default.ts @@ -8,5 +8,8 @@ if (config.test) { '**/src/jsx/dom/**/(*.)+(spec|test).+(ts|tsx|js)', 'src/jsx/hooks/dom.test.tsx', ] + if (config.test.coverage) { + config.test.coverage.reportsDirectory = './coverage/raw/jsx-runtime' + } } export default config diff --git a/.vitest.config/jsx-runtime-dom.ts b/.vitest.config/jsx-runtime-dom.ts index 11adf6ccf..c9a4a3a71 100644 --- a/.vitest.config/jsx-runtime-dom.ts +++ b/.vitest.config/jsx-runtime-dom.ts @@ -8,5 +8,8 @@ if (config.test) { '**/src/jsx/dom/**/(*.)+(spec|test).+(ts|tsx|js)', 'src/jsx/hooks/dom.test.tsx', ] + if (config.test.coverage) { + config.test.coverage.reportsDirectory = './coverage/raw/jsx-dom' + } } export default config diff --git a/codecov.yml b/codecov.yml new file mode 100644 index 000000000..b4889fe95 --- /dev/null +++ b/codecov.yml @@ -0,0 +1,13 @@ +# Edit "test.coverage.exclude" option in vitest.config.ts to exclude specific files from coverage reports. +# We can also use "ignore" option in codecov.yml, but it is not recognized by vitest, so results may differ on local. + +coverage: + status: + patch: + default: + target: 80% + informational: true # Don't fail the build even if coverage is below target + project: + default: + target: 75% + threshold: 1% diff --git a/runtime_tests/bun/vitest.config.ts b/runtime_tests/bun/vitest.config.ts index 5d244e3f1..6b1cae417 100644 --- a/runtime_tests/bun/vitest.config.ts +++ b/runtime_tests/bun/vitest.config.ts @@ -1,9 +1,14 @@ /// import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ test: { globals: true, include: ['**/runtime_tests/bun/**/*.+(ts|tsx|js)'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-bun', + }, }, }) diff --git a/runtime_tests/fastly/vitest.config.ts b/runtime_tests/fastly/vitest.config.ts index 89baa3577..c0480b139 100644 --- a/runtime_tests/fastly/vitest.config.ts +++ b/runtime_tests/fastly/vitest.config.ts @@ -1,6 +1,7 @@ /// import fastlyCompute from 'vite-plugin-fastly-js-compute' import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ plugins: [fastlyCompute()], @@ -8,5 +9,9 @@ export default defineConfig({ globals: true, include: ['**/runtime_tests/fastly/**/(*.)+(test).+(ts|tsx)'], exclude: ['**/runtime_tests/fastly/vitest.config.ts'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-fastly', + }, }, }) diff --git a/runtime_tests/lambda-edge/vitest.config.ts b/runtime_tests/lambda-edge/vitest.config.ts index 568ecc05c..105ac9505 100644 --- a/runtime_tests/lambda-edge/vitest.config.ts +++ b/runtime_tests/lambda-edge/vitest.config.ts @@ -1,5 +1,6 @@ /// import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ test: { @@ -9,5 +10,9 @@ export default defineConfig({ globals: true, include: ['**/runtime_tests/lambda-edge/**/*.+(ts|tsx|js)'], exclude: ['**/runtime_tests/lambda-edge/vitest.config.ts'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-lambda-edge', + }, }, }) diff --git a/runtime_tests/lambda/vitest.config.ts b/runtime_tests/lambda/vitest.config.ts index 0fd446dc6..f030cf410 100644 --- a/runtime_tests/lambda/vitest.config.ts +++ b/runtime_tests/lambda/vitest.config.ts @@ -1,5 +1,6 @@ /// import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ test: { @@ -9,5 +10,9 @@ export default defineConfig({ globals: true, include: ['**/runtime_tests/lambda/**/*.+(ts|tsx|js)'], exclude: ['**/runtime_tests/lambda/vitest.config.ts', '**/runtime_tests/lambda/mock.ts'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-lambda', + }, }, }) diff --git a/runtime_tests/node/vitest.config.ts b/runtime_tests/node/vitest.config.ts index a2c74bbbe..5387c11fd 100644 --- a/runtime_tests/node/vitest.config.ts +++ b/runtime_tests/node/vitest.config.ts @@ -1,5 +1,6 @@ /// import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ test: { @@ -9,5 +10,9 @@ export default defineConfig({ globals: true, include: ['**/runtime_tests/node/**/*.+(ts|tsx|js)'], exclude: ['**/runtime_tests/node/vitest.config.ts'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-node', + }, }, }) diff --git a/runtime_tests/wrangler/vitest.config.ts b/runtime_tests/wrangler/vitest.config.ts index 4b8af9b11..7eb5c620a 100644 --- a/runtime_tests/wrangler/vitest.config.ts +++ b/runtime_tests/wrangler/vitest.config.ts @@ -1,10 +1,15 @@ /// import { defineConfig } from 'vitest/config' +import config from '../../vitest.config' export default defineConfig({ test: { globals: true, include: ['**/runtime_tests/wrangler/**/(*.)+(test).+(ts|tsx)'], exclude: ['**/runtime_tests/wrangler/vitest.config.ts'], + coverage: { + ...config.test?.coverage, + reportsDirectory: './coverage/raw/runtime-wrangler', + }, }, }) diff --git a/vitest.config.ts b/vitest.config.ts index 90470c8b3..575f1e112 100644 --- a/vitest.config.ts +++ b/vitest.config.ts @@ -12,8 +12,18 @@ export default defineConfig({ exclude: [...configDefaults.exclude, '**/sandbox/**', '**/*.case.test.+(ts|tsx|js)'], setupFiles: ['./src/test-utils/setup-vitest.ts'], coverage: { + enabled: true, provider: 'v8', - reporter: ['text'], + reportsDirectory: './coverage/raw/default', + reporter: ['json'], + exclude: [ + ...(configDefaults.coverage.exclude ?? []), + 'benchmarks', + 'runtime_tests', + 'build.ts', + 'src/test-utils', + 'src/**/types.ts', // types are compile-time only, so their coverage cannot be measured + ] }, pool: 'forks', },