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
use std::collections::HashMap;
use std::fmt;
use std::fmt::{Debug, Display};

/// The registers of the machine.
pub trait MachineRegister: Debug + Display {
    /// Reads a register.
    fn read(&self, key: char) -> i64;

    /// Writes a register.
    fn write(&mut self, key: char, value: i64);

    /// Clears the register.
    fn clear(&mut self);

    /// Increments (or decrements if by < 0) a register.
    fn increment(&mut self, key: char, by: i64) {
        self.write(key, self.read(key) + by)
    }

    /// Writes a boolean value (1 = true, 0 = false) to a register.
    fn write_bool(&mut self, key: char, condition: bool) {
        self.write(key, i64::from(condition));
    }
}

/// The default implementation using a HashMap.
#[derive(Debug)]
pub struct HashMapRegister {
    registers: HashMap<char, i64>,
}

impl HashMapRegister {
    /// Constructs a new HashMapRegister.
    pub fn new() -> Self {
        HashMapRegister {
            registers: HashMap::new(),
        }
    }
}

impl Default for HashMapRegister {
    fn default() -> Self {
        HashMapRegister::new()
    }
}

impl fmt::Display for HashMapRegister {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        let mut keys: Vec<char> = self.registers.keys().copied().collect();
        keys.sort_unstable();

        write!(f, "[")?;
        for (i, key) in keys.iter().enumerate() {
            if i != 0 {
                write!(f, ", ")?;
            }
            write!(f, "{}=", key)?;
            fmt::Display::fmt(&self.read(*key), f)?;
        }
        write!(f, "]")
    }
}

impl MachineRegister for HashMapRegister {
    fn read(&self, key: char) -> i64 {
        *self.registers.get(&key).unwrap_or(&0)
    }

    fn write(&mut self, key: char, value: i64) {
        *self.registers.entry(key).or_insert(0) = value;
    }

    fn clear(&mut self) {
        self.registers.clear();
    }
}

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

    #[test]
    fn test_read_empty_register() {
        let registers = HashMapRegister::new();
        assert_eq!(registers.read('a'), 0);
    }

    #[test]
    fn test_write_and_read_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 10);
        assert_eq!(registers.read('a'), 10);
    }

    #[test]
    fn test_increment_empty_register() {
        let mut registers = HashMapRegister::new();
        registers.increment('a', 5);
        assert_eq!(registers.read('a'), 5);
    }

    #[test]
    fn test_increment_non_empty_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 5);
        registers.increment('a', 5);
        assert_eq!(registers.read('a'), 10);
    }

    #[test]
    fn test_format_register() {
        let mut registers = HashMapRegister::new();
        registers.write('a', 10);
        registers.write('d', -5);
        registers.write('c', 12);
        registers.write('b', 0);
        assert_eq!(format!("{}", registers), "[a=10, b=0, c=12, d=-5]");
    }
}