1// devela::data::codec::encode::impls
2//
3//! General implementations of [`Decodable`] and [`Encodable`].
4//
56use crate::{
7 sf, CStr, Char, Decodable, Encodable, Fmt, FmtArguments, FmtError, FmtResult, FmtWrite,
8 IoError, IoErrorKind, IoRead, IoResult, IoWrite, NoData, Slice,
9};
10crate::_use! {basic::from_utf8}
1112#[cfg(feature = "alloc")]
13crate::items! {
14use crate::{iif, Cow, CString, Box, String, ToOwned, Vec};
1516impl<W: IoWrite> Encodable<W> for Vec<u8> {
17fn encode(&self, writer: &mut W) -> IoResult<usize> {
18 writer.write(self.as_slice()) } }
19impl<R: IoRead> Decodable<R> for Vec<u8> {
20type Output = Vec<u8>;
21fn decode(reader: &mut R) -> IoResult<Self::Output> {
22let mut buf = Vec::new();
23 reader.read_to_end(&mut buf)?;
24Ok(buf) } }
2526impl<W: IoWrite, T: Encodable<W>> Encodable<W> for Box<T> {
27fn encode(&self, writer: &mut W) -> IoResult<usize> {
28self.as_ref().encode(writer) } }
29impl<R: IoRead, T: Decodable<R>> Decodable<R> for Box<T> {
30type Output = Box<T::Output>;
31fn decode(reader: &mut R) -> IoResult<Self::Output> {
32Ok(Box::new(T::decode(reader)?)) } }
3334impl<W: IoWrite> Encodable<W> for String {
35fn encode(&self, writer: &mut W) -> IoResult<usize> {
36 writer.write(self.as_bytes()) } }
37impl<R: IoRead> Decodable<R> for String {
38type Output = String;
39fn decode(reader: &mut R) -> IoResult<Self::Output> {
40let mut buf = Vec::new();
41 reader.read_to_end(&mut buf)?;
42 from_utf8(&buf).map(|s| s.to_owned())
43 .map_err(|_| IoError::new(IoErrorKind::InvalidData, "Invalid UTF-8"))
44 }
45 }
46impl<W: IoWrite> Encodable<W> for CString {
47fn encode(&self, writer: &mut W) -> IoResult<usize> {
48 writer.write(self.as_bytes()) } }
49impl<R: IoRead> Decodable<R> for CString {
50type Output = CString;
51fn decode(reader: &mut R) -> IoResult<Self::Output> {
52let mut buf = Vec::new();
53let mut byte = [0u8; 1];
54loop {
55 reader.read_exact(&mut byte)?;
56iif![byte[0] == 0; break];
57 buf.push(byte[0]);
58 }
59 CString::new(buf).map_err(|_|
60 IoError::new(IoErrorKind::InvalidData, "Invalid C string")) } }
6162// only Encodable (for now?)
63impl<W: IoWrite, T: Clone + Encodable<W>> Encodable<W> for Cow<'_, T> {
64fn encode(&self, writer: &mut W) -> IoResult<usize> {
65self.as_ref().encode(writer) } }
66}
6768/* fmt */
6970#[rustfmt::skip]
71impl<W: IoWrite> Encodable<W> for FmtArguments<'_> {
72fn encode(&self, writer: &mut W) -> IoResult<usize> {
73// Allows to use `Fmt::write` to format `FmtArguments` directly into `IoWrite`.
74struct InlineFmtAdapter<'a, W: IoWrite> {
75 writer: &'a mut W,
76 error: Option<IoError>,
77 total: usize,
78 }
79impl<W: IoWrite> FmtWrite for InlineFmtAdapter<'_, W> {
80fn write_str(&mut self, s: &str) -> FmtResult<()> {
81match self.writer.write(s.as_bytes()) {
82Ok(n) => { self.total += n; Ok(()) }
83Err(e) => { self.error = Some(e); Err(FmtError) }
84 }
85 }
86 }
87let mut adapter = InlineFmtAdapter { writer, error: None, total: 0 };
88if Fmt::write(&mut adapter, *self).is_ok() { Ok(adapter.total) }
89else { Err(adapter.error
90 .unwrap_or_else(|| IoError::new(IoErrorKind::Other, "Formatting failed"))) }
91 }
92}
9394/* primitives: char, byte, bool */
9596impl<W: IoWrite> Encodable<W> for char {
97fn encode(&self, w: &mut W) -> IoResult<usize> {
98let mut buf = [0; 4];
99let s = self.encode_utf8(&mut buf);
100 w.write(s.as_bytes())
101 }
102}
103impl<R: IoRead> Decodable<R> for char {
104type Output = char;
105fn decode(reader: &mut R) -> IoResult<Self::Output> {
106let mut buf = [0u8; 4]; // UTF-8 can be up to 4 bytes
107reader.read_exact(&mut buf[..1])?;
108let len = Char::utf8_len(buf[0]) as usize;
109 reader.read_exact(&mut buf[1..len])?;
110let utf8_str = from_utf8(&buf[..len])
111 .map_err(|_| IoError::new(IoErrorKind::InvalidData, "Invalid UTF-8 sequence"))?;
112 utf8_str
113 .chars()
114 .next()
115 .ok_or_else(|| IoError::new(IoErrorKind::UnexpectedEof, "Empty UTF-8 sequence"))
116 }
117}
118119impl<W: IoWrite> Encodable<W> for u8 {
120fn encode(&self, writer: &mut W) -> IoResult<usize> {
121 writer.write(Slice::from_ref(self))
122 }
123}
124impl<R: IoRead> Decodable<R> for u8 {
125type Output = u8;
126fn decode(reader: &mut R) -> IoResult<u8> {
127let mut buf = [0u8; size_of::<u8>()];
128 reader.read_exact(&mut buf)?;
129Ok(buf[0])
130 }
131}
132133impl<W: IoWrite> Encodable<W> for i8 {
134fn encode(&self, writer: &mut W) -> IoResult<usize> {
135 (*self as u8).encode(writer)
136 }
137}
138impl<R: IoRead> Decodable<R> for i8 {
139type Output = i8;
140fn decode(reader: &mut R) -> IoResult<i8> {
141let mut buf = [0u8; size_of::<i8>()];
142 reader.read_exact(&mut buf)?;
143Ok(buf[0] as i8)
144 }
145}
146147impl<W: IoWrite> Encodable<W> for bool {
148fn encode(&self, writer: &mut W) -> IoResult<usize> {
149 u8::from(*self).encode(writer)
150 }
151}
152impl<R: IoRead> Decodable<R> for bool {
153type Output = bool;
154fn decode(reader: &mut R) -> IoResult<Self::Output> {
155let mut buf = [0u8; 1];
156 reader.read_exact(&mut buf)?;
157match buf[0] {
1580 => Ok(false),
1591 => Ok(true),
160_ => Err(IoError::new(IoErrorKind::InvalidData, "Invalid boolean value")),
161 }
162 }
163}
164165// only Encodable
166/// Allows encoding of `&T` directly, forwarding calls to `T::encode()`.
167impl<T: Encodable<W> + ?Sized, W: IoWrite> Encodable<W> for &T {
168fn encode(&self, writer: &mut W) -> IoResult<usize> {
169 T::encode(self, writer)
170 }
171}
172// only Decodable
173// impl<R: IoRead, T: Decodable<R>> Decodable<IoTake<R>> for T {
174// type Output = T::Output;
175// fn decode(reader: &mut IoTake<R>) -> IoResult<Self::Output> {
176// T::decode(reader)
177// }
178// }
179180/* slices, arrays */
181182sf! {
183impl<W: IoWrite, const SIZE: usize> Encodable<W> for [u8; SIZE] {
184fn encode(&self, writer: &mut W) -> IoResult<usize> {
185 writer.write(self)
186 }
187 }
188impl<R: IoRead, const SIZE: usize> Decodable<R> for [u8; SIZE] {
189type Output = [u8; SIZE];
190fn decode(reader: &mut R) -> IoResult<Self::Output> {
191let mut buf = [0u8; SIZE];
192 reader.read_exact(&mut buf)?;
193Ok(buf)
194 }
195 }
196197impl<W: IoWrite> Encodable<W> for str {
198fn encode(&self, writer: &mut W) -> IoResult<usize> {
199 writer.write(self.as_bytes())
200 }
201 }
202impl<'a> Decodable<&'a mut &[u8]> for &'a str {
203type Output = Self;
204fn decode(reader: &mut &'a mut &[u8]) -> IoResult<Self::Output> {
205// In this specialized case, we assume the reader is already limited
206 // to exactly the bytes for the string.
207let s = from_utf8(reader)
208 .map_err(|_| IoError::new(IoErrorKind::InvalidData, "Invalid UTF-8"))?;
209// Consume the entire slice.
210**reader = &[];
211Ok(s)
212 }
213 }
214215// only Encodable
216impl<W: IoWrite> Encodable<W> for [u8] {
217fn encode(&self, writer: &mut W) -> IoResult<usize> {
218 writer.write(self)
219 }
220 }
221// only Encodable
222impl<W: IoWrite> Encodable<W> for CStr {
223fn encode(&self, writer: &mut W) -> IoResult<usize> {
224 writer.write(self.to_bytes_with_nul())
225 }
226 }
227}
228229/* tuples */
230231// https://users.rust-lang.org/t/macro-to-impl-trait-for-tuple/79165/3
232macro_rules! impl_encodable_for_tuple {
233 ($($T:tt)*) => { $crate::paste! {
234impl<WRITER, A, $($T,)*> Encodable<WRITER> for (A,$($T,)*)
235where
236WRITER: IoWrite,
237 A: Encodable<WRITER>,
238 $($T: Encodable<WRITER>,)*
239 {
240fn encode(&self, writer: &mut WRITER) -> IoResult<usize> {
241let (a, $([<$T:lower>],)*) = self;
242let mut total = 0;
243 total += a.encode(writer)?;
244 $(
245 total += [<$T:lower>].encode(writer)?;
246 )*
247Ok(total)
248 }
249250 }
251 }};
252}
253impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U V W X Y Z);
254impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U V W X Y);
255impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U V W X);
256impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U V W);
257impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U V);
258impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T U);
259impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S T);
260impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R S);
261impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q R);
262impl_encodable_for_tuple!(B C D E F G H I J K L M N O P Q);
263impl_encodable_for_tuple!(B C D E F G H I J K L M N O P);
264impl_encodable_for_tuple!(B C D E F G H I J K L M N O);
265impl_encodable_for_tuple!(B C D E F G H I J K L M N);
266impl_encodable_for_tuple!(B C D E F G H I J K L M);
267impl_encodable_for_tuple!(B C D E F G H I J K L);
268impl_encodable_for_tuple!(B C D E F G H I J K);
269impl_encodable_for_tuple!(B C D E F G H I J);
270impl_encodable_for_tuple!(B C D E F G H I);
271impl_encodable_for_tuple!(B C D E F G H);
272impl_encodable_for_tuple!(B C D E F G);
273impl_encodable_for_tuple!(B C D E F);
274impl_encodable_for_tuple!(B C D E);
275impl_encodable_for_tuple!(B C D);
276impl_encodable_for_tuple!(B C);
277impl_encodable_for_tuple!(B);
278impl_encodable_for_tuple!();
279280/* option, unit */
281282sf! {
283impl<T: Encodable<W>, W: IoWrite> Encodable<W> for Option<T> {
284fn encode(&self, writer: &mut W) -> IoResult<usize> {
285match self { Some(value) => value.encode(writer), None => Ok(0) }
286 }
287 }
288impl<W: IoWrite> Encodable<W> for NoData {
289fn encode(&self, _writer: &mut W) -> IoResult<usize> { Ok(0) }
290 }
291}
292293#[cfg(test)]
294mod tests {
295use super::*;
296297#[test]
298fn option() {
299let mut writer = &mut [0u8; 32] as &mut [u8];
300None::<Option<u8>>.encode(&mut writer).unwrap();
301Some(42u8).encode(&mut writer).unwrap();
302 }
303}