cargo-mutants

https://github.com/sourcefrog/cargo-mutants

Tests crates.io libs.rs Maturity: Beta

cargo-mutants is a mutation testing tool for Rust. It helps you improve your program's quality by finding functions whose body could be replaced without causing any tests to fail.

Coverage measurements can be helpful, but they really tell you what code is reached by a test, and not whether the test really checks anything about the behavior of the code. Mutation tests give different information, about whether the tests really check the code's behavior.

Install

sh cargo install cargo-mutants

Using cargo-mutants

Just run cargo mutants in a Rust source directory, and it will point out functions that may be inadequately tested:

sh ; cargo mutants Freshen source tree ... ok in 0.031s Copy source and build products to scratch directory ... 192 MB in 0.116s Unmutated baseline ... ok in 0.235s Auto-set test timeout to 20.0s Found 17 mutants to test src/lib.rs:168: replace <??>::new -> CopyOptions < 'f > with Default::default() ... NOT CAUGHT in 0.736s src/lib.rs:386: replace Error::source -> Option < & (dyn std :: error :: Error + 'static) > with Default::default() ... NOT CAUGHT in 0.643s src/lib.rs:485: replace copy_symlink -> Result < () > with Ok(Default::default()) ... NOT CAUGHT in 0.767s

In v0.5.1 of the cp_r crate, the copy_symlink function was reached by a test but not adequately tested.

Command-line options

-d, --dir: Test the Rust tree in the given directory, rather than the default directory.

-f, --file FILE: Mutate only functions in files matching the given name or glob. If the glob contains / it matches against the path from the source tree root; otherwise it matches only against the file name. If used together with --exclude argument, then the files to be examined are matched before the files to be excluded.

-e, --exclude FILE: Exclude files from mutants generation, matching the given name or glob. If the glob contains / it matches against the path from the source tree root; otherwise it matches only against the file name. If used together with --file argument, then the files to be examined are matched before the files to be excluded.

--list: Show what mutants could be generated, without running them.

--diff: With --list, also include a diff of the source change for each mutant.

--json: With --list, show the list in json.

--check: Run cargo check on all generated mutants to find out which ones are viable, but don't actually run the tests.

