tinyklv 0.1.1

The simplest Key-Length-Value (KLV) framework in Rust
Documentation
// --------------------------------------------------
// local
// --------------------------------------------------
pub use super::*;

/// Encodes the value portion of a KLV field to owned stream-type `O`.
///
/// Decode counterpart: [`DecodeValue`](crate::traits::DecodeValue)
///
/// Trait for encoding ***data only*** to owned stream-type `O`, where `O` is an owned stream-type of [`winnow::stream::Stream`], with elements `T`.
///
/// ```text
///                                  This is what is encoded
///                                 vvvvvvvvvvvvvvvvvvvvvvvvvv
/// [ ... key ... | ... length ... | ..... value (self) ..... ]
///                                 ^^^^^^^^^^^^^^^^^^^^^^^^^^
/// ```
///
/// ***This trait IS automatically implemented for structs deriving the [`tinyklv::Klv`](crate::Klv) trait, in which every field has an associated encoder for it's type. Otherwise, this trait CAN be implemented manually.***
///
/// Common stream types include `&[u8]` and `&str`, therefore the return type of encoding is likely an owned value like [`Vec<u8>`] or [`String`].
///
/// For custom encoding functions, ***no need to use this trait***. Instead, please ensure the functions signature matches the following:
///
/// ```rust ignore
/// fn encoder_fn_name(..) -> O;
/// ```
///
/// # Example
///
/// ```rust
/// use tinyklv::Klv;
/// use tinyklv::prelude::*;
/// use tinyklv::traits::EncodeValue;
///
/// struct InnerValue {}
///
/// fn ex01_encoder(input: &InnerValue) -> Vec<u8> {
///     return vec![0x65, 0x66, 0x67, 0x68];
/// }
///
/// fn ex02_encoder(input: &InnerValue) -> Vec<u8> {
///     return String::from("Y2K").into_bytes();
/// }
///
/// impl EncodeValue<Vec<u8>> for InnerValue {
///     fn encode_value(&self) -> Vec<u8> {
///         return String::from("KLV").to_lowercase().into_bytes();
///     }
/// }
///
/// #[derive(Klv)]
/// #[klv(
///     stream = &[u8],
///     sentinel = b"\x00",
///     key(enc = tinyklv::codecs::binary::enc::u8),
///     len(enc = tinyklv::codecs::binary::enc::u8_from_usize),
/// )]
/// #[klv(allow_unimplemented_decode)]
/// struct MyStruct {
///     #[klv(key = 0x07, enc = ex01_encoder)]
///     example_one: InnerValue,
///
///     #[klv(key = 0x0A, enc = ex02_encoder)]
///     example_two: InnerValue,
///
///     #[klv(key = 0x8A, enc = InnerValue::encode_value)]
///     example_three: InnerValue,
/// }
///
/// let my_struct_value_encoded = MyStruct {
///     example_one: InnerValue {},
///     example_two: InnerValue {},
///     example_three: InnerValue {},
/// }.encode_value();
///
/// assert_eq!(my_struct_value_encoded, vec![
///     // example 1
///     0x07,               // example 1 key
///     0x04,               // example 1 length
///                         // example 1 value
///     0x65, 0x66, 0x67, 0x68,
///
///     // example 2
///     0x0A,               // example 2 key
///     0x03,               // example 2 length
///     0x59, 0x32, 0x4B,   // example 2 value
///
///     // example 3
///     0x8A,               // example 3 key
///     0x03,               // example 3 length
///     0x6B, 0x6C, 0x76,   // example 3 value
/// ]);
///
/// let my_struct_encoded = MyStruct {
///     example_one: InnerValue {},
///     example_two: InnerValue {},
///     example_three: InnerValue {},
/// }.encode_frame(); // See: `tinyklv::prelude::EncodeFrame` -> This prepends the key and length
///
/// assert_eq!(my_struct_encoded, vec![
///     0x00,               // sentinel
///     0x10,               // total length
///
///     // example 1
///     0x07,               // example 1 key
///     0x04,               // example 1 length
///                         // example 1 value
///     0x65, 0x66, 0x67, 0x68,
///
///     // example 2
///     0x0A,               // example 2 key
///     0x03,               // example 2 length
///     0x59, 0x32, 0x4B,   // example 2 value
///
///     // example 3
///     0x8A,               // example 3 key
///     0x03,               // example 3 length
///     0x6B, 0x6C, 0x76,   // example 3 value
/// ]);
/// ```
pub trait EncodeValue<O: EncodedOutput> {
    #[must_use = "encoded value is discarded; call `.into_klv(...)` or assign the result"]
    fn encode_value(&self) -> O;
}

