casper_types/
validator_change.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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
use crate::bytesrepr::{self, FromBytes, ToBytes};
#[cfg(any(feature = "testing", test))]
use crate::testing::TestRng;
use alloc::vec::Vec;
#[cfg(feature = "datasize")]
use datasize::DataSize;
#[cfg(feature = "json-schema")]
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};

/// A change to a validator's status between two eras.
#[derive(Serialize, Deserialize, Debug, Eq, PartialEq, Ord, PartialOrd)]
#[cfg_attr(feature = "json-schema", derive(JsonSchema))]
#[cfg_attr(feature = "datasize", derive(DataSize))]
pub enum ValidatorChange {
    /// The validator got newly added to the validator set.
    Added,
    /// The validator was removed from the validator set.
    Removed,
    /// The validator was banned from this era.
    Banned,
    /// The validator was excluded from proposing new blocks in this era.
    CannotPropose,
    /// We saw the validator misbehave in this era.
    SeenAsFaulty,
}

impl ValidatorChange {
    /// Returns a random `ValidatorChange`.
    #[cfg(any(feature = "testing", test))]
    pub fn random(rng: &mut TestRng) -> Self {
        use rand::Rng;

        match rng.gen_range(0..5) {
            ADDED_TAG => ValidatorChange::Added,
            REMOVED_TAG => ValidatorChange::Removed,
            BANNED_TAG => ValidatorChange::Banned,
            CANNOT_PROPOSE_TAG => ValidatorChange::CannotPropose,
            SEEN_AS_FAULTY_TAG => ValidatorChange::SeenAsFaulty,
            _ => unreachable!(),
        }
    }
}

const ADDED_TAG: u8 = 0;
const REMOVED_TAG: u8 = 1;
const BANNED_TAG: u8 = 2;
const CANNOT_PROPOSE_TAG: u8 = 3;
const SEEN_AS_FAULTY_TAG: u8 = 4;

impl ToBytes for ValidatorChange {
    fn to_bytes(&self) -> Result<Vec<u8>, bytesrepr::Error> {
        let mut buffer = bytesrepr::allocate_buffer(self)?;
        self.write_bytes(&mut buffer)?;
        Ok(buffer)
    }

    fn write_bytes(&self, writer: &mut Vec<u8>) -> Result<(), bytesrepr::Error> {
        match self {
            ValidatorChange::Added => ADDED_TAG,
            ValidatorChange::Removed => REMOVED_TAG,
            ValidatorChange::Banned => BANNED_TAG,
            ValidatorChange::CannotPropose => CANNOT_PROPOSE_TAG,
            ValidatorChange::SeenAsFaulty => SEEN_AS_FAULTY_TAG,
        }
        .write_bytes(writer)
    }

    fn serialized_length(&self) -> usize {
        bytesrepr::U8_SERIALIZED_LENGTH
    }
}

impl FromBytes for ValidatorChange {
    fn from_bytes(bytes: &[u8]) -> Result<(Self, &[u8]), bytesrepr::Error> {
        let (tag, remainder) = u8::from_bytes(bytes)?;
        let id = match tag {
            ADDED_TAG => ValidatorChange::Added,
            REMOVED_TAG => ValidatorChange::Removed,
            BANNED_TAG => ValidatorChange::Banned,
            CANNOT_PROPOSE_TAG => ValidatorChange::CannotPropose,
            SEEN_AS_FAULTY_TAG => ValidatorChange::SeenAsFaulty,
            _ => return Err(bytesrepr::Error::NotRepresentable),
        };
        Ok((id, remainder))
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::testing::TestRng;

    #[test]
    fn bytesrepr_roundtrip() {
        let rng = &mut TestRng::new();

        let val = ValidatorChange::random(rng);
        bytesrepr::test_serialization_roundtrip(&val);
    }
}