--no-copy-target: Don't copy the /target directory from the source, and don't freshen the source directory before copying it. The first "baseline" build in the scratch directory will be a clean build with nothing in /target. This will typically be slower (which is why /target is copied by default) but it might help in debugging any issues with the build. (And, in niche cases where there is a very large volume of old unreferenced content in /target, it might conceivably be faster, but that's probably better dealt with by cargo clean in the source directory.)

--no-shuffle: Test mutants in the fixed order they're found in the source rather than the default behavior of running them in random order. (Shuffling is intended to surface new and different mutants earlier on repeated partial runs of cargo-mutants.)

-v, --caught: Also print mutants that were caught by tests.

-V, --unviable: Also print mutants that failed cargo build.

--no-times: Don't print elapsed times.

--timeout: Set a fixed timeout for each cargo test run, to catch mutations that cause a hang. By default a timeout is automatically determined.

--cargo-arg: Passes the option argument to cargo check, build, and test. For example, --cargo-arg --release.

Passing arguments to cargo test

Command-line options following a -- delimiter are passed through to cargo test, which can be used for example to exclude doctests (which tend to be slow to build and run):

sh cargo mutants -- --all-targets

You can use a second double-dash to pass options through to the test targets:

sh cargo mutants -- -- --test-threads 1 --nocapture

Understanding the results

If tests fail in a clean copy of the tree, there might be an (intermittent) failure in the source directory, or there might be some problem that stops them passing when run from a different location, such as a relative path in Cargo.toml. Fix this first.

Otherwise, cargo mutants generates every mutant it can. All mutants fall in to one of these categories:

By default only missed mutants and timeouts are printed because they're the most actionable. Others can be shown with the -v and -V options.

Skipping functions

To mark functions so they are not mutated:

  1. Add a Cargo dependency on the mutants crate, version "0.0.3" or later. (This must be a regular dependency not a dev-dependency, because the annotation will be on non-test code.)

  2. Mark functions with #[mutants::skip] or other attributes containing mutants::skip (e.g. #[cfg_attr(test, mutants::skip)).

See testdata/tree/hang_avoided_by_attr/ for an example.

The crate is tiny and the attribute has no effect on the compiled code. It only flags the function for cargo-mutants.

Note: Currently, cargo-mutants does not (yet) evaluate attributes like cfg_attr, it only looks for the sequence mutants::skip in the attribute.

Exit codes

mutants.out

A mutants.out directory is created in the source directory. It contains:

Hangs and timeouts

Some mutations to the tree can cause the test suite to hang. For example, in this code, cargo-mutants might try changing should_stop to always return false:

rust while !should_stop() { // something }

cargo mutants automatically sets a timeout when running tests with mutations applied, and reports mutations that hit a timeout. The automatic timeout is the greater of 20 seconds, or 5x the time to run tests with no mutations.

The CARGO_MUTANTS_MINIMUM_TEST_TIMEOUT environment variable, measured in seconds, overrides the minimum time.

You can also set an explicit timeout with the --timeout option. In this case the timeout is also applied to tests run with no mutation.

The timeout does not apply to cargo check or cargo build, only cargo test.

When a test times out, you can mark it with #[mutants::skip] so that future cargo mutants runs go faster.

Performance

Most of the runtime for cargo-mutants is spent in running the program test suite and in running incremental builds: both are done once per viable mutant.

So, anything you can do to make the cargo build and cargo test suite faster will have a multiplicative effect on cargo mutants run time, and of course will also make normal development more pleasant.

There's lots of good advice on the web, including https://matklad.github.io/2021/09/04/fast-rust-builds.html.

In particular, on Linux, using the Mold linker can improve build times significantly: because cargo-mutants does many incremental builds, link time is important.

Rust doctests are pretty slow, so if you're using them only as testable documentation and not to assert correctness of the code, you can skip them with cargo mutants -- --all-targets.

Hard-to-test cases

Some functions don't cause a test suite failure if emptied, but also cannot be removed. For example, functions to do with managing caches or that have other performance side effects.

Ideally, these should be tested, but doing so in a way that's not flaky can be difficult. cargo-mutants can help in a few ways:

Continuous integration

Here is an example of a GitHub Actions workflow that runs mutation tests and uploads the results as an artifact. This will fail if it finds any uncaught mutants.

```yml name: cargo-mutants

on: [pull_request, push]

jobs: cargo-mutants: runs-on: ubuntu-latest steps: - uses: actions/checkout@v2 - uses: actions-rs/toolchain@v1 with: toolchain: stable - name: Install cargo-mutants run: cargo install cargo-mutants - name: Run mutant tests run: cargo mutants -- --all-features - name: Archive results uses: actions/upload-artifact@v3 if: failure() with: name: mutation-report path: mutants.out ```

How to help

Experience reports in GitHub Discussions or Bugs are very welcome:

It's especially helpful if you can either point to an open source tree that will reproduce the problem (or success) or at least describe how to reproduce it.

Goals

The goal of cargo-mutants is to be easy to run on any Rust source tree, and to tell you something interesting about areas where bugs might be lurking or the tests might be insufficient.

Being easy to use means:

Showing interesting results mean:

How it works

The basic approach is:

The file is parsed using the syn crate, but mutations are applied textually, rather than to the token stream, so that unmutated code retains its prior formatting, comments, line numbers, etc. This makes it possible to show a text diff of the mutation and should make it easier to understand any error messages from the build of the mutated code.

For more details, see DESIGN.md.

Related work

cargo-mutants was inspired by reading about the Descartes mutation-testing tool for Java described in Increment magazine's testing issue.

It's an interesting insight that mutation at the level of a whole function is a practical sweet-spot to discover missing tests, while still making it feasible to exhaustively generate every mutant, at least for moderate-sized trees.

See also: more information on how cargo-mutants compares to other techniques and tools.

Supported Rust versions

Building cargo-mutants requires a recent stable Rust toolchain.

Currently it is tested with 1.58.

After installing cargo-mutants, you should be able to use it to run tests under any toolchain, even toolchains that are too old to build cargo-mutants, using the standard + option to cargo:

sh cargo +1.48 mutants

Limitations, caveats, known bugs, and future enhancements

CAUTION: This tool builds and runs code with machine-generated modifications. If the code under test, or the test suite, has side effects such as writing or deleting files, running it with mutations may be dangerous. Think first about what side effects the test suite could possibly have, and/or run it in a restricted or disposable environment.

cargo-mutants behavior, output formats, command-line syntax, json output formats, etc, may change from one release to the next.

cargo-mutants does not yet understand cargo workspaces, and it will only test the root package. https://github.com/sourcefrog/cargo-mutants/issues/45

cargo-mutants sees the AST of the tree but doesn't fully "understand" the types. Possibly it could learn to get type information from the compiler (or rust-analyzer?), which would help it generate more interesting viable mutants, and fewer unviable mutants.

To make this faster on large trees, we could keep several scratch trees and test them in parallel, which is likely to exploit CPU resources more thoroughly than Cargo's own parallelism: in particular Cargo tends to fall down to a single task during linking, and often comes down to running a single straggler test at a time. https://github.com/sourcefrog/cargo-mutants/issues/39

Code of Conduct

Interaction with or participation in this project is governed by the Rust Code of Conduct.