mirror of
https://github.com/serde-rs/serde.git
synced 2025-09-28 13:30:48 +00:00
642 lines
19 KiB
Rust
642 lines
19 KiB
Rust
// Copyright 2017 Serde Developers
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
use serde::de::{self, Deserialize, DeserializeSeed, EnumAccess, IntoDeserializer, MapAccess,
|
|
SeqAccess, VariantAccess, Visitor};
|
|
use serde::de::value::{MapAccessDeserializer, SeqAccessDeserializer};
|
|
|
|
use error::Error;
|
|
use token::Token;
|
|
|
|
#[derive(Debug)]
|
|
pub struct Deserializer<'de> {
|
|
tokens: &'de [Token],
|
|
}
|
|
|
|
macro_rules! assert_next_token {
|
|
($de:expr, $expected:expr) => {
|
|
match $de.next_token_opt() {
|
|
Some(token) if token == $expected => {}
|
|
Some(other) => {
|
|
panic!("expected Token::{} but deserialization wants Token::{}",
|
|
other, $expected)
|
|
}
|
|
None => {
|
|
panic!("end of tokens but deserialization wants Token::{}",
|
|
$expected)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
macro_rules! unexpected {
|
|
($token:expr) => {
|
|
panic!("deserialization did not expect this token: {}", $token)
|
|
}
|
|
}
|
|
|
|
macro_rules! end_of_tokens {
|
|
() => {
|
|
panic!("ran out of tokens to deserialize")
|
|
}
|
|
}
|
|
|
|
impl<'de> Deserializer<'de> {
|
|
pub fn new(tokens: &'de [Token]) -> Self {
|
|
Deserializer { tokens: tokens }
|
|
}
|
|
|
|
fn peek_token_opt(&self) -> Option<Token> {
|
|
self.tokens.first().cloned()
|
|
}
|
|
|
|
fn peek_token(&self) -> Token {
|
|
match self.peek_token_opt() {
|
|
Some(token) => token,
|
|
None => end_of_tokens!(),
|
|
}
|
|
}
|
|
|
|
pub fn next_token_opt(&mut self) -> Option<Token> {
|
|
match self.tokens.split_first() {
|
|
Some((&first, rest)) => {
|
|
self.tokens = rest;
|
|
Some(first)
|
|
}
|
|
None => None,
|
|
}
|
|
}
|
|
|
|
fn next_token(&mut self) -> Token {
|
|
match self.tokens.split_first() {
|
|
Some((&first, rest)) => {
|
|
self.tokens = rest;
|
|
first
|
|
}
|
|
None => end_of_tokens!(),
|
|
}
|
|
}
|
|
|
|
pub fn remaining(&self) -> usize {
|
|
self.tokens.len()
|
|
}
|
|
|
|
fn visit_seq<V>(
|
|
&mut self,
|
|
len: Option<usize>,
|
|
end: Token,
|
|
visitor: V,
|
|
) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
let value = try!(
|
|
visitor.visit_seq(
|
|
DeserializerSeqVisitor {
|
|
de: self,
|
|
len: len,
|
|
end: end.clone(),
|
|
},
|
|
)
|
|
);
|
|
assert_next_token!(self, end);
|
|
Ok(value)
|
|
}
|
|
|
|
fn visit_map<V>(
|
|
&mut self,
|
|
len: Option<usize>,
|
|
end: Token,
|
|
visitor: V,
|
|
) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
let value = try!(
|
|
visitor.visit_map(
|
|
DeserializerMapVisitor {
|
|
de: self,
|
|
len: len,
|
|
end: end.clone(),
|
|
},
|
|
)
|
|
);
|
|
assert_next_token!(self, end);
|
|
Ok(value)
|
|
}
|
|
}
|
|
|
|
impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|
type Error = Error;
|
|
|
|
forward_to_deserialize_any! {
|
|
bool i8 i16 i32 i64 u8 u16 u32 u64 f32 f64 char str string bytes
|
|
byte_buf unit seq map identifier ignored_any
|
|
}
|
|
|
|
fn deserialize_any<V>(self, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
let token = self.next_token();
|
|
match token {
|
|
Token::Bool(v) => visitor.visit_bool(v),
|
|
Token::I8(v) => visitor.visit_i8(v),
|
|
Token::I16(v) => visitor.visit_i16(v),
|
|
Token::I32(v) => visitor.visit_i32(v),
|
|
Token::I64(v) => visitor.visit_i64(v),
|
|
Token::U8(v) => visitor.visit_u8(v),
|
|
Token::U16(v) => visitor.visit_u16(v),
|
|
Token::U32(v) => visitor.visit_u32(v),
|
|
Token::U64(v) => visitor.visit_u64(v),
|
|
Token::F32(v) => visitor.visit_f32(v),
|
|
Token::F64(v) => visitor.visit_f64(v),
|
|
Token::Char(v) => visitor.visit_char(v),
|
|
Token::Str(v) => visitor.visit_str(v),
|
|
Token::BorrowedStr(v) => visitor.visit_borrowed_str(v),
|
|
Token::String(v) => visitor.visit_string(v.to_owned()),
|
|
Token::Bytes(v) => visitor.visit_bytes(v),
|
|
Token::BorrowedBytes(v) => visitor.visit_borrowed_bytes(v),
|
|
Token::ByteBuf(v) => visitor.visit_byte_buf(v.to_vec()),
|
|
Token::None => visitor.visit_none(),
|
|
Token::Some => visitor.visit_some(self),
|
|
Token::Unit => visitor.visit_unit(),
|
|
Token::UnitStruct(_name) => visitor.visit_unit(),
|
|
Token::NewtypeStruct(_name) => visitor.visit_newtype_struct(self),
|
|
Token::Seq(len) => self.visit_seq(len, Token::SeqEnd, visitor),
|
|
Token::Tuple(len) => self.visit_seq(Some(len), Token::TupleEnd, visitor),
|
|
Token::TupleStruct(_, len) => self.visit_seq(Some(len), Token::TupleStructEnd, visitor),
|
|
Token::Map(len) => self.visit_map(len, Token::MapEnd, visitor),
|
|
Token::Struct(_, len) => self.visit_map(Some(len), Token::StructEnd, visitor),
|
|
Token::Enum(_) => {
|
|
let variant = self.next_token();
|
|
let next = self.peek_token();
|
|
match (variant, next) {
|
|
(Token::Str(variant), Token::Unit) => {
|
|
self.next_token();
|
|
visitor.visit_str(variant)
|
|
}
|
|
(Token::Bytes(variant), Token::Unit) => {
|
|
self.next_token();
|
|
visitor.visit_bytes(variant)
|
|
}
|
|
(Token::U32(variant), Token::Unit) => {
|
|
self.next_token();
|
|
visitor.visit_u32(variant)
|
|
}
|
|
(variant, Token::Unit) => unexpected!(variant),
|
|
(variant, _) => {
|
|
visitor.visit_map(EnumMapVisitor::new(self, variant, EnumFormat::Any))
|
|
}
|
|
}
|
|
}
|
|
Token::UnitVariant(_, variant) => visitor.visit_str(variant),
|
|
Token::NewtypeVariant(_, variant) => {
|
|
visitor.visit_map(EnumMapVisitor::new(self, Token::Str(variant), EnumFormat::Any),)
|
|
}
|
|
Token::TupleVariant(_, variant, _) => {
|
|
visitor.visit_map(EnumMapVisitor::new(self, Token::Str(variant), EnumFormat::Seq),)
|
|
}
|
|
Token::StructVariant(_, variant, _) => {
|
|
visitor.visit_map(EnumMapVisitor::new(self, Token::Str(variant), EnumFormat::Map),)
|
|
}
|
|
Token::SeqEnd | Token::TupleEnd | Token::TupleStructEnd | Token::MapEnd |
|
|
Token::StructEnd | Token::TupleVariantEnd | Token::StructVariantEnd => {
|
|
unexpected!(token);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::Unit |
|
|
Token::None => {
|
|
self.next_token();
|
|
visitor.visit_none()
|
|
}
|
|
Token::Some => {
|
|
self.next_token();
|
|
visitor.visit_some(self)
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
|
|
fn deserialize_enum<V>(
|
|
self,
|
|
name: &'static str,
|
|
_variants: &'static [&'static str],
|
|
visitor: V,
|
|
) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::Enum(n) if name == n => {
|
|
self.next_token();
|
|
|
|
visitor.visit_enum(DeserializerEnumVisitor { de: self })
|
|
}
|
|
Token::UnitVariant(n, _) |
|
|
Token::NewtypeVariant(n, _) |
|
|
Token::TupleVariant(n, _, _) |
|
|
Token::StructVariant(n, _, _) if name == n => {
|
|
visitor.visit_enum(DeserializerEnumVisitor { de: self })
|
|
}
|
|
_ => {
|
|
unexpected!(self.next_token());
|
|
}
|
|
}
|
|
}
|
|
|
|
fn deserialize_unit_struct<V>(self, name: &'static str, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::UnitStruct(_) => {
|
|
assert_next_token!(self, Token::UnitStruct(name));
|
|
visitor.visit_unit()
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
|
|
fn deserialize_newtype_struct<V>(self, name: &'static str, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::NewtypeStruct(_) => {
|
|
assert_next_token!(self, Token::NewtypeStruct(name));
|
|
visitor.visit_newtype_struct(self)
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
|
|
fn deserialize_tuple<V>(self, len: usize, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::Unit |
|
|
Token::UnitStruct(_) => {
|
|
self.next_token();
|
|
visitor.visit_unit()
|
|
}
|
|
Token::Seq(_) => {
|
|
self.next_token();
|
|
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
|
}
|
|
Token::Tuple(_) => {
|
|
self.next_token();
|
|
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
|
}
|
|
Token::TupleStruct(_, _) => {
|
|
self.next_token();
|
|
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
|
|
fn deserialize_tuple_struct<V>(
|
|
self,
|
|
name: &'static str,
|
|
len: usize,
|
|
visitor: V,
|
|
) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::Unit => {
|
|
self.next_token();
|
|
visitor.visit_unit()
|
|
}
|
|
Token::UnitStruct(_) => {
|
|
assert_next_token!(self, Token::UnitStruct(name));
|
|
visitor.visit_unit()
|
|
}
|
|
Token::Seq(_) => {
|
|
self.next_token();
|
|
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
|
}
|
|
Token::Tuple(_) => {
|
|
self.next_token();
|
|
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
|
}
|
|
Token::TupleStruct(_, n) => {
|
|
assert_next_token!(self, Token::TupleStruct(name, n));
|
|
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
|
|
fn deserialize_struct<V>(
|
|
self,
|
|
name: &'static str,
|
|
fields: &'static [&'static str],
|
|
visitor: V,
|
|
) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.peek_token() {
|
|
Token::Struct(_, n) => {
|
|
assert_next_token!(self, Token::Struct(name, n));
|
|
self.visit_map(Some(fields.len()), Token::StructEnd, visitor)
|
|
}
|
|
Token::Map(_) => {
|
|
self.next_token();
|
|
self.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
|
}
|
|
_ => self.deserialize_any(visitor),
|
|
}
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
struct DeserializerSeqVisitor<'a, 'de: 'a> {
|
|
de: &'a mut Deserializer<'de>,
|
|
len: Option<usize>,
|
|
end: Token,
|
|
}
|
|
|
|
impl<'de, 'a> SeqAccess<'de> for DeserializerSeqVisitor<'a, 'de> {
|
|
type Error = Error;
|
|
|
|
fn next_element_seed<T>(&mut self, seed: T) -> Result<Option<T::Value>, Error>
|
|
where
|
|
T: DeserializeSeed<'de>,
|
|
{
|
|
if self.de.peek_token_opt() == Some(self.end) {
|
|
return Ok(None);
|
|
}
|
|
self.len = self.len.map(|len| len.saturating_sub(1));
|
|
seed.deserialize(&mut *self.de).map(Some)
|
|
}
|
|
|
|
fn size_hint(&self) -> Option<usize> {
|
|
self.len
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
struct DeserializerMapVisitor<'a, 'de: 'a> {
|
|
de: &'a mut Deserializer<'de>,
|
|
len: Option<usize>,
|
|
end: Token,
|
|
}
|
|
|
|
impl<'de, 'a> MapAccess<'de> for DeserializerMapVisitor<'a, 'de> {
|
|
type Error = Error;
|
|
|
|
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Error>
|
|
where
|
|
K: DeserializeSeed<'de>,
|
|
{
|
|
if self.de.peek_token_opt() == Some(self.end) {
|
|
return Ok(None);
|
|
}
|
|
self.len = self.len.map(|len| len.saturating_sub(1));
|
|
seed.deserialize(&mut *self.de).map(Some)
|
|
}
|
|
|
|
fn next_value_seed<V>(&mut self, seed: V) -> Result<V::Value, Error>
|
|
where
|
|
V: DeserializeSeed<'de>,
|
|
{
|
|
seed.deserialize(&mut *self.de)
|
|
}
|
|
|
|
fn size_hint(&self) -> Option<usize> {
|
|
self.len
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
struct DeserializerEnumVisitor<'a, 'de: 'a> {
|
|
de: &'a mut Deserializer<'de>,
|
|
}
|
|
|
|
impl<'de, 'a> EnumAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|
type Error = Error;
|
|
type Variant = Self;
|
|
|
|
fn variant_seed<V>(self, seed: V) -> Result<(V::Value, Self), Error>
|
|
where
|
|
V: DeserializeSeed<'de>,
|
|
{
|
|
match self.de.peek_token() {
|
|
Token::UnitVariant(_, v) |
|
|
Token::NewtypeVariant(_, v) |
|
|
Token::TupleVariant(_, v, _) |
|
|
Token::StructVariant(_, v, _) => {
|
|
let de = v.into_deserializer();
|
|
let value = try!(seed.deserialize(de));
|
|
Ok((value, self))
|
|
}
|
|
_ => {
|
|
let value = try!(seed.deserialize(&mut *self.de));
|
|
Ok((value, self))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'de, 'a> VariantAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|
type Error = Error;
|
|
|
|
fn unit_variant(self) -> Result<(), Error> {
|
|
match self.de.peek_token() {
|
|
Token::UnitVariant(_, _) => {
|
|
self.de.next_token();
|
|
Ok(())
|
|
}
|
|
_ => Deserialize::deserialize(self.de),
|
|
}
|
|
}
|
|
|
|
fn newtype_variant_seed<T>(self, seed: T) -> Result<T::Value, Self::Error>
|
|
where
|
|
T: DeserializeSeed<'de>,
|
|
{
|
|
match self.de.peek_token() {
|
|
Token::NewtypeVariant(_, _) => {
|
|
self.de.next_token();
|
|
seed.deserialize(self.de)
|
|
}
|
|
_ => seed.deserialize(self.de),
|
|
}
|
|
}
|
|
|
|
fn tuple_variant<V>(self, len: usize, visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.de.peek_token() {
|
|
Token::TupleVariant(_, _, enum_len) => {
|
|
let token = self.de.next_token();
|
|
|
|
if len == enum_len {
|
|
self.de
|
|
.visit_seq(Some(len), Token::TupleVariantEnd, visitor)
|
|
} else {
|
|
unexpected!(token);
|
|
}
|
|
}
|
|
Token::Seq(Some(enum_len)) => {
|
|
let token = self.de.next_token();
|
|
|
|
if len == enum_len {
|
|
self.de.visit_seq(Some(len), Token::SeqEnd, visitor)
|
|
} else {
|
|
unexpected!(token);
|
|
}
|
|
}
|
|
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
|
}
|
|
}
|
|
|
|
fn struct_variant<V>(self, fields: &'static [&'static str], visitor: V) -> Result<V::Value, Error>
|
|
where
|
|
V: Visitor<'de>,
|
|
{
|
|
match self.de.peek_token() {
|
|
Token::StructVariant(_, _, enum_len) => {
|
|
let token = self.de.next_token();
|
|
|
|
if fields.len() == enum_len {
|
|
self.de
|
|
.visit_map(Some(fields.len()), Token::StructVariantEnd, visitor)
|
|
} else {
|
|
unexpected!(token);
|
|
}
|
|
}
|
|
Token::Map(Some(enum_len)) => {
|
|
let token = self.de.next_token();
|
|
|
|
if fields.len() == enum_len {
|
|
self.de
|
|
.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
|
} else {
|
|
unexpected!(token);
|
|
}
|
|
}
|
|
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
|
}
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
struct EnumMapVisitor<'a, 'de: 'a> {
|
|
de: &'a mut Deserializer<'de>,
|
|
variant: Option<Token>,
|
|
format: EnumFormat,
|
|
}
|
|
|
|
enum EnumFormat {
|
|
Seq,
|
|
Map,
|
|
Any,
|
|
}
|
|
|
|
impl<'a, 'de> EnumMapVisitor<'a, 'de> {
|
|
fn new(de: &'a mut Deserializer<'de>, variant: Token, format: EnumFormat) -> Self {
|
|
EnumMapVisitor {
|
|
de: de,
|
|
variant: Some(variant),
|
|
format: format,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'de, 'a> MapAccess<'de> for EnumMapVisitor<'a, 'de> {
|
|
type Error = Error;
|
|
|
|
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Error>
|
|
where
|
|
K: DeserializeSeed<'de>,
|
|
{
|
|
match self.variant.take() {
|
|
Some(Token::Str(variant)) => seed.deserialize(variant.into_deserializer()).map(Some),
|
|
Some(Token::Bytes(variant)) => {
|
|
seed.deserialize(BytesDeserializer { value: variant })
|
|
.map(Some)
|
|
}
|
|
Some(Token::U32(variant)) => seed.deserialize(variant.into_deserializer()).map(Some),
|
|
Some(other) => unexpected!(other),
|
|
None => Ok(None),
|
|
}
|
|
}
|
|
|
|
fn next_value_seed<V>(&mut self, seed: V) -> Result<V::Value, Error>
|
|
where
|
|
V: DeserializeSeed<'de>,
|
|
{
|
|
match self.format {
|
|
EnumFormat::Seq => {
|
|
let value = {
|
|
let visitor = DeserializerSeqVisitor {
|
|
de: self.de,
|
|
len: None,
|
|
end: Token::TupleVariantEnd,
|
|
};
|
|
try!(seed.deserialize(SeqAccessDeserializer::new(visitor)))
|
|
};
|
|
assert_next_token!(self.de, Token::TupleVariantEnd);
|
|
Ok(value)
|
|
}
|
|
EnumFormat::Map => {
|
|
let value = {
|
|
let visitor = DeserializerMapVisitor {
|
|
de: self.de,
|
|
len: None,
|
|
end: Token::StructVariantEnd,
|
|
};
|
|
try!(seed.deserialize(MapAccessDeserializer::new(visitor)))
|
|
};
|
|
assert_next_token!(self.de, Token::StructVariantEnd);
|
|
Ok(value)
|
|
}
|
|
EnumFormat::Any => seed.deserialize(&mut *self.de),
|
|
}
|
|
}
|
|
}
|
|
|
|
struct BytesDeserializer {
|
|
value: &'static [u8],
|
|
}
|
|
|
|
impl<'de> de::Deserializer<'de> for BytesDeserializer {
|
|
type Error = Error;
|
|
|
|
fn deserialize_any<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
|
where
|
|
V: de::Visitor<'de>,
|
|
{
|
|
visitor.visit_bytes(self.value)
|
|
}
|
|
|
|
forward_to_deserialize_any! {
|
|
bool i8 i16 i32 i64 u8 u16 u32 u64 f32 f64 char str string bytes
|
|
byte_buf option unit unit_struct newtype_struct seq tuple tuple_struct
|
|
map struct enum identifier ignored_any
|
|
}
|
|
}
|