#cli-command #integration-tests #test-suite #parameterized #run-command #test-output

extel

An extended testing library for scaffolding tests as quickly and easily as possible

7 releases

0.2.2 Oct 16, 2023
0.2.1 Oct 2, 2023
0.1.3 Sep 24, 2023

#647 in Testing

46 downloads per month

MIT license

39KB
633 lines

Extel - Extended Testing Library

Extel is a testing library that intends to help create scalable test suites with stateless tests. Extel's primary purpose is to make writing tests fast, easy, and performant. A common use case for Extel is when writing integration tests for external binaries/executables. Extel comes with some macros for creating and parsing command results to streamline the process of creating and running CLI commands.

Think of Extel like a scripting language for tests. There's no need for importing a set of wacky modules, or writing weird redirections in a shell language to capture stdout. Instead, you can use Extel to save the output or status results of the commands you want to run, and then chose where your test output goes, too!

Usage

Extel is intended to function on zero argument or single argument functions (the former being called single tests and the latter parameterized tests). After creating your test function, you can register it using the init_test_suite macro. This will scaffold a struct containing pointers to the functions passed in. Calling the run function on the generated struct will immediately go through all tests and collect the results into a vector for the user to parse through if they so wish.

Note that all Extel test functions expect an ExtelResult in its return. Using another type will cause the macros generating the test suite to fail, or will return some weird errors while using the parameters proc macro.

use extel::prelude::*;
use extel_parameterized::parameters;

fn single_test() -> ExtelResult {
    let mut my_cmd = cmd!("echo -n \"hello world\"");
    let output = my_cmd.output()?;
    let string_output = String::from_utf8(output.stdout)?;

    extel_assert!(
        string_output == *"hello world",
        "expected 'hello world', got '{}'",
        string_output
    )
}

#[parameters(1, 2, -2, 4)]
fn param_test(x: i32) -> ExtelResult {
    extel_assert!(x >= 0, "{} < 0", x)
}

fn main() {
    init_test_suite!(ExtelDemo, single_test, param_test);
    ExtelDemo::run(TestConfig::default());
}

Extel supports error propagation through ?. If the From trait for your custom error is not supported, you can use the err! macro to help map errors to a generic TestFailed variant.

fn cool_err_handling() -> ExtelResult {
    let invalid_utf8 = *b"\xE0\x80\x80";
    let utf8_check = String::from_utf8(invalid_utf8.into())?;
    let io_check = std::fs::File::open("./file_not_found.txt")?;
    let your_custom_err = get_custom_result().map_err(|e| err!("{}", e))?;
    pass!()
}

Dependencies

~230–680KB
~16K SLoC