Writing Tests
The following focuses on writing an integration test. However, writing unit tests is also encouraged!
Testsuite
Cargo has a wide variety of integration tests that execute the cargo
binary
and verify its behavior, located in the testsuite
directory. The
support
crate and snapbox
contain many helpers to make this process easy.
There are two styles of tests that can roughly be categorized as
- functional tests
- The fixture is programmatically defined
- The assertions may be in-source snapshots, hard-coded strings, or programmatically generated
- Easier to share in an issue as a code block is completely self-contained
- ui tests
- The fixture is file-based
- The assertions use file-backed snapshots that can be updated with an env variable
- Easier to review the expected behavior of the command as more details are included
- Easier to get up and running from an existing project
- Easier to reason about as everything is just files in the repo
These tests typically work by creating a temporary “project” with a
Cargo.toml
file, executing the cargo
binary process, and checking the
stdout and stderr output against the expected output.
Functional Tests
Generally, a functional test will be placed in tests/testsuite/<command>.rs
and will look roughly like:
use cargo_test_support::prelude::*;
use cargo_test_support::str;
use cargo_test_support::project;
#[cargo_test]
fn <description>() {
let p = project()
.file("src/main.rs", r#"fn main() { println!("hi!"); }"#)
.build();
p.cargo("run --bin foo")
.with_stderr_data(str![[r#"
[COMPILING] foo [..]
[FINISHED] [..]
[RUNNING] `target/debug/foo`
"#]])
.with_stdout_data(str![["hi!"]])
.run();
}
The #[cargo_test]
attribute is used in place of
#[test]
to inject some setup code and declare requirements for running the
test.
ProjectBuilder
via project()
:
- Each project is in a separate directory in the sandbox
- If you do not specify a
Cargo.toml
manifest usingfile()
, one is automatically created with a project name offoo
usingbasic_manifest()
.
Execs
via p.cargo(...)
:
- This executes the command and evaluates different assertions
- See
support::compare
for an explanation of the string pattern matching. Patterns are used to make it easier to match against the expected output.
- See
Testing Nightly Features
If you are testing a Cargo feature that only works on “nightly” Cargo, then
you need to call masquerade_as_nightly_cargo
on the process builder and pass
the name of the feature as the reason, like this:
p.cargo("build").masquerade_as_nightly_cargo(&["print-im-a-teapot"])
If you are testing a feature that only works on nightly rustc (such as
benchmarks), then you should use the nightly
option of the cargo_test
attribute, like this:
#[cargo_test(nightly, reason = "-Zfoo is unstable")]
This will cause the test to be ignored if not running on the nightly toolchain.
Specifying Dependencies
You should not write any tests that use the network such as contacting crates.io. Typically, simple path dependencies are the easiest way to add a dependency. Example:
let p = project()
.file("Cargo.toml", r#"
[package]
name = "foo"
version = "1.0.0"
[dependencies]
bar = {path = "bar"}
"#)
.file("src/lib.rs", "extern crate bar;")
.file("bar/Cargo.toml", &basic_manifest("bar", "1.0.0"))
.file("bar/src/lib.rs", "")
.build();
If you need to test with registry dependencies, see
support::registry::Package
for creating packages you can depend on.
If you need to test git dependencies, see support::git
to create a git
dependency.
Cross compilation
There are some utilities to help support tests that need to work against a target other than the host. See Running cross tests for more an introduction on cross compilation tests.
Tests that need to do cross-compilation should include this at the top of the test to disable it in scenarios where cross compilation isn’t available:
if cargo_test_support::cross_compile::disabled() {
return;
}
The name of the target can be fetched with the cross_compile::alternate()
function. The name of the host target can be fetched with
cargo_test_support::rustc_host()
.
The cross-tests need to distinguish between targets which can build versus
those which can actually run the resulting executable. Unfortunately, macOS is
currently unable to run an alternate target (Apple removed 32-bit support a
long time ago). For building, x86_64-apple-darwin
will target
x86_64-apple-ios
as its alternate. However, the iOS target can only execute
binaries if the iOS simulator is installed and configured. The simulator is
not available in CI, so all tests that need to run cross-compiled binaries are
disabled on CI. If you are running on macOS locally, and have the simulator
installed, then it should be able to run them.
If the test needs to run the cross-compiled binary, then it should have something like this to exit the test before doing so:
if cargo_test_support::cross_compile::can_run_on_host() {
return;
}
UI Tests
UI Tests are a bit more spread out and generally look like:
tests/testsuite/<command>/mod.rs
:
mod <case>;
tests/testsuite/<command>/<case>/mod.rs
:
use cargo_test_support::compare::assert_ui;
use cargo_test_support::current_dir;
use cargo_test_support::file;
use cargo_test_support::prelude::*;
use cargo_test_support::Project;
#[cargo_test]
fn case() {
let project = Project::from_template(current_dir!().join("in"));
let project_root = project.root();
let cwd = &project_root;
snapbox::cmd::Command::cargo_ui()
.arg("run")
.arg_line("--bin foo")
.current_dir(cwd)
.assert()
.success()
.stdout_matches(file!("stdout.log"))
.stderr_matches(file!("stderr.log"));
assert_ui().subset_matches(current_dir!().join("out"), &project_root);
}
Then populate
tests/testsuite/<command>/<case>/in
with the project’s directory structuretests/testsuite/<command>/<case>/out
with the files you want verifiedtests/testsuite/<command>/<case>/stdout.log
with nothingtests/testsuite/<command>/<case>/stderr.log
with nothing
#[cargo_test]
:
- This is used in place of
#[test]
- This attribute injects code which does some setup before starting the
test, creating a filesystem “sandbox” under the “cargo integration test”
directory for each test such as
/path/to/cargo/target/cit/t123/
- The sandbox will contain a
home
directory that will be used instead of your normal home directory
Project
:
- The project is copied from a directory in the repo
- Each project is in a separate directory in the sandbox
Command
via Command::cargo_ui()
:
- Set up and run a command.
OutputAssert
via Command::assert()
:
- Perform assertions on the result of the
Command
Assert
via assert_ui()
:
- Verify the command modified the file system as expected
Updating Snapshots
The project, stdout, and stderr snapshots can be updated by running with the
SNAPSHOTS=overwrite
environment variable, like:
$ SNAPSHOTS=overwrite cargo test
Be sure to check the snapshots to make sure they make sense.
Testing Nightly Features
If you are testing a Cargo feature that only works on “nightly” Cargo, then
you need to call masquerade_as_nightly_cargo
on the process builder and pass
the name of the feature as the reason, like this:
snapbox::cmd::Command::cargo()
.masquerade_as_nightly_cargo(&["print-im-a-teapot"])
If you are testing a feature that only works on nightly rustc (such as
benchmarks), then you should use the nightly
option of the cargo_test
attribute, like this:
#[cargo_test(nightly, reason = "-Zfoo is unstable")]
This will cause the test to be ignored if not running on the nightly toolchain.
Platform-specific Notes
When checking output, use /
for paths even on Windows: the actual output
of \
on Windows will be replaced with /
.
Be careful when executing binaries on Windows. You should not rename, delete, or overwrite a binary immediately after running it. Under some conditions Windows will fail with errors like “directory not empty” or “failed to remove” or “access is denied”.
Debugging tests
In some cases, you may need to dig into a test that is not working as you expect, or you just generally want to experiment within the sandbox environment. The general process is:
-
Build the sandbox for the test you want to investigate. For example:
cargo test --test testsuite -- features2::inactivate_targets
. -
In another terminal, head into the sandbox directory to inspect the files and run
cargo
directly.-
The sandbox directories start with
t0
for the first test.cd target/tmp/cit/t0
-
Set up the environment so that the sandbox configuration takes effect:
export CARGO_HOME=$(pwd)/home/.cargo
-
Most tests create a
foo
project, so head into that:cd foo
-
-
Run whatever cargo command you want. See Running Cargo for more details on running the correct
cargo
process. Some examples:/path/to/my/cargo/target/debug/cargo check
- Using a debugger like
lldb
orgdb
:lldb /path/to/my/cargo/target/debug/cargo
- Set a breakpoint, for example:
b generate_root_units
- Run with arguments:
r check