plutus_ledger_api/generators/correct/
primitive.rs

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
//! Proptest strategies for most common primitive types
//!
//! These strategies always return valid values.
use num_bigint::{BigInt, BigUint, Sign};
use num_traits::identities::Zero;
use proptest::arbitrary::{any, StrategyFor};
use proptest::char::CharStrategy;
use proptest::collection::vec;
use proptest::collection::{btree_map, btree_set};
use proptest::option;
use proptest::prelude::{prop_oneof, Just};
use proptest::result::maybe_err;
use proptest::strategy::Strategy;
use std::collections::{BTreeMap, BTreeSet};

/// Strategy to generate an arbitrary boolean
pub fn arb_bool() -> StrategyFor<bool> {
    any::<bool>()
}

/// Strategy to generate an arbitrary `Sign`
/// Only used internally, to generate `BigInt`s
fn arb_sign() -> impl Strategy<Value = Sign> {
    // NoSign is only used for 0 values so we're not generating it here
    prop_oneof![Just(Sign::Minus), Just(Sign::Plus)]
}

/// Strategy to generate an arbitrary BigInt
pub fn arb_integer() -> impl Strategy<Value = BigInt> {
    // Wrapping around BigUint.
    (arb_sign(), arb_biguint(2)).prop_map(|(sign, nat)| {
        // NoSign is only used for 0 values.
        BigInt::from_biguint(if nat.is_zero() { Sign::NoSign } else { sign }, nat)
    })
}

/// Strategy to generate an arbitrary non-negative BigInt
pub fn arb_natural(n: usize) -> impl Strategy<Value = BigInt> {
    arb_biguint(n).prop_map(|x| {
        BigInt::from_biguint(
            if x.is_zero() {
                Sign::NoSign
            } else {
                Sign::Plus
            },
            x,
        )
    })
}

/// Helper function to generate a well typed arbitrary natural number
/// Generating `n` vectors of random u32 values, which gives a max bound of u32::MAX ^ n
fn arb_biguint(n: usize) -> impl Strategy<Value = BigUint> {
    vec(any::<u32>(), n).prop_map(BigUint::new)
}

/// Strategy to generate an arbitrary character
pub fn arb_char<'a>() -> CharStrategy<'a> {
    any::<char>()
}

/// Strategy to generate an arbitrary bytestring
pub fn arb_bytes() -> StrategyFor<Vec<u8>> {
    any::<Vec<u8>>()
}

/// Strategy to generate an arbitrary string
pub fn arb_text() -> StrategyFor<String> {
    any::<String>()
}

/// Strategy to generate a complicated data structure
pub fn arb_complicated(
) -> impl Strategy<Value = BTreeMap<String, Result<BTreeSet<char>, Option<Result<Vec<u8>, bool>>>>>
{
    btree_map(
        arb_text(),
        maybe_err(
            btree_set(arb_char(), 20),
            option::of(maybe_err(arb_bytes(), arb_bool())),
        ),
        20,
    )
}