/// Trait for prepending encoded data with its key and length.
///
/// ```text
///     This is what is prepended
///  vvvvvvvvvvvvvvvvvvvvvvvvvvvvvv
/// [ ... key ... | ... length ... | ..... value (self) ..... ]
///  ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
/// ```
///
/// ***This trait is automatically implemented for all potential encoded-like datatypes. There is no need to implement it manually.***
///
/// # Example
///
/// ```rust
/// use tinyklv::prelude::*;
/// use tinyklv::traits::EncodeValue;
///
/// struct MyStruct {}
///
/// impl EncodeValue<Vec<u8>> for MyStruct {
///     fn encode_value(&self) -> Vec<u8> {
///         return "a value".as_bytes().to_vec();
///     }
/// }
///
/// let my_struct = MyStruct {};
///
/// let key_len_val_of_my_struct = my_struct.encode_value().into_klv(
///     [0xFF, 0xBB],   // encoded key (must implement into iter)
///     |x: usize|      // length encoder
///         (x as u8).to_be_bytes().to_vec(),
/// );
///
/// assert_eq!(key_len_val_of_my_struct, [
///     0xFF, 0xBB,                     // key
///     0x07,                           // length
///     97, 32, 118, 97, 108, 117, 101, // value
/// ]);
/// ```
///
/// See [`EncodeValue`] for more information.
pub trait IntoKlv<O: EncodedOutput> {
    #[must_use = "the KLV-wrapped output is discarded; assign or extend into a buffer"]
    fn into_klv(self, encoded_key: impl Into<O>, len_encoder: fn(usize) -> O) -> O;
}
/// [`IntoKlv`] implementation for all types O that implement [`EncodedOutput<T>`]
impl<O: EncodedOutput> IntoKlv<O> for O {
    #[inline(always)]
    fn into_klv(self, encoded_key: impl Into<O>, len_encoder: fn(usize) -> O) -> O {
        O::from_iter(
            encoded_key
                .into()
                .into_iter()
                .chain(len_encoder(self.as_ref().len()))
                .chain(self),
        )
    }
}
/// [`IntoKlv`] implementation for all types [`Option<O>`] that implement [`EncodedOutput<T>`]
///
/// `None` produces empty output (field omitted from encoded packet). This is
/// typically the correct behavior for optional KLV fields.
impl<O: EncodedOutput> IntoKlv<O> for Option<O> {
    #[inline(always)]
    fn into_klv(self, encoded_key: impl Into<O>, len_encoder: fn(usize) -> O) -> O {
        match self {
            Some(x) => x.into_klv(encoded_key, len_encoder),
            None => O::from_iter(std::iter::empty::<O::Element>()),
        }
    }
}

