use core::{convert::TryFrom, ops::Deref};
use crate::{datum_error, DatumAtom, DatumError, DatumOffset, DatumResult, DatumToken};
use serde::{
de::{EnumAccess, IntoDeserializer, MapAccess, SeqAccess, VariantAccess},
forward_to_deserialize_any, Deserializer,
};
use crate::serde::error;
use crate::serde::error::error_from_datum;
pub struct PlainDeserializer<'iterator, B: Default + Deref<Target = str>> {
iterator: &'iterator mut dyn Iterator<Item = DatumResult<DatumToken<B>>>,
hold: Option<DatumToken<B>>,
last_seen_offset: DatumOffset,
}
impl<'iterator, B: Default + Deref<Target = str>> PlainDeserializer<'iterator, B> {
pub fn from_iterator(
iterator: &'iterator mut dyn Iterator<Item = DatumResult<DatumToken<B>>>,
) -> Self {
Self {
iterator,
hold: None,
last_seen_offset: 0,
}
}
pub fn has_next_token(&mut self) -> error::Result<bool> {
if self.hold.is_some() {
Ok(true)
} else {
let res = self.iterator.next();
if let Some(v) = res {
self.hold = Some(v.map_err(error_from_datum)?);
Ok(true)
} else {
Ok(false)
}
}
}
fn next_token(&mut self, eof_error: DatumError) -> error::Result<DatumToken<B>> {
if let Some(token) = self.hold.take() {
Ok(token)
} else {
let res = self.iterator.next();
if let Some(v) = res {
if let Ok(tkn) = &v {
self.last_seen_offset = tkn.offset();
}
v.map_err(error_from_datum)
} else {
Err(error_from_datum(eof_error))
}
}
}
fn expect_list_end(&mut self) -> error::Result<()> {
if let DatumToken::ListEnd(_) = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"unexpected EOF, expected list end"
))? {
Ok(())
} else {
Err(error_from_datum(datum_error!(
BadData,
self.last_seen_offset,
"expected list end and got something else"
)))
}
}
}
struct AccessWrapper<'a, 'iterator, B: Default + Deref<Target = str>>(
&'a mut PlainDeserializer<'iterator, B>,
);
impl<'de, 'a, B: Default + Deref<Target = str>> SeqAccess<'de> for AccessWrapper<'a, '_, B> {
type Error = error::Error;
fn next_element_seed<T: serde::de::DeserializeSeed<'de>>(
&mut self,
seed: T,
) -> Result<Option<T::Value>, Self::Error> {
let token = self.0.next_token(datum_error!(
Interrupted,
self.0.last_seen_offset,
"seq: unexpected EOF, expected next element or list end"
))?;
if let DatumToken::ListEnd(_) = token {
self.0.hold = Some(token);
Ok(None)
} else {
self.0.hold = Some(token);
seed.deserialize(&mut *self.0).map(Some)
}
}
}
impl<'de, 'a, B: Default + Deref<Target = str>> MapAccess<'de> for AccessWrapper<'a, '_, B> {
type Error = error::Error;
fn next_key_seed<K: serde::de::DeserializeSeed<'de>>(
&mut self,
seed: K,
) -> Result<Option<K::Value>, Self::Error> {
self.next_element_seed(seed)
}
fn next_value_seed<V: serde::de::DeserializeSeed<'de>>(
&mut self,
seed: V,
) -> Result<V::Value, Self::Error> {
seed.deserialize(&mut *self.0)
}
}
impl<'de, 'a, B: Default + Deref<Target = str>> EnumAccess<'de> for AccessWrapper<'a, '_, B> {
type Error = error::Error;
type Variant = Self;
fn variant_seed<V: serde::de::DeserializeSeed<'de>>(
self,
seed: V,
) -> Result<(V::Value, Self::Variant), Self::Error> {
Ok((seed.deserialize(&mut *self.0)?, self))
}
}
impl<'de, 'a, B: Default + Deref<Target = str>> VariantAccess<'de> for AccessWrapper<'a, '_, B> {
type Error = error::Error;
fn unit_variant(self) -> Result<(), Self::Error> {
Ok(())
}
fn newtype_variant_seed<T: serde::de::DeserializeSeed<'de>>(
self,
seed: T,
) -> Result<T::Value, Self::Error> {
seed.deserialize(&mut NewtypeVariantDeserializer(self.0))
}
fn tuple_variant<V: serde::de::Visitor<'de>>(
self,
_len: usize,
visitor: V,
) -> Result<V::Value, Self::Error> {
visitor.visit_seq(self)
}
fn struct_variant<V: serde::de::Visitor<'de>>(
self,
_fields: &'static [&'static str],
visitor: V,
) -> Result<V::Value, Self::Error> {
visitor.visit_map(self)
}
}
impl<'de, 'a, B: Default + Deref<Target = str>> Deserializer<'de>
for &'a mut PlainDeserializer<'_, B>
{
type Error = error::Error;
fn deserialize_any<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
let token = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"any: Unexpected EOF, expected value"
))?;
if let DatumToken::ListStart(_) = token {
let res = visitor.visit_seq(AccessWrapper(self))?;
self.expect_list_end()?;
Ok(res)
} else if let DatumToken::ListEnd(_) = token {
Err(error_from_datum(datum_error!(
BadData,
self.last_seen_offset,
"any: unexpected list end"
)))
} else {
match DatumAtom::try_from(token) {
Ok(atom) => match atom {
DatumAtom::String(b) => visitor.visit_str(&b),
DatumAtom::Symbol(b) => visitor.visit_str(&b),
DatumAtom::Integer(i) => visitor.visit_i64(i),
DatumAtom::Float(v) => visitor.visit_f64(v),
DatumAtom::Boolean(v) => visitor.visit_bool(v),
DatumAtom::Nil => visitor.visit_unit(),
},
Err(err) => Err(error_from_datum(err)),
}
}
}
fn deserialize_u64<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
let token = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"u64: unexpected EOF, expected u64"
))?;
match token {
DatumToken::Integer(_, v) => visitor.visit_u64(v as u64),
_ => {
self.hold = Some(token);
self.deserialize_any(visitor)
}
}
}
fn is_human_readable(&self) -> bool {
true
}
fn deserialize_enum<V: serde::de::Visitor<'de>>(
self,
_name: &'static str,
_variants: &'static [&'static str],
visitor: V,
) -> Result<V::Value, Self::Error> {
let token = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"enum: unexpected EOF, expected value"
))?;
match token {
DatumToken::Symbol(_, text) => visitor.visit_enum(text.into_deserializer()),
DatumToken::ListStart(_) => {
let res = visitor.visit_enum(AccessWrapper(self))?;
self.expect_list_end()?;
Ok(res)
}
_ => Err(error_from_datum(datum_error!(
BadData,
self.last_seen_offset,
"enum: expected symbol or list-with-variant, got something else"
))),
}
}
fn deserialize_option<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
let token = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"enum: unexpected EOF, expected value"
))?;
match &token {
DatumToken::SpecialID(_, v) => {
if v.eq_ignore_ascii_case("nil") {
return visitor.visit_none();
}
}
_ => {}
}
self.hold = Some(token);
visitor.visit_some(self)
}
fn deserialize_unit<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
let token = self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"unit: unexpected EOF, expected value"
))?;
match &token {
DatumToken::ListStart(_) => {
self.expect_list_end()?;
visitor.visit_unit()
}
_ => {
self.hold = Some(token);
self.deserialize_any(visitor)
}
}
}
fn deserialize_map<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
match self.next_token(datum_error!(
Interrupted,
self.last_seen_offset,
"map: unexpected EOF, expected list"
))? {
DatumToken::ListStart(_) => {
let res = visitor.visit_map(AccessWrapper(self))?;
self.expect_list_end()?;
Ok(res)
}
_ => Err(error_from_datum(datum_error!(
BadData,
self.last_seen_offset,
"map: expected list, got something else"
))),
}
}
fn deserialize_tuple<V: serde::de::Visitor<'de>>(
self,
_len: usize,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.deserialize_any(visitor)
}
deserializer_invariants!();
forward_to_deserialize_any! {
seq
}
}
struct NewtypeVariantDeserializer<'base, 'iterator, B: Default + Deref<Target = str>>(
&'base mut PlainDeserializer<'iterator, B>,
);
impl<'de, 'a, B: Default + Deref<Target = str>> Deserializer<'de>
for &'a mut NewtypeVariantDeserializer<'_, '_, B>
{
type Error = error::Error;
fn is_human_readable(&self) -> bool {
true
}
fn deserialize_any<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.0.deserialize_any(visitor)
}
fn deserialize_u64<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.0.deserialize_u64(visitor)
}
fn deserialize_option<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.0.deserialize_option(visitor)
}
fn deserialize_unit<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.0.deserialize_unit(visitor)
}
fn deserialize_enum<V: serde::de::Visitor<'de>>(
self,
_name: &'static str,
_variants: &'static [&'static str],
visitor: V,
) -> Result<V::Value, Self::Error> {
visitor.visit_enum(AccessWrapper(&mut self.0))
}
fn deserialize_map<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
visitor.visit_map(AccessWrapper(&mut self.0))
}
fn deserialize_seq<V: serde::de::Visitor<'de>>(
self,
visitor: V,
) -> Result<V::Value, Self::Error> {
visitor.visit_seq(AccessWrapper(&mut self.0))
}
fn deserialize_tuple<V: serde::de::Visitor<'de>>(
self,
_len: usize,
visitor: V,
) -> Result<V::Value, Self::Error> {
self.deserialize_seq(visitor)
}
deserializer_invariants!();
}