cuprate_epee_encoding/
lib.rs1#![cfg_attr(not(feature = "std"), no_std)]
2#[cfg(test)]
63use hex as _;
64
65extern crate alloc;
66
67use alloc::string::ToString;
68use core::str::from_utf8 as str_from_utf8;
69
70use bytes::{Buf, BufMut, Bytes, BytesMut};
71
72pub mod container_as_blob;
73pub mod error;
74mod io;
75pub mod macros;
76pub mod marker;
77mod value;
78mod varint;
79
80pub use error::*;
81use io::*;
82pub use marker::{InnerMarker, Marker};
83pub use value::EpeeValue;
84pub use varint::{read_varint, write_varint};
85
86const HEADER: &[u8] = b"\x01\x11\x01\x01\x01\x01\x02\x01\x01";
89const MAX_STRING_LEN_POSSIBLE: usize = 2000000000;
91const MAX_DEPTH_OF_SKIPPED_OBJECTS: u8 = 20;
93const MAX_NUM_FIELDS: u64 = 1000;
95
96pub trait EpeeObjectBuilder<T>: Default + Sized {
98 fn add_field<B: Buf>(&mut self, name: &str, b: &mut B) -> Result<bool>;
104
105 fn finish(self) -> Result<T>;
107}
108
109pub trait EpeeObject: Sized {
111 type Builder: EpeeObjectBuilder<Self>;
112
113 fn number_of_fields(&self) -> u64;
115
116 fn write_fields<B: BufMut>(self, w: &mut B) -> Result<()>;
118}
119
120pub fn from_bytes<T: EpeeObject, B: Buf>(buf: &mut B) -> Result<T> {
122 read_head_object(buf)
123}
124
125pub fn to_bytes<T: EpeeObject>(val: T) -> Result<BytesMut> {
127 let mut buf = BytesMut::new();
128 write_head_object(val, &mut buf)?;
129 Ok(buf)
130}
131
132fn read_header<B: Buf>(r: &mut B) -> Result<()> {
133 let buf = checked_read(r, |b: &mut B| b.copy_to_bytes(HEADER.len()), HEADER.len())?;
134
135 if &*buf != HEADER {
136 return Err(Error::Format("Data does not contain header"));
137 }
138 Ok(())
139}
140
141fn write_header<B: BufMut>(w: &mut B) -> Result<()> {
142 checked_write(w, BufMut::put_slice, HEADER, HEADER.len())
143}
144
145fn write_head_object<T: EpeeObject, B: BufMut>(val: T, w: &mut B) -> Result<()> {
146 write_header(w)?;
147 val.write(w)
148}
149
150fn read_head_object<T: EpeeObject, B: Buf>(r: &mut B) -> Result<T> {
151 read_header(r)?;
152 let mut skipped_objects = 0;
153 read_object(r, &mut skipped_objects)
154}
155
156fn read_field_name_bytes<B: Buf>(r: &mut B) -> Result<Bytes> {
157 let len: usize = r.get_u8().into();
158
159 checked_read(r, |b: &mut B| b.copy_to_bytes(len), len)
160}
161
162fn write_field_name<B: BufMut>(val: &str, w: &mut B) -> Result<()> {
163 checked_write_primitive(w, BufMut::put_u8, val.len().try_into()?)?;
164 let slice = val.as_bytes();
165 checked_write(w, BufMut::put_slice, slice, slice.len())
166}
167
168pub fn write_field<T: EpeeValue, B: BufMut>(val: T, field_name: &str, w: &mut B) -> Result<()> {
170 if val.should_write() {
171 write_field_name(field_name, w)?;
172 write_epee_value(val, w)?;
173 }
174 Ok(())
175}
176
177fn read_object<T: EpeeObject, B: Buf>(r: &mut B, skipped_objects: &mut u8) -> Result<T> {
178 let mut object_builder = T::Builder::default();
179
180 let number_o_field = read_varint(r)?;
181
182 if number_o_field > MAX_NUM_FIELDS {
183 return Err(Error::Format(
184 "Data has object with more fields than the maximum allowed",
185 ));
186 }
187
188 for _ in 0..number_o_field {
189 let field_name_bytes = read_field_name_bytes(r)?;
190 let field_name = str_from_utf8(&field_name_bytes)?;
191
192 if !object_builder.add_field(field_name, r)? {
193 skip_epee_value(r, skipped_objects)?;
194 }
195 }
196 object_builder.finish()
197}
198
199pub fn read_marker<B: Buf>(r: &mut B) -> Result<Marker> {
202 Marker::try_from(checked_read_primitive(r, Buf::get_u8)?)
203}
204
205pub fn read_epee_value<T: EpeeValue, B: Buf>(r: &mut B) -> Result<T> {
208 let marker = read_marker(r)?;
209 T::read(r, &marker)
210}
211
212fn write_epee_value<T: EpeeValue, B: BufMut>(val: T, w: &mut B) -> Result<()> {
215 checked_write_primitive(w, BufMut::put_u8, T::MARKER.as_u8())?;
216 val.write(w)
217}
218
219pub fn write_bytes<T: AsRef<[u8]>, B: BufMut>(t: T, w: &mut B) -> Result<()> {
246 let bytes = t.as_ref();
247 let len = bytes.len();
248
249 write_varint(len, w)?;
250
251 if w.remaining_mut() < len {
252 return Err(Error::IO("Not enough capacity to write bytes"));
253 }
254
255 w.put_slice(bytes);
256
257 Ok(())
258}
259
260pub fn write_iterator<T, I, B>(iterator: I, w: &mut B) -> Result<()>
288where
289 T: EpeeValue,
290 I: Iterator<Item = T> + ExactSizeIterator,
291 B: BufMut,
292{
293 write_varint(iterator.len(), w)?;
294 for item in iterator {
295 item.write(w)?;
296 }
297 Ok(())
298}
299
300#[derive(Default)]
302struct SkipObjectBuilder;
303
304impl EpeeObjectBuilder<SkipObject> for SkipObjectBuilder {
305 fn add_field<B: Buf>(&mut self, _name: &str, _r: &mut B) -> Result<bool> {
306 Ok(false)
307 }
308
309 fn finish(self) -> Result<SkipObject> {
310 Ok(SkipObject)
311 }
312}
313
314struct SkipObject;
316
317impl EpeeObject for SkipObject {
318 type Builder = SkipObjectBuilder;
319
320 fn number_of_fields(&self) -> u64 {
321 panic!("This is a helper function to use when de-serialising")
322 }
323
324 fn write_fields<B: BufMut>(self, _w: &mut B) -> Result<()> {
325 panic!("This is a helper function to use when de-serialising")
326 }
327}
328
329fn skip_epee_value<B: Buf>(r: &mut B, skipped_objects: &mut u8) -> Result<()> {
332 let marker = read_marker(r)?;
333
334 let len = if marker.is_seq { read_varint(r)? } else { 1 };
335
336 if let Some(size) = marker.inner_marker.size() {
337 let bytes_to_skip = size
338 .checked_mul(len.try_into()?)
339 .ok_or(Error::Value("List is too big".to_string()))?;
340 return advance(bytes_to_skip, r);
341 };
342
343 for _ in 0..len {
344 match marker.inner_marker {
345 InnerMarker::I64
346 | InnerMarker::U64
347 | InnerMarker::F64
348 | InnerMarker::I32
349 | InnerMarker::U32
350 | InnerMarker::I16
351 | InnerMarker::U16
352 | InnerMarker::I8
353 | InnerMarker::U8
354 | InnerMarker::Bool => unreachable!("These types are constant size."),
355 InnerMarker::String => {
356 let len = read_varint(r)?;
357 advance(len, r)?;
358 }
359 InnerMarker::Object => {
360 *skipped_objects += 1;
361 if *skipped_objects > MAX_DEPTH_OF_SKIPPED_OBJECTS {
362 return Err(Error::Format("Depth of skipped objects exceeded maximum"));
363 }
364 read_object::<SkipObject, _>(r, skipped_objects)?;
365 *skipped_objects -= 1;
366 }
367 };
368 }
369 Ok(())
370}
371
372fn advance<B: Buf>(n: usize, b: &mut B) -> Result<()> {
373 checked_read(b, |b: &mut B| b.advance(n), n)
374}