From qa-mutation-testing
Configures StrykerJS for mutation testing of JavaScript / TypeScript / React / Vue / Svelte / Node - picks the test-runner plugin (`@stryker-mutator/jest-runner`, `mocha-runner`, `vitest-runner`, `karma-runner`), authors `stryker.conf.json` with mutate globs + thresholds, runs incremental mode for PRs (only mutate changed files), and reports the mutation score. Use when a JS/TS test suite has ≥80% line coverage and the team wants to verify the tests actually catch bugs (not just touch lines).
How this skill is triggered — by the user, by Claude, or both
Slash command
/qa-mutation-testing:stryker-mutationThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Per [stryker-intro][si]:
Per stryker-intro:
"Stryker started as a pure JavaScript mutation testing framework" and now "supports most JavaScript projects, including TypeScript, React, Angular, VueJS, Svelte, and NodeJS."
Mutation testing introduces small bugs (mutants) into the production code; if tests still pass, the test suite is too weak. A green suite with high mutation-survival rate is the bug coverage hides.
.toBeTruthy() etc); mutation testing makes the cost visible.npm install --save-dev \
@stryker-mutator/core \
@stryker-mutator/jest-runner # or mocha-runner / vitest-runner / etc.
Per stryker-intro, supported runners include Jest, Mocha, Karma, Vitest, Jasmine, Cucumber, Tap.
npx stryker init
Generates stryker.conf.json:
{
"$schema": "./node_modules/@stryker-mutator/core/schema/stryker-schema.json",
"packageManager": "npm",
"reporters": ["progress", "clear-text", "html"],
"testRunner": "jest",
"coverageAnalysis": "perTest",
"mutate": ["src/**/*.ts", "!src/**/*.test.ts"],
"thresholds": { "high": 80, "low": 60, "break": 50 }
}
Key fields:
| Field | Use |
|---|---|
mutate | Glob patterns: which files to mutate; exclude tests + types. |
testRunner | jest / mocha / vitest / karma / etc. |
coverageAnalysis | perTest (fastest - only re-run tests that touched the mutated line). |
thresholds.break | Mutation score below this → npx stryker run exits non-zero (CI gate). |
incremental | true to only mutate files changed since last run. |
concurrency | Worker count (default = CPU count). |
npx stryker run
Per-mutant output:
[Survived] Conditional Boundary
src/cart.ts:42:5
- if (item.qty < 0) throw new Error(...);
+ if (item.qty <= 0) throw new Error(...);
Tests run: 12 (all passed — mutant survived).
Survived mutant = tests passed despite the introduced bug = the tests don't actually catch this regression. Add a test for qty = 0.
Ran 142 tests.
Mutants killed: 198
Mutants survived: 24
Mutants timed out: 3
Mutants no coverage: 12
Mutation score: 84.7%
Per stryker-intro: thresholds via the thresholds object
gate the build:
>= high (80) → green.>= low (60) → yellow.< break (50) → red, exit code != 0.The HTML report shows per-file mutation breakdown - drill in to see which mutants survived and where to add tests.
Full mutation runs are slow (5-30 min on medium codebases). For PRs, incremental mode only mutates changed files:
npx stryker run --incremental
This stores state in .stryker-tmp/ (commit to git so subsequent
runs benefit). Per-PR runs typically complete in <2 min.
- name: Mutation testing (changed files only)
if: github.event_name == 'pull_request'
run: npx stryker run --incremental
- name: Full mutation run
if: github.ref == 'refs/heads/main' && github.event.schedule == '0 2 * * 0'
run: npx stryker run
- uses: actions/upload-artifact@v4
if: always()
with:
name: stryker-report
path: reports/mutation/
Pattern: incremental on PRs (fast feedback) + full weekly run on main (catches drift in unchanged code).
| Anti-pattern | Why it fails | Fix |
|---|---|---|
Setting break: 100 | Impossible bar; team disables. | Start at 50-60; ratchet up over time. |
Running on every PR without --incremental | 30-min PR feedback loop; team disables. | --incremental for PRs (Step 5). |
| Mutating test files | Mutants in test code don't reflect production quality. | Exclude in mutate glob (Step 2). |
| Ignoring "no coverage" mutants | Files in mutate glob without any test coverage; mutation testing wastes effort. | Either add tests OR exclude those files from mutate. |
Using coverageAnalysis: off | Re-runs entire suite per mutant; very slow. | perTest (the default; Step 2). |
| Hard-coding break threshold without team agreement | Fails surprise PRs; team disables. | Land threshold via PR with team review. |
tsconfig.tsbuildinfo)
for mutation runs if not needed.stryker-net-mutation -
.NET sibling.pitest-mutation - Java sibling.mutmut-mutation - Python sibling.mull-mutation - C/C++ sibling.mutation-survivor-explainer - agent that suggests the missing test for a survivor.npx claudepluginhub testland/qa --plugin qa-mutation-testingProvides a checklist for code reviews covering functionality, security, performance, maintainability, tests, and quality. Use for pull requests, audits, team standards, and developer training.