3 unstable releases

Uses new Rust 2024

new 0.1.0 Apr 1, 2025
0.0.2 Mar 28, 2025
0.0.1 Mar 28, 2025

#178 in Testing

Download history 277/week @ 2025-03-26

277 downloads per month

MIT/Apache

47KB
579 lines

Crate Docs Status Apache 2.0 Licensed MIT Licensed

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 add an empty main function and mark it with #[rustest::main] attribute.

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. Don't forget to define the main function.

use rustest::{test, main};

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

#[main]
fn 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]
fn 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:u32=[1, 5])]
fn ParametrizedFixture(p: Param) -> u32 {
    *p
}

#[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

#[fixture]
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

~1.3–1.9MB
~34K SLoC