use crate::buffer::{BufReader, BufWriter, CountWriter};
use crate::de::{BasicSmallVecVisitor, Deserialize, DeserializeSeed, Deserializer as _};
use crate::ser::Serialize;
use crate::Typespace;
use smallvec::SmallVec;
pub mod de;
pub mod eq;
pub mod ser;
pub use de::Deserializer;
pub use ser::Serializer;
pub use crate::buffer::DecodeError;
pub fn to_writer<W: BufWriter, T: Serialize + ?Sized>(w: &mut W, value: &T) -> Result<(), ser::BsatnError> {
value.serialize(Serializer::new(w))
}
pub fn to_vec<T: Serialize + ?Sized>(value: &T) -> Result<Vec<u8>, ser::BsatnError> {
let mut v = Vec::new();
to_writer(&mut v, value)?;
Ok(v)
}
pub fn to_len<T: Serialize + ?Sized>(value: &T) -> Result<usize, ser::BsatnError> {
let mut writer = CountWriter::default();
to_writer(&mut writer, value)?;
Ok(writer.finish())
}
pub fn from_reader<'de, T: Deserialize<'de>>(reader: &mut impl BufReader<'de>) -> Result<T, DecodeError> {
T::deserialize(Deserializer::new(reader))
}
pub fn from_slice<'de, T: Deserialize<'de>>(bytes: &'de [u8]) -> Result<T, DecodeError> {
from_reader(&mut &*bytes)
}
macro_rules! codec_funcs {
($ty:ty) => {
impl $ty {
pub fn decode<'a>(bytes: &mut impl BufReader<'a>) -> Result<Self, DecodeError> {
from_reader(bytes)
}
pub fn encode(&self, bytes: &mut impl BufWriter) {
to_writer(bytes, self).unwrap()
}
}
};
(val: $ty:ty) => {
impl $ty {
pub fn decode<'a>(
ty: &<Self as crate::Value>::Type,
bytes: &mut impl BufReader<'a>,
) -> Result<Self, DecodeError> {
crate::WithTypespace::new(&Typespace::new(Vec::new()), ty).deserialize(Deserializer::new(bytes))
}
pub fn decode_smallvec<'a>(
ty: &<Self as crate::Value>::Type,
bytes: &mut impl BufReader<'a>,
) -> Result<SmallVec<[Self; 1]>, DecodeError> {
Deserializer::new(bytes).deserialize_array_seed(
BasicSmallVecVisitor,
crate::WithTypespace::new(&Typespace::new(Vec::new()), ty),
)
}
pub fn encode(&self, bytes: &mut impl BufWriter) {
to_writer(bytes, self).unwrap()
}
}
};
}
codec_funcs!(crate::AlgebraicType);
codec_funcs!(crate::ProductType);
codec_funcs!(crate::SumType);
codec_funcs!(crate::ProductTypeElement);
codec_funcs!(crate::SumTypeVariant);
codec_funcs!(val: crate::AlgebraicValue);
codec_funcs!(val: crate::ProductValue);
codec_funcs!(val: crate::SumValue);
#[cfg(test)]
mod tests {
use super::to_vec;
use crate::proptest::generate_typed_value;
use crate::{meta_type::MetaType, AlgebraicType, AlgebraicValue};
use proptest::prelude::*;
use proptest::proptest;
#[test]
fn type_to_binary_equivalent() {
check_type(&AlgebraicType::meta_type());
}
#[track_caller]
fn check_type(ty: &AlgebraicType) {
let mut through_value = Vec::new();
ty.as_value().encode(&mut through_value);
let mut direct = Vec::new();
ty.encode(&mut direct);
assert_eq!(direct, through_value);
}
proptest! {
#[test]
fn bsatn_enc_de_roundtrips((ty, val) in generate_typed_value()) {
let bytes = to_vec(&val).unwrap();
let val_decoded = AlgebraicValue::decode(&ty, &mut &bytes[..]).unwrap();
prop_assert_eq!(val, val_decoded);
}
}
}