/// Full KLV encode pipeline: prepends key and length to [`EncodeValue`] output.
///
/// Decode counterpart: [`DecodeFrame`](crate::traits::DecodeFrame)
///
/// Trait for encoding data to its full key-length-value representation.
///
/// ```text
///                 This is what is encoded
///  vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv
/// [ ... key ... | ... length ... | ..... value (self) ..... ]
///  ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
/// ```
///
/// ***This trait IS automatically implemented for structs deriving the [`tinyklv::Klv`](crate::Klv) trait, in which every field has an associated encoder for it's type. Otherwise, this trait CAN be implemented manually.***
///
/// Since this function and [`EncodeValue`] have the same function signature, this could cause confusion. Therefore, the general workflow for all KLV encoding should be:
///
/// 1. Encode the struct/value
///
/// This can be done by implementing the [`EncodeValue`] trait.
///
/// 2. Use the [`IntoKlv`] function to convert the struct/value into its key-length-value, providing the encoded key/recognition sentinel, alongside the length encoder.
///
/// Then, youre done: now you can produce the key-length-value representation of your struct with the following snippet:
///
/// ```rust
/// use tinyklv::prelude::*;
/// use tinyklv::traits::EncodeValue;
///
/// struct MyStruct {}
///
/// impl EncodeValue<Vec<u8>> for MyStruct {
///     fn encode_value(&self) -> Vec<u8> {
///         return "example".as_bytes().to_vec();
///     }
/// }
///
/// let my_struct = MyStruct {};
///
/// let key_len_val_of_my_struct = my_struct.encode_value().into_klv(
///     [0xFF, 0xBB],   // encoded key (must implement into iter)
///     |x: usize|      // length encoder
///         (x as u8).to_be_bytes().to_vec(),
/// );
///
/// assert_eq!(key_len_val_of_my_struct, [
///     0xFF, 0xBB,                         // key
///     0x07,                               // length
///     101, 120, 97, 109, 112, 108, 101,   // value
/// ]);
/// ```
///
/// However, you can also implement the [`EncodeFrame`] trait instead to combine both of these operations:
///
/// ```rust
/// use tinyklv::prelude::*;
/// use tinyklv::traits::{EncodeValue, EncodeFrame};
///
/// struct MyStruct {}
///
/// impl EncodeValue<Vec<u8>> for MyStruct {
///     fn encode_value(&self) -> Vec<u8> {
///         return "example".as_bytes().to_vec();
///     }
/// }
///
/// impl EncodeFrame<Vec<u8>> for MyStruct {
///     fn encode_frame(&self) -> Vec<u8> {
///         return self.encode_value().into_klv(
///             [0xFF, 0xBB],   // encoded key (must implement into iter)
///             |x: usize|      // length encoder
///                 (x as u8).to_be_bytes().to_vec(),
///         );
///     }
/// }
///
/// let my_struct = MyStruct {};
///
/// let key_len_val_of_my_struct = my_struct.encode_frame();
///
/// assert_eq!(key_len_val_of_my_struct, [
///     0xFF, 0xBB,                         // key
///     0x07,                               // length
///     101, 120, 97, 109, 112, 108, 101,   // value
/// ]);
/// ```
///
/// Furthermore, you can use the [`crate::Klv`] macro to implement both of these for you. (Note that the examples above on a blank struct aren't representative of how the macro works, so a field has been added to the struct):
///
/// ```rust
/// use tinyklv::Klv;
/// use tinyklv::prelude::*;
///
/// fn string_encoder(input: &String) -> Vec<u8> {
///     return input.as_bytes().to_vec();
/// }
///
/// #[derive(Klv)]
/// #[klv(
///     stream = &[u8],
///     sentinel = b"\x00",
///     allow_unimplemented_decode,
///     key(enc = tinyklv::codecs::binary::enc::u8),
///     len(enc = tinyklv::codecs::binary::enc::u8_from_usize),
/// )]  // this implements `tinyklv::prelude::EncodeValue` and `tinyklv::prelude::EncodeFrame`
///     // given a key and length encoder are provided
/// struct MyStruct {
///     #[klv(key = 0xFF, enc = string_encoder)]
///     value: String,
/// }
///
/// // using `tinyklv::Klv` to encode as KLV
/// let mystruct_klv_1 = MyStruct {
///     value: "example".into()
/// }.encode_frame();                                   // `tinyklv::prelude::EncodeFrame` implementation
///
/// // using manual implementation to encode as KLV
/// let mystruct_klv_2 = MyStruct {
///     value: "example".into()
/// }.encode_value().into_klv(
///     [0x00],                                         // recognition sentinel (must implement into iter)
///     tinyklv::codecs::binary::enc::u8_from_usize,    // length encoder
/// );                                                  // this is now equivalent to the macro call
///
/// assert_eq!(mystruct_klv_1, mystruct_klv_2);
/// ```
pub trait EncodeFrame<O: EncodedOutput> {
    #[must_use = "encoded frame is discarded; assign or send the result"]
    fn encode_frame(&self) -> O;
}