radicle_node/
deserializer.rs1use std::io;
2use std::marker::PhantomData;
3
4use crate::bounded;
5use crate::prelude::BoundedVec;
6use crate::service::message::Message;
7use crate::wire;
8
9#[derive(Debug)]
13pub struct Deserializer<const B: usize, D = Message> {
14 unparsed: BoundedVec<u8, B>,
15 item: PhantomData<D>,
16}
17
18impl<const B: usize, D: wire::Decode> Default for Deserializer<B, D> {
19 fn default() -> Self {
20 Self::new(wire::Size::MAX as usize + 1)
21 }
22}
23
24impl<const B: usize, D> TryFrom<Vec<u8>> for Deserializer<B, D> {
25 type Error = bounded::Error;
26
27 fn try_from(unparsed: Vec<u8>) -> Result<Self, Self::Error> {
28 BoundedVec::try_from(unparsed).map(|unparsed| Self {
29 unparsed,
30 item: PhantomData,
31 })
32 }
33}
34
35impl<const B: usize, D: wire::Decode> Deserializer<B, D> {
36 pub fn new(capacity: usize) -> Self {
38 Self {
39 unparsed: BoundedVec::with_capacity(capacity)
40 .expect("Deserializer::new: capacity exceeds maximum"),
41 item: PhantomData,
42 }
43 }
44
45 pub fn input(&mut self, bytes: &[u8]) -> Result<(), bounded::Error> {
47 self.unparsed.extend_from_slice(bytes)
48 }
49
50 pub fn deserialize_next(&mut self) -> Result<Option<D>, wire::Error> {
52 let mut reader = io::Cursor::new(self.unparsed.as_slice());
53
54 match D::decode(&mut reader) {
55 Ok(msg) => {
56 let pos = reader.position() as usize;
57 self.unparsed.drain(..pos);
58
59 Ok(Some(msg))
60 }
61 Err(err) if err.is_eof() => Ok(None),
62 Err(err) => Err(err),
63 }
64 }
65
66 pub fn unparsed(&mut self) -> impl ExactSizeIterator<Item = u8> + '_ {
68 self.unparsed.drain(..)
69 }
70
71 pub fn is_empty(&self) -> bool {
73 self.unparsed.is_empty()
74 }
75
76 pub fn len(&self) -> usize {
78 self.unparsed.len()
79 }
80}
81
82impl<const B: usize, D: wire::Decode> io::Write for Deserializer<B, D> {
83 fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
84 self.input(buf).map_err(|_| io::ErrorKind::OutOfMemory)?;
85
86 Ok(buf.len())
87 }
88
89 fn flush(&mut self) -> io::Result<()> {
90 Ok(())
91 }
92}
93
94impl<const B: usize, D: wire::Decode> Iterator for Deserializer<B, D> {
95 type Item = Result<D, wire::Error>;
96
97 fn next(&mut self) -> Option<Self::Item> {
98 self.deserialize_next().transpose()
99 }
100}
101
102#[cfg(test)]
103mod test {
104 use super::*;
105 use qcheck_macros::quickcheck;
106
107 use crate::test::assert_matches;
108
109 const MSG_HELLO: &[u8] = &[5, b'h', b'e', b'l', b'l', b'o'];
110 const MSG_BYE: &[u8] = &[3, b'b', b'y', b'e'];
111
112 #[test]
113 fn test_decode_next() {
114 let mut decoder = Deserializer::<1024, String>::new(8);
115
116 decoder.input(&[3, b'b']).unwrap();
117 assert_matches!(decoder.deserialize_next(), Ok(None));
118 assert_eq!(decoder.unparsed.len(), 2);
119
120 decoder.input(&[b'y']).unwrap();
121 assert_matches!(decoder.deserialize_next(), Ok(None));
122 assert_eq!(decoder.unparsed.len(), 3);
123
124 decoder.input(&[b'e']).unwrap();
125 assert_matches!(decoder.deserialize_next(), Ok(Some(s)) if s.as_str() == "bye");
126 assert_eq!(decoder.unparsed.len(), 0);
127 assert!(decoder.is_empty());
128 }
129
130 #[test]
131 fn test_unparsed() {
132 let mut decoder = Deserializer::<1024, String>::new(8);
133
134 decoder.input(&[3, b'b', b'y']).unwrap();
135 assert_eq!(decoder.unparsed().collect::<Vec<_>>(), vec![3, b'b', b'y']);
136 assert!(decoder.is_empty());
137 }
138
139 #[quickcheck]
140 fn prop_decode_next(chunk_size: usize) {
141 let mut bytes = vec![];
142 let mut msgs = vec![];
143 let mut decoder = Deserializer::<1024, String>::new(8);
144
145 let chunk_size = 1 + chunk_size % MSG_HELLO.len() + MSG_BYE.len();
146
147 bytes.extend_from_slice(MSG_HELLO);
148 bytes.extend_from_slice(MSG_BYE);
149
150 for chunk in bytes.as_slice().chunks(chunk_size) {
151 decoder.input(chunk).unwrap();
152
153 while let Some(msg) = decoder.deserialize_next().unwrap() {
154 msgs.push(msg);
155 }
156 }
157
158 assert_eq!(decoder.unparsed.len(), 0);
159 assert_eq!(msgs.len(), 2);
160 assert_eq!(msgs[0], String::from("hello"));
161 assert_eq!(msgs[1], String::from("bye"));
162 }
163}