use core::marker;
#[cfg(feature = "alloc")]
use alloc::vec::Vec;
#[cfg(feature = "std")]
use std::io;
use musli::de::Decode;
use musli::en::Encode;
use musli::mode::{DefaultMode, Mode};
use crate::de::WireDecoder;
use crate::en::WireEncoder;
use crate::error::BufferError;
use crate::fixed_bytes::FixedBytes;
use crate::int::{BigEndian, Fixed, FixedUsize, LittleEndian, NetworkEndian, Variable};
use crate::integer_encoding::{WireIntegerEncoding, WireUsizeEncoding};
use crate::reader::{Reader, SliceReader};
use crate::tag::MAX_INLINE_LEN;
use crate::writer::{Buffer, Writer};
pub const DEFAULT: Encoding = Encoding::new();
#[inline]
pub fn encode<W, T>(writer: W, value: &T) -> Result<(), W::Error>
where
W: Writer,
T: ?Sized + Encode<DefaultMode>,
{
DEFAULT.encode(writer, value)
}
#[cfg(feature = "std")]
#[inline]
pub fn to_writer<W, T>(writer: W, value: &T) -> Result<(), io::Error>
where
W: io::Write,
T: ?Sized + Encode<DefaultMode>,
{
DEFAULT.to_writer(writer, value)
}
#[inline]
pub fn to_buffer<T>(value: &T) -> Result<Buffer, BufferError>
where
T: ?Sized + Encode<DefaultMode>,
{
DEFAULT.to_buffer(value)
}
#[cfg(feature = "alloc")]
#[inline]
pub fn to_vec<T>(value: &T) -> Result<Vec<u8>, BufferError>
where
T: ?Sized + Encode<DefaultMode>,
{
DEFAULT.to_vec(value)
}
#[inline]
pub fn to_fixed_bytes<const N: usize, T>(value: &T) -> Result<FixedBytes<N>, BufferError>
where
T: ?Sized + Encode<DefaultMode>,
{
DEFAULT.to_fixed_bytes::<N, _>(value)
}
#[inline]
pub fn decode<'de, R, T>(reader: R) -> Result<T, R::Error>
where
R: Reader<'de>,
T: Decode<'de, DefaultMode>,
{
DEFAULT.decode(reader)
}
#[inline]
pub fn from_slice<'de, T>(bytes: &'de [u8]) -> Result<T, BufferError>
where
T: Decode<'de, DefaultMode>,
{
DEFAULT.from_slice(bytes)
}
pub struct Encoding<M = DefaultMode, I = Variable, L = Variable, const P: usize = MAX_INLINE_LEN>
where
I: WireIntegerEncoding,
L: WireUsizeEncoding,
{
_marker: marker::PhantomData<(M, I, L)>,
}
impl Encoding<DefaultMode, Variable, Variable, MAX_INLINE_LEN> {
pub const fn new() -> Self {
Encoding {
_marker: marker::PhantomData,
}
}
}
impl<M, const P: usize, I, L> Encoding<M, I, L, P>
where
M: Mode,
I: WireIntegerEncoding,
L: WireUsizeEncoding,
{
pub const fn with_mode<T>(self) -> Encoding<T, I, L, P>
where
T: Mode,
{
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_variable_integers(self) -> Encoding<M, Variable, L, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_integers(self) -> Encoding<M, Fixed, L, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_integers_le(self) -> Encoding<M, Fixed<LittleEndian>, L, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_integers_be(self) -> Encoding<M, Fixed<BigEndian>, L, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_integers_ne(self) -> Encoding<M, Fixed<NetworkEndian>, L, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_variable_lengths(self) -> Encoding<M, I, Variable, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_lengths(self) -> Encoding<M, I, FixedUsize<u32>, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_fixed_lengths64(self) -> Encoding<M, I, FixedUsize<u64>, P> {
Encoding {
_marker: marker::PhantomData,
}
}
pub const fn with_max_pack<const N: usize>(self) -> Encoding<M, I, L, N> {
Encoding {
_marker: marker::PhantomData,
}
}
#[inline]
pub fn encode<W, T>(self, mut writer: W, value: &T) -> Result<(), W::Error>
where
W: Writer,
T: ?Sized + Encode<M>,
{
T::encode(value, WireEncoder::<_, I, L, P>::new(&mut writer))
}
#[cfg(feature = "std")]
#[inline]
pub fn to_writer<W, T>(self, write: W, value: &T) -> Result<(), io::Error>
where
W: io::Write,
T: ?Sized + Encode<M>,
{
let mut writer = crate::wrap::wrap(write);
T::encode(value, WireEncoder::<_, I, L, P>::new(&mut writer))
}
#[inline]
pub fn to_buffer<T>(self, value: &T) -> Result<Buffer, BufferError>
where
T: ?Sized + Encode<M>,
{
let mut data = Buffer::new();
T::encode(value, WireEncoder::<_, I, L, P>::new(&mut data))?;
Ok(data)
}
#[cfg(feature = "alloc")]
#[inline]
pub fn to_vec<T>(self, value: &T) -> Result<Vec<u8>, BufferError>
where
T: ?Sized + Encode<M>,
{
Ok(self.to_buffer(value)?.into_vec())
}
#[inline]
pub fn to_fixed_bytes<const N: usize, T>(self, value: &T) -> Result<FixedBytes<N>, BufferError>
where
T: ?Sized + Encode<M>,
{
let mut bytes = FixedBytes::new();
T::encode(value, WireEncoder::<_, I, L, P>::new(&mut bytes))?;
Ok(bytes)
}
#[inline]
pub fn decode<'de, R, T>(self, reader: R) -> Result<T, R::Error>
where
R: Reader<'de>,
T: Decode<'de, M>,
{
T::decode(WireDecoder::<_, I, L>::new(reader.with_position()))
}
#[inline]
pub fn from_slice<'de, T>(self, bytes: &'de [u8]) -> Result<T, BufferError>
where
T: Decode<'de, M>,
{
T::decode(WireDecoder::<_, I, L>::new(
SliceReader::new(bytes).with_position(),
))
}
}
impl<M, I, L, const P: usize> Clone for Encoding<M, I, L, P>
where
M: Mode,
I: WireIntegerEncoding,
L: WireUsizeEncoding,
{
#[inline]
fn clone(&self) -> Self {
Self {
_marker: marker::PhantomData,
}
}
}
impl<M, I, L, const P: usize> Copy for Encoding<M, I, L, P>
where
M: Mode,
I: WireIntegerEncoding,
L: WireUsizeEncoding,
{
}