2 releases

Uses new Rust 2024

new 0.0.2 Mar 28, 2025
0.0.1 Mar 28, 2025

#9 in #harness


Used in rustest

MIT/Apache

23KB
337 lines

rustest: Helps you better test programs

The rustest framework makes it easy to write small, readable tests, and can scale to support complex functional testing for applications and libraries.

Think about pytest, but for rust.

Features

  • Fixture Management: Easily define and manage test fixtures, including:
  • fixture scopes (unique, test, global)
  • setup and teardown functionalities
  • fixtures dependencies
  • Parametrized Tests: Run tests with different parameters, generating multiple test cases from a single test function.
  • Easy to use: Define tests using standard #[test] attributes, providing flexibility and familiarity.

Why a new test framework ?

When adding tests to the waj crate I was needed to have a global (static) fixture with teardown. Something like :

#[fixture]
fn Command() -> std::process::Command {
    std::process::Command::new("bash")
        .stdout(Stdio::piped())
        .arg("-c")
        .arg("while true; do sleep 1; done")
}

#[fixture(scope=global, teardown=|v| v.kill())]
fn RunningProcess(cmd: Command) -> std::io::Result<Box<std::process::Child>> {
    Ok(Box::new(cmd.spawn()?))
}

I have found not test framework allowing to have teardown on global fixtures. Storing the running process in a static LazyLock allow to have a simple "fixture" but, as statics are not drop, no teardown either.

So I had to implement it, and I finish with a "full" test framework. This became this crate.

This crate, and its API, take inspirations from:

It is based on libtest-mimic to run the tests.

Getting Started

Setup

Add rustest to your Cargo.toml file:

$ cargo add --dev rustest

Rustest comes with its own test harness, so you must deactivate the default one in Cargo.toml:

# In Cargo.toml

[[test]]
name = "test_name" # for a test located at "tests/test_name.rs"
harness = false

[[test]]
name = "other_test" # for a test located at "tests/other_test.rs"
harness = false

You also need to add a main function in each of your integration tests. To do so simply call the macro rustest::main! {} at end of the integration test (after #[test] definitions).

Usage Examples

Here are some examples demonstrating rustest's key features. The file tests/test.rs shows all rustest's features and acts as examples and documentation.

Simple Test:

Simple tests are as simple as with standard test library. You just have to define the main function by invocquing the main! macro.

use rustest::{test, main};

#[test]
fn simple_test() {
    assert_eq!(5*6, 30)
}

main! {}

Failing Tests

Tests can be marked as expecting to fail. Either with #[xfail] attribute or #[test(xfail)]

use rustest::{test, main};

#[test(xfail)]
fn failing_test() {
    assert_eq!(5*6, 31)
}

#[test]
#[xfail]
fn failing_test_bis() {
    assert_eq!(5*6, 31)
}

main! {}

Fixture Example:

You can define any fixtures using the #[fixture] attribute on a function.

// This define a fixture name ANumber which can be deref to u32.
// The function will be called to everytime we need a `ANumber` to populate the fixture
#[fixture]
fn ANumber() -> u32 {
    5
}

// Fixtures are requested by their types.
#[test]
fn test_with_fixture(number: ANumber) {
    assert_eq!(*number, 5);
}

Fixture teardown

You can define a teardown function to be called when the fixture is drop:

#[fixture(teardown:|v| println!("Teardown with value {}", v))]
fn TeardownNumber() -> u32 {
    5
}

// Print "Teardown with value 5" at end of test.
#[test]
fn test_with_teardown_fixture(number: TeardownNumber) {
    assert_eq!(*number, 5);
}    

Fixture Scope:

By default, fixtures are created each time they are requested.

static GLOBAL_COUNTER: AtomicU32 = AtomicU32::new(0);

#[fixture]
fn Counter() -> u32 {
    GLOBAL_COUNTER.fetch_add(1, Ordering::Relaxed)
}

#[test]
fn test_counter(counter1: Counter, counter2: Counter) {
    assert_ne!(*counter1, *counter2);
    assert_eq!(*counter1, 0);
    assert_eq!(*counter2, 1);
}

With scope=test, we create only one fixture (of each type) per test.

This will create twice the TestCounter

#[fixture(scope=test)]
fn TestCounter() -> u32 {
    GLOBAL_COUNTER.fetch_add(1, Ordering::Relaxed)
}

#[test]
fn test_local_counter1(counter1: TestCounter, counter2: TestCounter) {
    assert_eq!(*counter1, *counter2);
    assert_eq!(*counter1, 2);
    assert_eq!(*counter2, 2);
}

#[test]
fn test_local_counter2(counter1: TestCounter, counter2: TestCounter) {
    assert_eq!(*counter1, *counter2);
    assert_eq!(*counter1, 3);
    assert_eq!(*counter2, 3);
}

A global scope make the fixture created only once:

#[fixture(scope=global)]
fn GlobalCounter() -> u32 {
    GLOBAL_COUNTER.fetch_add(1, Ordering::Relaxed)
}

#[test]
fn test_global_counter1(counter1: GlobalCounter, counter2: GlobalCounter) {
    assert_eq!(*counter1, *counter2);
    assert_eq!(*counter1, 4);
    assert_eq!(*counter2, 4);
}

#[test]
fn test_global_counter2(counter1: GlobalCounter, counter2: GlobalCounter) {
    assert_eq!(*counter1, *counter2);
    assert_eq!(*counter1, 4);
    assert_eq!(*counter2, 4);
}

Parametrized Fixture:

#[fixture(params = [1, 5])]
fn ParametrizedFixture(param: rustest::FixtureParam<u32>) -> u32 {
    param.0
}

#[test]
fn test_parametrized_fixture(param: ParametrizedFixture) {
    assert!([1, 5].contains(&param));
}

Fixtures can use fixtures

fn ANumberAsString(number: ANumber) -> String {
    format!("This is a number : {}", *number)
}

#[test]
fn test_number_string(text: ANumberAsString) {
    assert_eq!(*text, "This is a number : 5")
}

Fixtures can be Generic

fn NumberAsString<Source>(number: Source) -> String
where
    Source: rustest::Fixture<Type =u32>
{
    format!("This is a number : {}", *number)
}

#[fixture]
fn TheNumber6() -> u32 {
    6
}

#[test]
fn test_number_string_5(text: NumberAsString<ANumber>) {
    assert_eq!(*text, "This is a number : 5")
}

#[test]
fn test_number_string_6(text: NumberAsString<TheNumber6>) {
    assert_eq!(*text, "This is a number : 6")
}

Running Tests:

Execute your tests using the standard cargo test command. Rustest uses libtest-mimic which provides a compatible interface for running your tests.

cargo test

Contributing

Rustest is pretty young. Issue reports and PR are welcomed !

License

Licensed under either of

Dependencies

~0.5–1MB
~22K SLoC