hegeltest 0.4.2

Property-based testing for Rust, built on Hypothesis
Documentation
mod common;

use common::utils::assert_all_examples;
use hegel::TestCase;
use hegel::generators::{self as gs, Generator};

#[hegel::test]
fn test_compose_basic(tc: TestCase) {
    let value = tc.draw(hegel::compose!(|tc| {
        tc.draw(gs::integers::<i32>().min_value(0).max_value(100))
    }));
    assert!((0..=100).contains(&value));
}

#[hegel::test]
fn test_compose_dependent_generation(tc: TestCase) {
    let (x, y) = tc.draw(hegel::compose!(|tc| {
        let x = tc.draw(gs::integers::<i32>().min_value(0).max_value(50));
        let y = tc.draw(gs::integers::<i32>().min_value(x).max_value(100));
        (x, y)
    }));
    assert!(y >= x);
    assert!((0..=50).contains(&x));
    assert!((0..=100).contains(&y));
}

#[hegel::test]
fn test_compose_with_map(tc: TestCase) {
    let value = tc.draw(
        hegel::compose!(|tc| { tc.draw(gs::integers::<i32>().min_value(0).max_value(10)) })
            .map(|n| n * 2),
    );
    assert!(value % 2 == 0);
    assert!((0..=20).contains(&value));
}

#[hegel::test]
fn test_compose_with_filter(tc: TestCase) {
    let value = tc.draw(
        hegel::compose!(|tc| { tc.draw(gs::integers::<i32>().min_value(0).max_value(100)) })
            .filter(|n| n % 2 == 0),
    );
    assert!(value % 2 == 0);
}

#[hegel::test]
fn test_compose_with_boxed(tc: TestCase) {
    let g =
        hegel::compose!(|tc| { tc.draw(gs::integers::<i32>().min_value(0).max_value(50)) }).boxed();
    let value = tc.draw(g);
    assert!((0..=50).contains(&value));
}

#[test]
fn test_compose_assert_all_examples() {
    assert_all_examples(
        hegel::compose!(|tc| {
            let x = tc.draw(gs::integers::<i32>().min_value(0).max_value(100));
            let y = tc.draw(gs::integers::<i32>().min_value(0).max_value(100));
            (x, y)
        }),
        |&(x, y)| (0..=100).contains(&x) && (0..=100).contains(&y),
    );
}

#[hegel::test]
fn test_compose_inside_one_of(tc: TestCase) {
    let value: i32 = tc.draw(hegel::one_of!(
        hegel::compose!(|tc| { tc.draw(gs::integers::<i32>().min_value(0).max_value(10)) }),
        gs::integers::<i32>().min_value(100).max_value(110),
    ));
    assert!((0..=10).contains(&value) || (100..=110).contains(&value));
}

#[hegel::test]
fn test_compose_list_with_index(tc: TestCase) {
    let (list, index) = tc.draw(hegel::compose!(|tc| {
        let list = tc.draw(gs::vecs(gs::integers::<i32>()).min_size(1).max_size(20));
        let index = tc.draw(
            gs::integers::<usize>()
                .min_value(0)
                .max_value(list.len() - 1),
        );
        (list, index)
    }));
    assert!(!list.is_empty());
    assert!(index < list.len());
}

#[hegel::test]
fn test_compose_nested(tc: TestCase) {
    // tc.draw() works inside nested compose blocks
    let (_, inner_val) = tc.draw(hegel::compose!(|tc| {
        tc.draw(hegel::compose!(|tc| {}));
        let v = tc.draw(gs::integers::<i32>());
        ((), v)
    }));
    // Just verify it doesn't panic and produces a value
    let _ = inner_val;
}

#[hegel::test]
fn test_compose_string_building(tc: TestCase) {
    let s = tc.draw(hegel::compose!(|tc| {
        let prefix = tc.draw(gs::sampled_from(vec!["hello", "world"]));
        let n = tc.draw(gs::integers::<i32>().min_value(0).max_value(99));
        format!("{}-{}", prefix, n)
    }));
    assert!(s.starts_with("hello-") || s.starts_with("world-"));
}