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
//! @todo move all this out to the serialized bytes crate
use crate::prelude::*;
use holochain_serialized_bytes::prelude::*;
use rand::seq::SliceRandom;

#[derive(Clone, Copy)]
/// there are many different types of things that we could reasonably serialize in our examples
/// a list of things that we serialize iteratively (Predictable) or randomly (Unpredictable)
pub enum ThingsToSerialize {
    Unit,
    Bool,
    Number,
    String,
}

pub const THINGS_TO_SERIALIZE: [ThingsToSerialize; 4] = [
    ThingsToSerialize::Unit,
    ThingsToSerialize::Bool,
    ThingsToSerialize::Number,
    ThingsToSerialize::String,
];

/// Serialization wrapper for bools
#[derive(serde::Serialize, serde::Deserialize, SerializedBytes, Debug)]
struct BoolWrap(bool);
/// Serialization wrapper for u32 (number)
#[derive(serde::Serialize, serde::Deserialize, SerializedBytes, Debug)]
struct U32Wrap(u32);
/// Serialzation wrapper for Strings
#[derive(serde::Serialize, serde::Deserialize, SerializedBytes, Debug)]
struct StringWrap(String);

fixturator!(
    SerializedBytes,
    { SerializedBytes::try_from(()).unwrap() },
    {
        // randomly select a thing to serialize
        let thing_to_serialize = THINGS_TO_SERIALIZE
            .to_vec()
            .choose(&mut crate::rng())
            .unwrap()
            .to_owned();

        // serialize a thing based on a delegated fixturator
        match thing_to_serialize {
            ThingsToSerialize::Unit =>
            {
                #[allow(clippy::unit_arg)]
                UnitFixturator::new(Unpredictable)
                    .next()
                    .unwrap()
                    .try_into()
                    .unwrap()
            }
            ThingsToSerialize::Bool => BoolWrap(BoolFixturator::new(Unpredictable).next().unwrap())
                .try_into()
                .unwrap(),
            ThingsToSerialize::Number => U32Wrap(U32Fixturator::new(Unpredictable).next().unwrap())
                .try_into()
                .unwrap(),
            ThingsToSerialize::String => {
                StringWrap(StringFixturator::new(Unpredictable).next().unwrap())
                    .try_into()
                    .unwrap()
            }
        }
    },
    {
        let mut index = get_fixt_index!();
        // iteratively select a thing to serialize
        let thing_to_serialize = THINGS_TO_SERIALIZE
            .iter()
            .copied()
            .cycle()
            .nth(index)
            .unwrap();

        // serialize a thing based on a delegated fixturator
        let ret: SerializedBytes = match thing_to_serialize {
            ThingsToSerialize::Unit =>
            {
                #[allow(clippy::unit_arg)]
                UnitFixturator::new_indexed(Predictable, index)
                    .next()
                    .unwrap()
                    .try_into()
                    .unwrap()
            }
            ThingsToSerialize::Bool => BoolWrap(
                BoolFixturator::new_indexed(Predictable, index)
                    .next()
                    .unwrap(),
            )
            .try_into()
            .unwrap(),
            ThingsToSerialize::Number => U32Wrap(
                U32Fixturator::new_indexed(Predictable, index)
                    .next()
                    .unwrap(),
            )
            .try_into()
            .unwrap(),
            ThingsToSerialize::String => StringWrap(
                StringFixturator::new_indexed(Predictable, index)
                    .next()
                    .unwrap(),
            )
            .try_into()
            .unwrap(),
        };

        index += 1;
        set_fixt_index!(index);
        ret
    }
);