// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use std::hash::Hash; use std::num; use std::collections::{HashMap, TreeMap}; #[deriving(Clone, PartialEq, Show)] pub enum Token { Null, Bool(bool), Int(int), I8(i8), I16(i16), I32(i32), I64(i64), Uint(uint), U8(u8), U16(u16), U32(u32), U64(u64), F32(f32), F64(f64), Char(char), Str(&'static str), String(String), Option(bool), TupleStart(uint), StructStart(&'static str, uint), EnumStart(&'static str, &'static str, uint), SeqStart(uint), MapStart(uint), End, } macro_rules! to_result { ($expr:expr, $err:expr) => { match $expr { Some(value) => Ok(value), None => $err, } } } pub trait Deserializer: Iterator> { fn end_of_stream_error(&self) -> Result; fn syntax_error(&self) -> Result; #[inline] fn expect_token(&mut self) -> Result { match self.next() { Some(Ok(token)) => Ok(token), Some(Err(err)) => Err(err), None => self.end_of_stream_error(), } } #[inline] fn expect_null(&mut self, token: Token) -> Result<(), E> { match token { Null => Ok(()), TupleStart(_) => { match try!(self.expect_token()) { End => Ok(()), _ => self.syntax_error(), } } _ => self.syntax_error(), } } #[inline] fn expect_bool(&mut self, token: Token) -> Result { match token { Bool(value) => Ok(value), _ => self.syntax_error(), } } #[inline] fn expect_num(&mut self, token: Token) -> Result { match token { Int(x) => to_result!(num::cast(x), self.syntax_error()), I8(x) => to_result!(num::cast(x), self.syntax_error()), I16(x) => to_result!(num::cast(x), self.syntax_error()), I32(x) => to_result!(num::cast(x), self.syntax_error()), I64(x) => to_result!(num::cast(x), self.syntax_error()), Uint(x) => to_result!(num::cast(x), self.syntax_error()), U8(x) => to_result!(num::cast(x), self.syntax_error()), U16(x) => to_result!(num::cast(x), self.syntax_error()), U32(x) => to_result!(num::cast(x), self.syntax_error()), U64(x) => to_result!(num::cast(x), self.syntax_error()), F32(x) => to_result!(num::cast(x), self.syntax_error()), F64(x) => to_result!(num::cast(x), self.syntax_error()), _ => self.syntax_error(), } } #[inline] fn expect_char(&mut self, token: Token) -> Result { match token { Char(value) => Ok(value), _ => self.syntax_error(), } } #[inline] fn expect_str(&mut self, token: Token) -> Result<&'static str, E> { match token { Str(value) => Ok(value), _ => self.syntax_error(), } } #[inline] fn expect_string(&mut self, token: Token) -> Result { match token { Str(value) => Ok(value.to_string()), String(value) => Ok(value), _ => self.syntax_error(), } } #[inline] fn expect_option< T: Deserializable >(&mut self, token: Token) -> Result, E> { match token { Option(false) => Ok(None), Option(true) => { let value: T = try!(Deserializable::deserialize(self)); Ok(Some(value)) } _ => self.syntax_error(), } } #[inline] fn expect_tuple_start(&mut self, token: Token) -> Result { match token { TupleStart(len) => Ok(len), _ => self.syntax_error(), } } #[inline] fn expect_tuple_end(&mut self) -> Result<(), E> { match try!(self.expect_token()) { End => Ok(()), _ => self.syntax_error(), } } #[inline] fn expect_struct_start(&mut self, token: Token, name: &str) -> Result<(), E> { match token { StructStart(n, _) => { if name == n { Ok(()) } else { self.syntax_error() } } _ => self.syntax_error(), } } #[inline] fn expect_struct_field< T: Deserializable >(&mut self, name: &str) -> Result { match try!(self.expect_token()) { Str(n) => { if name != n { return self.syntax_error(); } } String(n) => { if name != n.as_slice() { return self.syntax_error(); } } _ => { return self.syntax_error(); } } Deserializable::deserialize(self) } #[inline] fn expect_struct_end(&mut self) -> Result<(), E> { match try!(self.expect_token()) { End => Ok(()), _ => self.syntax_error(), } } #[inline] fn expect_enum_start(&mut self, token: Token, name: &str, variants: &[&str]) -> Result { match token { EnumStart(n, v, _) => { if name == n { match variants.iter().position(|variant| *variant == v) { Some(position) => Ok(position), None => self.syntax_error(), } } else { self.syntax_error() } } _ => self.syntax_error(), } } #[inline] fn expect_enum_end(&mut self) -> Result<(), E> { match try!(self.expect_token()) { End => Ok(()), _ => self.syntax_error(), } } /* #[inline] fn expect_collection< T: Deserializable, C: FromIterator >(&mut self, token: Token) -> Result { // By default we don't care what our source input was. We can take // anything that's a Collection. We'll error out later if the types // are wrong. let len = match token { TupleStart(len) => len, SeqStart(len) => len, MapStart(len) => len, _ => { return self.syntax_error(); } }; expect_rest_of_collection(self, len) } */ #[inline] fn expect_seq_start(&mut self, token: Token) -> Result { match token { SeqStart(len) => Ok(len), _ => self.syntax_error(), } } #[inline] fn expect_map_start(&mut self, token: Token) -> Result { match token { MapStart(len) => Ok(len), _ => self.syntax_error(), } } } ////////////////////////////////////////////////////////////////////////////// /* // FIXME: https://github.com/mozilla/rust/issues/11751 #[inline] fn expect_rest_of_collection< E, D: Deserializer, T: Deserializable, C: FromIterator >(d: &mut D, len: uint) -> Result { let iter = d.by_ref().batch(|d| { let d = d.iter(); match try!(d.expect_token()) { End => None, token => { let value: Result = Deserializable::deserialize_token(d, token); Some(value) } } }); result::collect_with_capacity(iter, len) } */ ////////////////////////////////////////////////////////////////////////////// pub trait Deserializable { #[inline] fn deserialize< D: Deserializer, E >(d: &mut D) -> Result { let token = try!(d.expect_token()); Deserializable::deserialize_token(d, token) } fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result; } ////////////////////////////////////////////////////////////////////////////// macro_rules! impl_deserializable { ($ty:ty, $method:ident) => { impl Deserializable for $ty { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result<$ty, E> { d.$method(token) } } } } impl_deserializable!(bool, expect_bool) impl_deserializable!(int, expect_num) impl_deserializable!(i8, expect_num) impl_deserializable!(i16, expect_num) impl_deserializable!(i32, expect_num) impl_deserializable!(i64, expect_num) impl_deserializable!(uint, expect_num) impl_deserializable!(u8, expect_num) impl_deserializable!(u16, expect_num) impl_deserializable!(u32, expect_num) impl_deserializable!(u64, expect_num) impl_deserializable!(f32, expect_num) impl_deserializable!(f64, expect_num) impl_deserializable!(char, expect_char) impl_deserializable!(&'static str, expect_str) impl_deserializable!(String, expect_string) ////////////////////////////////////////////////////////////////////////////// impl Deserializable for Option { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result, E> { d.expect_option(token) } } ////////////////////////////////////////////////////////////////////////////// macro_rules! deserialize_seq { ($seq:expr) => { { loop { match d.next() { Some(Ok(End)) => { break; } Some(Ok(token)) => { let v = try!(Deserializable::deserialize_token(d, token)); $seq.push(v) } Some(Err(err)) => { return Err(err); } None => { return d.end_of_stream_error(); } } } Ok($seq) } } } impl Deserializable for Vec { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result, E> { let len = try!(d.expect_seq_start(token)); let mut value = Vec::with_capacity(len); deserialize_seq!(value) } } ////////////////////////////////////////////////////////////////////////////// macro_rules! deserialize_map { ($seq:expr) => { { loop { match d.next() { Some(Ok(End)) => { break; } Some(Ok(token)) => { let k = try!(Deserializable::deserialize_token(d, token)); let v = try!(Deserializable::deserialize(d)); $seq.insert(k, v); } Some(Err(err)) => { return Err(err); } None => { return d.end_of_stream_error(); } } } Ok($seq) } } } impl< K: Deserializable + Eq + Hash, V: Deserializable > Deserializable for HashMap { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result, E> { let len = try!(d.expect_map_start(token)); let mut value = HashMap::with_capacity(len); deserialize_map!(value) } } impl< K: Deserializable + Ord, V: Deserializable > Deserializable for TreeMap { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result, E> { let _len = try!(d.expect_map_start(token)); let mut value = TreeMap::new(); deserialize_map!(value) } } ////////////////////////////////////////////////////////////////////////////// macro_rules! peel { ($name:ident, $($other:ident,)*) => (impl_deserialize_tuple!($($other,)*)) } macro_rules! impl_deserialize_tuple { () => { impl Deserializable for () { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result<(), E> { d.expect_null(token) } } }; ( $($name:ident,)+ ) => { impl< $($name: Deserializable),* > Deserializable for ($($name,)*) { #[inline] #[allow(uppercase_variables)] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result<($($name,)*), E> { try!(d.expect_tuple_start(token)); let result = ($({ let $name = try!(Deserializable::deserialize(d)); $name },)*); try!(d.expect_tuple_end()); Ok(result) } } peel!($($name,)*) } } impl_deserialize_tuple! { T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, } ////////////////////////////////////////////////////////////////////////////// /// Helper struct that will ignore tokens while taking in consideration /// recursive structures. pub struct IgnoreTokens; impl Deserializable for IgnoreTokens { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result { match token { Option(true) => { Deserializable::deserialize(d) } EnumStart(_, _, _) => { loop { match try!(d.expect_token()) { End => { return Ok(IgnoreTokens); } token => { let _: IgnoreTokens = try!(Deserializable::deserialize_token(d, token)); } } } } StructStart(_, _) => { loop { match try!(d.expect_token()) { End => { return Ok(IgnoreTokens); } Str(_) | String(_) => { let _: IgnoreTokens = try!(Deserializable::deserialize(d)); } _token => { return d.syntax_error(); } } } } TupleStart(_) => { loop { match try!(d.expect_token()) { End => { return Ok(IgnoreTokens); } token => { let _: IgnoreTokens = try!(Deserializable::deserialize_token(d, token)); } } } } SeqStart(_) => { loop { match try!(d.expect_token()) { End => { return Ok(IgnoreTokens); } token => { let _: IgnoreTokens = try!(Deserializable::deserialize_token(d, token)); } } } } MapStart(_) => { loop { match try!(d.expect_token()) { End => { return Ok(IgnoreTokens); } token => { let _: IgnoreTokens = try!(Deserializable::deserialize_token(d, token)); let _: IgnoreTokens = try!(Deserializable::deserialize(d)); } } } } End => d.syntax_error(), _ => Ok(IgnoreTokens), } } } ////////////////////////////////////////////////////////////////////////////// /// Helper struct that will gather tokens while taking in consideration /// recursive structures. pub struct GatherTokens { tokens: Vec, } impl GatherTokens { #[inline] pub fn unwrap(self) -> Vec { self.tokens } #[inline] fn gather, E>(&mut self, d: &mut D) -> Result<(), E> { let token = try!(d.expect_token()); self.gather_token(d, token) } #[inline] fn gather_token, E>(&mut self, d: &mut D, token: Token) -> Result<(), E> { match token { token @ Option(true) => { self.tokens.push(token); self.gather(d) } EnumStart(name, variant, len) => { self.tokens.reserve_additional(len + 1); self.tokens.push(EnumStart(name, variant, len)); self.gather_seq(d) } StructStart(name, len) => { self.tokens.reserve_additional(len + 1); self.tokens.push(StructStart(name, len)); self.gather_struct(d) } TupleStart(len) => { self.tokens.reserve_additional(len + 1); self.tokens.push(TupleStart(len)); self.gather_seq(d) } SeqStart(len) => { self.tokens.reserve_additional(len + 1); self.tokens.push(SeqStart(len)); self.gather_seq(d) } MapStart(len) => { self.tokens.reserve_additional(len + 1); self.tokens.push(MapStart(len)); self.gather_map(d) } End => { d.syntax_error() } token => { self.tokens.push(token); Ok(()) } } } #[inline] fn gather_seq, E>(&mut self, d: &mut D) -> Result<(), E> { loop { match try!(d.expect_token()) { token @ End => { self.tokens.push(token); return Ok(()); } token => { try!(self.gather_token(d, token)); } } } } #[inline] fn gather_struct, E>(&mut self, d: &mut D) -> Result<(), E> { loop { match try!(d.expect_token()) { token @ End => { self.tokens.push(token); return Ok(()); } token @ Str(_) | token @ String(_) => { self.tokens.push(token); try!(self.gather(d)) } _token => { return d.syntax_error(); } } } } #[inline] fn gather_map, E>(&mut self, d: &mut D) -> Result<(), E> { loop { match try!(d.expect_token()) { End => { self.tokens.push(End); return Ok(()); } token => { try!(self.gather_token(d, token)); try!(self.gather(d)) } } } } } impl Deserializable for GatherTokens { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result { let mut tokens = GatherTokens { tokens: vec!(), }; try!(tokens.gather_token(d, token)); Ok(tokens) } } ////////////////////////////////////////////////////////////////////////////// #[cfg(test)] mod tests { use std::collections::TreeMap; use serialize::Decoder; use super::{Deserializer, Deserializable, Token}; use super::{ Null, Bool, Int, I8, I16, I32, I64, Uint, U8, U16, U32, U64, F32, F64, Char, Str, String, Option, TupleStart, StructStart, EnumStart, SeqStart, MapStart, End, }; macro_rules! treemap { ($($k:expr => $v:expr),*) => ({ let mut _m = ::std::collections::TreeMap::new(); $(_m.insert($k, $v);)* _m }) } ////////////////////////////////////////////////////////////////////////////// #[deriving(Clone, PartialEq, Show, Decodable)] struct Inner { a: (), b: uint, c: TreeMap>, } impl Deserializable for Inner { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result { try!(d.expect_struct_start(token, "Inner")); let a = try!(d.expect_struct_field("a")); let b = try!(d.expect_struct_field("b")); let c = try!(d.expect_struct_field("c")); try!(d.expect_struct_end()); Ok(Inner { a: a, b: b, c: c }) } } ////////////////////////////////////////////////////////////////////////////// #[deriving(Clone, PartialEq, Show, Decodable)] struct Outer { inner: Vec, } impl Deserializable for Outer { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result { try!(d.expect_struct_start(token, "Outer")); let inner = try!(d.expect_struct_field("inner")); try!(d.expect_struct_end()); Ok(Outer { inner: inner }) } } ////////////////////////////////////////////////////////////////////////////// #[deriving(Clone, PartialEq, Show, Decodable)] enum Animal { Dog, Frog(String, int) } impl Deserializable for Animal { #[inline] fn deserialize_token< D: Deserializer, E >(d: &mut D, token: Token) -> Result { match try!(d.expect_enum_start(token, "Animal", ["Dog", "Frog"])) { 0 => { try!(d.expect_enum_end()); Ok(Dog) } 1 => { let x0 = try!(Deserializable::deserialize(d)); let x1 = try!(Deserializable::deserialize(d)); try!(d.expect_enum_end()); Ok(Frog(x0, x1)) } _ => unreachable!(), } } } ////////////////////////////////////////////////////////////////////////////// #[deriving(Show)] enum Error { EndOfStream, SyntaxError, } ////////////////////////////////////////////////////////////////////////////// struct TokenDeserializer { tokens: Iter, } impl> TokenDeserializer { #[inline] fn new(tokens: Iter) -> TokenDeserializer { TokenDeserializer { tokens: tokens, } } } impl> Iterator> for TokenDeserializer { #[inline] fn next(&mut self) -> Option> { match self.tokens.next() { None => None, Some(token) => Some(Ok(token)), } } } impl> Deserializer for TokenDeserializer { fn end_of_stream_error(&self) -> Result { Err(EndOfStream) } fn syntax_error(&self) -> Result { Err(SyntaxError) } } ////////////////////////////////////////////////////////////////////////////// macro_rules! test_value { ($name:ident, [$($tokens:expr => $value:expr: $ty:ty),*]) => { #[test] fn $name() { $( let mut deserializer = TokenDeserializer::new($tokens.move_iter()); let value: $ty = Deserializable::deserialize(&mut deserializer).unwrap(); assert_eq!(value, $value); )+ } } } test_value!(test_primitives, [ vec!(Null) => (): (), vec!(Bool(true)) => true: bool, vec!(Bool(false)) => false: bool, vec!(Int(5)) => 5: int, vec!(I8(5)) => 5: i8, vec!(I16(5)) => 5: i16, vec!(I32(5)) => 5: i32, vec!(I64(5)) => 5: i64, vec!(Uint(5)) => 5: uint, vec!(U8(5)) => 5: u8, vec!(U16(5)) => 5: u16, vec!(U32(5)) => 5: u32, vec!(U64(5)) => 5: u64, vec!(F32(5.0)) => 5.0: f32, vec!(F64(5.0)) => 5.0: f64, vec!(Char('c')) => 'c': char, vec!(Str("abc")) => "abc": &str, vec!(String("abc".to_string())) => "abc".to_string(): String ]) test_value!(test_tuples, [ vec!( TupleStart(0), End, ) => (): (), vec!( TupleStart(2), Int(5), Str("a"), End, ) => (5, "a"): (int, &'static str), vec!( TupleStart(3), Null, TupleStart(0), End, TupleStart(2), Int(5), Str("a"), End, End, ) => ((), (), (5, "a")): ((), (), (int, &'static str)) ]) test_value!(test_options, [ vec!(Option(false)) => None: Option, vec!( Option(true), Int(5), ) => Some(5): Option ]) test_value!(test_structs, [ vec!( StructStart("Outer", 1), Str("inner"), SeqStart(0), End, End, ) => Outer { inner: vec!() }: Outer, vec!( StructStart("Outer", 1), Str("inner"), SeqStart(1), StructStart("Inner", 3), Str("a"), Null, Str("b"), Uint(5), Str("c"), MapStart(1), String("abc".to_string()), Option(true), Char('c'), End, End, End, End, ) => Outer { inner: vec!( Inner { a: (), b: 5, c: treemap!("abc".to_string() => Some('c')), }, ), }: Outer ]) test_value!(test_enums, [ vec!( EnumStart("Animal", "Dog", 0), End, ) => Dog: Animal, vec!( EnumStart("Animal", "Frog", 2), String("Henry".to_string()), Int(349), End, ) => Frog("Henry".to_string(), 349): Animal ]) test_value!(test_vecs, [ vec!( SeqStart(0), End, ) => vec!(): Vec, vec!( SeqStart(3), Int(5), Int(6), Int(7), End, ) => vec!(5, 6, 7): Vec, vec!( SeqStart(3), SeqStart(1), Int(1), End, SeqStart(2), Int(2), Int(3), End, SeqStart(3), Int(4), Int(5), Int(6), End, End, ) => vec!(vec!(1), vec!(2, 3), vec!(4, 5, 6)): Vec> ]) test_value!(test_treemaps, [ vec!( MapStart(0), End, ) => treemap!(): TreeMap, vec!( MapStart(2), Int(5), String("a".to_string()), Int(6), String("b".to_string()), End, ) => treemap!(5 => "a".to_string(), 6 => "b".to_string()): TreeMap ]) }