File: test_casing.rs

package info (click to toggle)
rust-test-casing 0.1.3-4
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 300 kB
  • sloc: makefile: 2
file content (121 lines) | stat: -rw-r--r-- 3,877 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
//! Integration tests for `test_casing` macro.

use async_std::task;

use std::error::Error;

use test_casing::{cases, test_casing, Product, TestCases};

// Cases can be reused across multiple tests.
const CASES: TestCases<i32> = cases!([2, 3, 5, 8]);

#[test_casing(4, CASES)]
#[test]
fn numbers_are_small(number: i32) {
    assert!((0..10).contains(&number));
}

#[test]
fn another_number_is_small() {
    numbers_are_small(1);
}

#[allow(unused_variables)] // should be retained on the target fn
#[test_casing(4, CASES)]
#[ignore = "testing that `#[ignore]` attr works"]
fn numbers_are_large(number: i32) {
    unimplemented!("implement later");
}

#[test_casing(4, CASES)]
fn numbers_are_small_with_errors(number: i32) -> Result<(), Box<dyn Error>> {
    if number < 10 {
        Ok(())
    } else {
        Err("number is too large".into())
    }
}

// It's possible to specify cases with multiple args. The semantics of args
// (e.g., whether any of them are expected values) is up to the user.
const MULTI_ARG_CASES: TestCases<(i32, &str)> = cases!([(2, "2"), (3, "3"), (5, "5")]);

#[test_casing(3, MULTI_ARG_CASES)]
#[test]
fn number_can_be_converted_to_string(number: i32, expected: &str) {
    assert_eq!(number.to_string(), expected);
}

#[test_casing(3, MULTI_ARG_CASES)]
fn number_can_be_converted_to_string_with_tuple_input((number, expected): (i32, &str)) {
    assert_eq!(number.to_string(), expected);
}

// `Product` allows testing a Cartesian product of the contained cases of arity in 2..8.
#[test_casing(12, Product((CASES, ["first", "second", "third"])))]
fn cartesian_product(number: i32, s: &str) {
    assert_ne!(number.to_string(), s);
}

// If it semantically makes sense, it's possible to borrow some of the returned case args
// using a `#[map(ref)]` attr on the arg. An optional transform on the reference in a form
// of a path can be specified as well. (Here, the transform is trivial and serves the purpose
// of assisting the Rust type inference.)
#[test_casing(5, cases!{(0..5).map(|i| (i.to_string(), i))})]
fn string_conversion(#[map(ref = String::as_str)] s: &str, expected: i32) {
    let actual: i32 = s.parse().unwrap();
    assert_eq!(actual, expected);
}

#[test_casing(3, ["not a number", "-", ""])]
#[should_panic(expected = "ParseIntError")]
fn string_conversion_fail(bogus_str: &str) {
    string_conversion(bogus_str, 42);
}

const STRING_CASES: TestCases<(String, i32)> = cases!((0..5).map(|i| (i.to_string(), i)));

#[test_casing(5, STRING_CASES)]
#[async_std::test]
async fn async_string_conversion_without_output(#[map(ref)] s: &str, expected: i32) {
    let actual: i32 = s.parse().unwrap();
    assert_eq!(actual, expected);
    let expected_string = task::spawn_blocking(move || expected.to_string()).await;
    assert_eq!(expected_string, s);
}

#[test_casing(5, STRING_CASES)]
#[async_std::test]
async fn async_string_conversion(#[map(ref)] s: &str, expected: i32) -> Result<(), Box<dyn Error>> {
    let actual: i32 = s.parse()?;
    assert_eq!(actual, expected);
    let expected_string = task::spawn_blocking(move || expected.to_string()).await;
    assert_eq!(expected_string, s);
    Ok(())
}

#[test]
fn unit_test_detection_works() {
    assert!(option_env!("CARGO_TARGET_TMPDIR").is_some());
}

// Tests paths to tests in modules.
mod random {
    use rand::{rngs::StdRng, Rng, SeedableRng};

    use std::iter;

    use test_casing::{cases, test_casing, TestCases};

    // The library can be used for randomized tests as well, but it's probably not the best choice
    // if the number of test cases should be large.
    const RANDOM_NUMBERS: TestCases<u32> = cases!({
        let mut rng = StdRng::seed_from_u64(123_456);
        iter::repeat_with(move || rng.gen())
    });

    #[test_casing(10, RANDOM_NUMBERS)]
    fn randomized_tests(value: u32) {
        assert!(value.to_string().len() <= 10);
    }
}