forked from amqp-rs/amq-protocol
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathflags.rs
117 lines (104 loc) · 3.67 KB
/
flags.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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
use crate::types::Boolean;
use serde::{Deserialize, Serialize};
/// A struct representing AMQP boolean flags for RPC
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)]
pub struct AMQPFlags {
flags: Vec<(String, Boolean)>,
}
impl AMQPFlags {
/// Add a boolean flag with a name
pub fn add_flag(&mut self, name: String, flag: Boolean) {
self.flags.push((name, flag));
}
/// Get the value of a boolean flag by name, if present
pub fn get_flag(&self, name: &str) -> Option<Boolean> {
self.flags.iter().find(|(n, _)| n == name).map(|(_, v)| *v)
}
/// Get the AMQPFlags serialized for AMQP RPC
pub fn get_bytes(&self) -> Vec<u8> {
self.flags
.chunks(8)
.map(|v| {
v.iter()
.enumerate()
.map(|(idx, (_, b))| if *b { 1 << idx } else { 0 })
.sum()
})
.collect()
}
/// Initialize AMQPFlags from AMQP RPC serialization
pub fn from_bytes<I: nom::InputIter<Item = u8>>(names: &[&str], bytes: I) -> AMQPFlags {
let flags = names
.iter()
.map(ToString::to_string)
.zip(bytes.iter_elements().flat_map(|b| {
let mut v = Vec::new();
for s in 0..8 {
v.push(((b & (1 << s)) >> s) == 1)
}
v
}))
.collect();
AMQPFlags { flags }
}
}
impl Default for AMQPFlags {
fn default() -> AMQPFlags {
AMQPFlags { flags: Vec::new() }
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_empty_flags() {
let empty: &[u8] = &[];
assert_eq!(AMQPFlags::default().get_bytes().as_slice(), empty);
}
#[test]
fn test_flags() {
let mut flags = AMQPFlags::default();
flags.add_flag("a".to_string(), true);
flags.add_flag("b".to_string(), false);
flags.add_flag("c".to_string(), false);
flags.add_flag("d".to_string(), true);
flags.add_flag("e".to_string(), true);
assert_eq!(flags.get_bytes().as_slice(), &[0b00011001])
}
#[test]
fn test_many_flags() {
let mut flags = AMQPFlags::default();
flags.add_flag("a".to_string(), true);
flags.add_flag("b".to_string(), false);
flags.add_flag("c".to_string(), false);
flags.add_flag("d".to_string(), true);
flags.add_flag("e".to_string(), true);
flags.add_flag("f".to_string(), true);
flags.add_flag("g".to_string(), false);
flags.add_flag("h".to_string(), false);
flags.add_flag("i".to_string(), true);
flags.add_flag("j".to_string(), true);
assert_eq!(flags.get_bytes().as_slice(), &[0b00111001, 0b00000011])
}
#[test]
fn test_lookup_flags() {
let mut flags = AMQPFlags::default();
flags.add_flag("a".to_string(), true);
flags.add_flag("b".to_string(), false);
flags.add_flag("c".to_string(), false);
flags.add_flag("d".to_string(), true);
flags.add_flag("e".to_string(), true);
flags.add_flag("f".to_string(), true);
flags.add_flag("g".to_string(), false);
flags.add_flag("h".to_string(), false);
flags.add_flag("i".to_string(), true);
flags.add_flag("j".to_string(), true);
assert_eq!(flags.get_flag("a"), Some(true));
assert_eq!(flags.get_flag("d"), Some(true));
assert_eq!(flags.get_flag("e"), Some(true));
assert_eq!(flags.get_flag("b"), Some(false));
assert_eq!(flags.get_flag("j"), Some(true));
assert_eq!(flags.get_flag("h"), Some(false));
assert_eq!(flags.get_flag("z"), None);
}
}