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
// SPDX-License-Identifier: Apache-2.0

//! Traits for encoding and decoding.
//!
//! This crate provides generic traits for encoding and decoding to a
//! `std::io::Read` or `std::io::Write` type, respectively.
//!
//! We often need to express that a type can be encoded or decoded. We
//! also need a way to express the type of the encoding or decoding as
//! well as parameters that may be used for that encoding or decoding.
//! This tiny crate solves this problem.
//!
//! # Examples
//!
//! Let's say we want `u8` to be able to be encoded in a (made up) format `Foo`
//! which simply writes the byte without modification. We can express this
//! encoding as follows:
//!
//! ```rust
//! use codicon::*;
//!
//! struct Foo;
//!
//! impl Encoder<Foo> for u8 {
//!     type Error = std::io::Error;
//!
//!     fn encode(&self, mut writer: impl Write, params: Foo) -> std::io::Result<()> {
//!         writer.write_all(std::slice::from_ref(self))?;
//!         Ok(())
//!     }
//! }
//!
//! let mut buf = [0u8; 1];
//! 7u8.encode(&mut buf.as_mut(), Foo).unwrap();
//! assert_eq!(buf[0], 7u8);
//! ```
//!
//! Note that we used a unit struct because the `Foo` encoding doesn't take any
//! options. But if you wanted to specify encoding options, you could just make
//! a type with parameters.
//!
//! Decoding works the same as encoding:
//!
//! ```rust
//! use codicon::*;
//!
//! struct Foo;
//!
//! impl Decoder<Foo> for u8 {
//!     type Error = std::io::Error;
//!
//!     fn decode(mut reader: impl Read, params: Foo) -> std::io::Result<Self> {
//!         let mut byte = 0u8;
//!         reader.read_exact(std::slice::from_mut(&mut byte))?;
//!         Ok(byte)
//!     }
//! }
//!
//! let buf = [7u8; 1];
//! assert_eq!(u8::decode(&mut buf.as_ref(), Foo).unwrap(), 7u8);
//! ```

pub use std::io::{Read, Write};

/// Trait used to express encoding relationships.
pub trait Encoder<T> {
    type Error;

    /// Encodes to the writer with the given parameters.
    fn encode(&self, writer: impl Write, params: T) -> Result<(), Self::Error>;
}

/// Trait used to express decoding relationships.
pub trait Decoder<T>: Sized {
    type Error;

    /// Decodes from the reader with the given parameters.
    fn decode(reader: impl Read, params: T) -> Result<Self, Self::Error>;
}