Fix #[derive_serialize] for enums
This commit is contained in:
parent
65cfcd03f8
commit
3fac47e01c
@ -6,10 +6,6 @@ authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
|||||||
[lib]
|
[lib]
|
||||||
name = "serde2"
|
name = "serde2"
|
||||||
|
|
||||||
#[[bin]]
|
|
||||||
#name = "serde2"
|
|
||||||
#path = "src/bin.rs"
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rustc-serialize = "*"
|
rustc-serialize = "*"
|
||||||
|
|
||||||
|
@ -3,7 +3,6 @@ name = "serde2_macros"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
||||||
|
|
||||||
[[lib]]
|
[lib]
|
||||||
name = "serde2_macros"
|
name = "serde2_macros"
|
||||||
path = "src/lib.rs"
|
|
||||||
plugin = true
|
plugin = true
|
||||||
|
@ -52,12 +52,13 @@ pub fn plugin_registrar(reg: &mut Registry) {
|
|||||||
Decorator(Box::new(expand_derive_deserialize)));
|
Decorator(Box::new(expand_derive_deserialize)));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn expand_derive_serialize<>(cx: &mut ExtCtxt,
|
fn expand_derive_serialize(
|
||||||
sp: Span,
|
cx: &mut ExtCtxt,
|
||||||
mitem: &MetaItem,
|
sp: Span,
|
||||||
item: &Item,
|
mitem: &MetaItem,
|
||||||
mut push: Box<FnMut(P<ast::Item>)>)
|
item: &Item,
|
||||||
{
|
mut push: Box<FnMut(P<ast::Item>)>
|
||||||
|
) {
|
||||||
let inline = cx.meta_word(sp, token::InternedString::new("inline"));
|
let inline = cx.meta_word(sp, token::InternedString::new("inline"));
|
||||||
let attrs = vec!(cx.attribute(sp, inline));
|
let attrs = vec!(cx.attribute(sp, inline));
|
||||||
|
|
||||||
@ -103,7 +104,7 @@ fn expand_derive_serialize<>(cx: &mut ExtCtxt,
|
|||||||
),
|
),
|
||||||
attributes: attrs,
|
attributes: attrs,
|
||||||
combine_substructure: combine_substructure(Box::new(|a, b, c| {
|
combine_substructure: combine_substructure(Box::new(|a, b, c| {
|
||||||
serialize_substructure(a, b, c)
|
serialize_substructure(a, b, c, item)
|
||||||
})),
|
})),
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
@ -112,11 +113,14 @@ fn expand_derive_serialize<>(cx: &mut ExtCtxt,
|
|||||||
trait_def.expand(cx, mitem, item, |item| push(item))
|
trait_def.expand(cx, mitem, item, |item| push(item))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn serialize_substructure(cx: &ExtCtxt, span: Span, substr: &Substructure) -> P<Expr> {
|
fn serialize_substructure(cx: &ExtCtxt,
|
||||||
|
span: Span,
|
||||||
|
substr: &Substructure,
|
||||||
|
item: &Item) -> P<Expr> {
|
||||||
let visitor = substr.nonself_args[0].clone();
|
let visitor = substr.nonself_args[0].clone();
|
||||||
|
|
||||||
match *substr.fields {
|
match (&item.node, &*substr.fields) {
|
||||||
Struct(ref fields) => {
|
(&ast::ItemStruct(..), &Struct(ref fields)) => {
|
||||||
if fields.is_empty() {
|
if fields.is_empty() {
|
||||||
serialize_tuple_struct(cx)
|
serialize_tuple_struct(cx)
|
||||||
} else {
|
} else {
|
||||||
@ -124,8 +128,14 @@ fn serialize_substructure(cx: &ExtCtxt, span: Span, substr: &Substructure) -> P<
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
EnumMatching(_idx, variant, ref fields) => {
|
(&ast::ItemEnum(_, ref generics), &EnumMatching(_idx, variant, ref fields)) => {
|
||||||
serialize_enum(cx, span, visitor, substr.type_ident, variant, fields)
|
serialize_enum(cx,
|
||||||
|
span,
|
||||||
|
visitor,
|
||||||
|
substr.type_ident,
|
||||||
|
variant,
|
||||||
|
&fields[],
|
||||||
|
generics)
|
||||||
}
|
}
|
||||||
|
|
||||||
_ => cx.bug("expected Struct or EnumMatching in derive_serialize")
|
_ => cx.bug("expected Struct or EnumMatching in derive_serialize")
|
||||||
@ -141,8 +151,7 @@ fn serialize_struct(cx: &ExtCtxt,
|
|||||||
span: Span,
|
span: Span,
|
||||||
visitor: P<Expr>,
|
visitor: P<Expr>,
|
||||||
type_ident: Ident,
|
type_ident: Ident,
|
||||||
fields: &Vec<FieldInfo>) -> P<Expr> {
|
fields: &[FieldInfo]) -> P<Expr> {
|
||||||
|
|
||||||
let type_name = cx.expr_str(
|
let type_name = cx.expr_str(
|
||||||
span,
|
span,
|
||||||
token::get_ident(type_ident));
|
token::get_ident(type_ident));
|
||||||
@ -203,44 +212,258 @@ fn serialize_struct(cx: &ExtCtxt,
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn serialize_enum(cx: &ExtCtxt,
|
|
||||||
span: Span,
|
|
||||||
visitor: P<Expr>,
|
|
||||||
type_ident: Ident,
|
fn serialize_enum(
|
||||||
variant: &ast::Variant,
|
cx: &ExtCtxt,
|
||||||
fields: &Vec<FieldInfo>) -> P<Expr> {
|
span: Span,
|
||||||
let type_name = cx.expr_str(
|
visitor: P<Expr>,
|
||||||
span,
|
type_ident: Ident,
|
||||||
token::get_ident(type_ident)
|
variant: &ast::Variant,
|
||||||
);
|
fields: &[FieldInfo],
|
||||||
let variant_name = cx.expr_str(
|
generics: &ast::Generics,
|
||||||
span,
|
) -> P<Expr> {
|
||||||
token::get_ident(variant.node.name)
|
let type_name = cx.expr_str(span, token::get_ident(type_ident));
|
||||||
|
let variant_ident = variant.node.name;
|
||||||
|
let variant_name = cx.expr_str(span, token::get_ident(variant_ident));
|
||||||
|
|
||||||
|
if fields.is_empty() {
|
||||||
|
quote_expr!(cx,
|
||||||
|
::serde2::ser::Visitor::visit_enum_unit(
|
||||||
|
$visitor,
|
||||||
|
$type_name,
|
||||||
|
$variant_name)
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
serialize_variant(
|
||||||
|
cx,
|
||||||
|
span,
|
||||||
|
visitor,
|
||||||
|
type_name,
|
||||||
|
variant_name,
|
||||||
|
generics,
|
||||||
|
variant,
|
||||||
|
fields)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
fn serialize_variant(
|
||||||
|
cx: &ExtCtxt,
|
||||||
|
span: Span,
|
||||||
|
visitor: P<ast::Expr>,
|
||||||
|
type_name: P<ast::Expr>,
|
||||||
|
variant_name: P<ast::Expr>,
|
||||||
|
generics: &ast::Generics,
|
||||||
|
variant: &ast::Variant,
|
||||||
|
fields: &[FieldInfo],
|
||||||
|
) -> P<Expr> {
|
||||||
|
// We'll take a reference to the values in the variant, so we need a new lifetime.
|
||||||
|
let lifetimes = generics.lifetimes.iter()
|
||||||
|
.map(|lifetime_def| lifetime_def.lifetime.clone())
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let mut generics = generics.clone();
|
||||||
|
generics.lifetimes.push(
|
||||||
|
cx.lifetime_def(
|
||||||
|
span,
|
||||||
|
cx.name_of("'__a"),
|
||||||
|
lifetimes
|
||||||
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let (
|
||||||
|
trait_name,
|
||||||
|
visitor_method_name,
|
||||||
|
tys,
|
||||||
|
): (Ident, Ident, Vec<P<ast::Ty>>) = match variant.node.kind {
|
||||||
|
ast::TupleVariantKind(ref args) => {
|
||||||
|
(
|
||||||
|
cx.ident_of("SeqVisitor"),
|
||||||
|
cx.ident_of("visit_enum_seq"),
|
||||||
|
args.iter()
|
||||||
|
.map(|arg| arg.ty.clone())
|
||||||
|
.collect()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
ast::StructVariantKind(ref struct_def) => {
|
||||||
|
(
|
||||||
|
cx.ident_of("SeqVisitor"),
|
||||||
|
cx.ident_of("visit_enum_seq"),
|
||||||
|
struct_def.fields.iter()
|
||||||
|
.map(|field| field.node.ty.clone())
|
||||||
|
.collect()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let len = fields.len();
|
let len = fields.len();
|
||||||
|
|
||||||
let stmts: Vec<P<ast::Stmt>> = fields.iter()
|
let visitor_field_names: Vec<ast::Ident> = (0 .. len)
|
||||||
.map(|&FieldInfo { ref self_, .. }| {
|
.map(|i| token::str_to_ident(&format!("field{}", i)))
|
||||||
quote_stmt!(
|
.collect();
|
||||||
cx,
|
|
||||||
try!($visitor.serialize_enum_elt(&$self_))
|
let mut visitor_fields = vec![
|
||||||
|
respan(
|
||||||
|
span,
|
||||||
|
ast::StructField_ {
|
||||||
|
kind: ast::NamedField(
|
||||||
|
cx.ident_of("state"),
|
||||||
|
ast::Visibility::Inherited,
|
||||||
|
),
|
||||||
|
id: ast::DUMMY_NODE_ID,
|
||||||
|
ty: cx.ty_ident(span, cx.ident_of("u32")),
|
||||||
|
attrs: Vec::new(),
|
||||||
|
}
|
||||||
|
),
|
||||||
|
];
|
||||||
|
|
||||||
|
visitor_fields.extend(
|
||||||
|
visitor_field_names.iter()
|
||||||
|
.zip(tys.iter())
|
||||||
|
.map(|(name, ty)| {
|
||||||
|
respan(
|
||||||
|
span,
|
||||||
|
ast::StructField_ {
|
||||||
|
kind: ast::NamedField(
|
||||||
|
*name,
|
||||||
|
ast::Visibility::Inherited,
|
||||||
|
),
|
||||||
|
id: ast::DUMMY_NODE_ID,
|
||||||
|
ty: cx.ty_rptr(
|
||||||
|
span,
|
||||||
|
ty.clone(),
|
||||||
|
Some(cx.lifetime(span, cx.name_of("'__a"))),
|
||||||
|
ast::MutImmutable,
|
||||||
|
),
|
||||||
|
attrs: Vec::new(),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
let visitor_ident = cx.ident_of("__Visitor");
|
||||||
|
|
||||||
|
let visitor_struct = cx.item_struct_poly(
|
||||||
|
span,
|
||||||
|
visitor_ident,
|
||||||
|
ast::StructDef {
|
||||||
|
fields: visitor_fields,
|
||||||
|
ctor_id: None,
|
||||||
|
},
|
||||||
|
generics,
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut visitor_field_exprs = vec![
|
||||||
|
cx.field_imm(
|
||||||
|
span,
|
||||||
|
cx.ident_of("state"),
|
||||||
|
quote_expr!(cx, 0),
|
||||||
|
),
|
||||||
|
];
|
||||||
|
|
||||||
|
visitor_field_exprs.extend(
|
||||||
|
visitor_field_names.iter()
|
||||||
|
.zip(fields.iter())
|
||||||
|
.map(|(name, field)| {
|
||||||
|
let e = cx.expr_addr_of(
|
||||||
|
span,
|
||||||
|
field.self_.clone(),
|
||||||
|
);
|
||||||
|
|
||||||
|
cx.field_imm(span, *name, e)
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
let visitor_expr = cx.expr_struct(
|
||||||
|
span,
|
||||||
|
cx.path_ident(span, visitor_ident),
|
||||||
|
visitor_field_exprs);
|
||||||
|
|
||||||
|
let mut first = true;
|
||||||
|
|
||||||
|
let visitor_arms: Vec<ast::Arm> = visitor_field_names.iter()
|
||||||
|
.zip(fields.iter())
|
||||||
|
.enumerate()
|
||||||
|
.map(|(state, (name, field))| {
|
||||||
|
let field_expr = cx.expr_field_access(
|
||||||
|
span,
|
||||||
|
cx.expr_self(span),
|
||||||
|
*name,
|
||||||
|
);
|
||||||
|
|
||||||
|
let visit_expr = match field.name {
|
||||||
|
Some(real_name) => {
|
||||||
|
let real_name = cx.expr_str(span, token::get_ident(real_name));
|
||||||
|
quote_expr!(cx,
|
||||||
|
::serde2::ser::Visitor::visit_map_elt(
|
||||||
|
visitor,
|
||||||
|
$first,
|
||||||
|
$real_name,
|
||||||
|
$field_expr,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
quote_expr!(cx,
|
||||||
|
::serde2::ser::Visitor::visit_seq_elt(
|
||||||
|
visitor,
|
||||||
|
$first,
|
||||||
|
$field_expr,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
first = false;
|
||||||
|
|
||||||
|
let state = state as u32;
|
||||||
|
|
||||||
|
quote_arm!(cx,
|
||||||
|
$state => {
|
||||||
|
self.state += 1;
|
||||||
|
Ok(Some(try!($visit_expr)))
|
||||||
|
}
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
quote_expr!(cx, {
|
quote_expr!(cx, {
|
||||||
try!($visitor.serialize_enum_start($type_name, $variant_name, $len));
|
$visitor_struct
|
||||||
$stmts
|
|
||||||
$visitor.serialize_enum_end()
|
impl<'__a> ::serde2::ser::$trait_name for __Visitor<'__a> {
|
||||||
|
fn visit<
|
||||||
|
V: ::serde2::ser::Visitor,
|
||||||
|
>(&mut self, visitor: &mut V) -> Result<Option<V::Value>, V::Error> {
|
||||||
|
match self.state {
|
||||||
|
$visitor_arms
|
||||||
|
_ => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
($len - self.state as usize, Some($len - self.state as usize))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
::serde2::ser::Visitor::$visitor_method_name(
|
||||||
|
$visitor,
|
||||||
|
$type_name,
|
||||||
|
$variant_name,
|
||||||
|
$visitor_expr,
|
||||||
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn expand_derive_deserialize(cx: &mut ExtCtxt,
|
pub fn expand_derive_deserialize(
|
||||||
sp: Span,
|
cx: &mut ExtCtxt,
|
||||||
mitem: &MetaItem,
|
sp: Span,
|
||||||
item: &Item,
|
mitem: &MetaItem,
|
||||||
mut push: Box<FnMut(P<ast::Item>)>)
|
item: &Item,
|
||||||
{
|
mut push: Box<FnMut(P<ast::Item>)>
|
||||||
|
) {
|
||||||
let inline = cx.meta_word(sp, token::InternedString::new("inline"));
|
let inline = cx.meta_word(sp, token::InternedString::new("inline"));
|
||||||
let attrs = vec!(cx.attribute(sp, inline));
|
let attrs = vec!(cx.attribute(sp, inline));
|
||||||
|
|
||||||
|
@ -1,620 +0,0 @@
|
|||||||
extern crate serde2;
|
|
||||||
|
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::option;
|
|
||||||
use std::string;
|
|
||||||
|
|
||||||
use serde2::de;
|
|
||||||
use serde2::de::{Deserialize, Deserializer};
|
|
||||||
|
|
||||||
#[derive(Show)]
|
|
||||||
pub enum Token {
|
|
||||||
Null,
|
|
||||||
Int(int),
|
|
||||||
String(string::String),
|
|
||||||
Option(bool),
|
|
||||||
SeqStart(uint),
|
|
||||||
MapStart(uint),
|
|
||||||
End,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Show)]
|
|
||||||
enum Error {
|
|
||||||
SyntaxError,
|
|
||||||
EndOfStreamError,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl de::Error for Error {
|
|
||||||
fn syntax_error() -> Error {
|
|
||||||
Error::SyntaxError
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end_of_stream_error() -> Error {
|
|
||||||
Error::EndOfStreamError
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct MyDeserializer<Iter> {
|
|
||||||
tokens: Iter,
|
|
||||||
peeked: option::Option<Token>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<Iter: Iterator<Item=Token>> MyDeserializer<Iter> {
|
|
||||||
pub fn new(tokens: Iter) -> MyDeserializer<Iter> {
|
|
||||||
MyDeserializer {
|
|
||||||
tokens: tokens,
|
|
||||||
peeked: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn next(&mut self) -> option::Option<Token> {
|
|
||||||
match self.peeked.take() {
|
|
||||||
Some(token) => { return Some(token); }
|
|
||||||
None => { }
|
|
||||||
}
|
|
||||||
|
|
||||||
self.tokens.next()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn peek<'a>(&'a mut self) -> option::Option<&'a Token> {
|
|
||||||
match self.peeked {
|
|
||||||
Some(_) => { }
|
|
||||||
None => { self.peeked = self.tokens.next(); }
|
|
||||||
}
|
|
||||||
|
|
||||||
self.peeked.as_ref()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<Iter: Iterator<Item=Token>> Deserializer<Error> for MyDeserializer<Iter> {
|
|
||||||
fn visit<
|
|
||||||
R,
|
|
||||||
V: de::Visitor<MyDeserializer<Iter>, R, Error>,
|
|
||||||
>(&mut self, visitor: &mut V) -> Result<R, Error> {
|
|
||||||
use serde2::de::Error;
|
|
||||||
|
|
||||||
match self.next() {
|
|
||||||
Some(Token::Null) => {
|
|
||||||
visitor.visit_null()
|
|
||||||
}
|
|
||||||
Some(Token::Int(v)) => {
|
|
||||||
visitor.visit_int(v)
|
|
||||||
}
|
|
||||||
Some(Token::String(v)) => {
|
|
||||||
visitor.visit_string(v)
|
|
||||||
}
|
|
||||||
Some(Token::Option(is_some)) => {
|
|
||||||
visitor.visit_option(MyOptionVisitor { d: self, is_some: is_some })
|
|
||||||
}
|
|
||||||
Some(Token::SeqStart(len)) => {
|
|
||||||
visitor.visit_seq(MySeqVisitor { d: self, len: len })
|
|
||||||
}
|
|
||||||
Some(Token::MapStart(len)) => {
|
|
||||||
visitor.visit_map(MyMapVisitor { d: self, len: len })
|
|
||||||
}
|
|
||||||
Some(Token::End) => {
|
|
||||||
Err(Error::syntax_error())
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
Err(Error::end_of_stream_error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_option<
|
|
||||||
R,
|
|
||||||
V: de::Visitor<MyDeserializer<Iter>, R, Error>,
|
|
||||||
>(&mut self, visitor: &mut V) -> Result<R, Error> {
|
|
||||||
match self.peek() {
|
|
||||||
Some(&Token::Null) => {
|
|
||||||
self.next();
|
|
||||||
visitor.visit_option(MyOptionVisitor {
|
|
||||||
d: self,
|
|
||||||
is_some: false,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Some(&Token::Option(is_some)) => {
|
|
||||||
self.next();
|
|
||||||
visitor.visit_option(MyOptionVisitor {
|
|
||||||
d: self,
|
|
||||||
is_some: is_some,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
visitor.visit_option(MyOptionVisitor {
|
|
||||||
d: self,
|
|
||||||
is_some: true,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct MyOptionVisitor<'a, Iter: 'a> {
|
|
||||||
d: &'a mut MyDeserializer<Iter>,
|
|
||||||
is_some: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<
|
|
||||||
'a,
|
|
||||||
Iter: Iterator<Item=Token>,
|
|
||||||
> de::OptionVisitor<MyDeserializer<Iter>, Error> for MyOptionVisitor<'a, Iter> {
|
|
||||||
fn visit<
|
|
||||||
T: Deserialize<MyDeserializer<Iter>, Error>,
|
|
||||||
>(&mut self) -> Result<option::Option<T>, Error> {
|
|
||||||
if self.is_some {
|
|
||||||
self.is_some = false;
|
|
||||||
let value = try!(Deserialize::deserialize(self.d));
|
|
||||||
Ok(Some(value))
|
|
||||||
} else {
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct MySeqVisitor<'a, Iter: 'a> {
|
|
||||||
d: &'a mut MyDeserializer<Iter>,
|
|
||||||
len: uint,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<
|
|
||||||
'a,
|
|
||||||
Iter: Iterator<Item=Token>,
|
|
||||||
> de::SeqVisitor<MyDeserializer<Iter>, Error> for MySeqVisitor<'a, Iter> {
|
|
||||||
fn visit<
|
|
||||||
T: Deserialize<MyDeserializer<Iter>, Error>
|
|
||||||
>(&mut self) -> Result<option::Option<T>, Error> {
|
|
||||||
use serde2::de::Error;
|
|
||||||
|
|
||||||
match self.d.peek() {
|
|
||||||
Some(&Token::End) => {
|
|
||||||
self.d.next();
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
self.len -= 1;
|
|
||||||
let value = try!(Deserialize::deserialize(self.d));
|
|
||||||
Ok(Some(value))
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
Err(Error::syntax_error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
use serde2::de::Error;
|
|
||||||
|
|
||||||
match self.d.next() {
|
|
||||||
Some(Token::End) => Ok(()),
|
|
||||||
Some(_) => Err(Error::syntax_error()),
|
|
||||||
None => Err(Error::end_of_stream_error()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (uint, option::Option<uint>) {
|
|
||||||
(self.len, Some(self.len))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct MyMapVisitor<'a, Iter: 'a> {
|
|
||||||
d: &'a mut MyDeserializer<Iter>,
|
|
||||||
len: uint,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<
|
|
||||||
'a,
|
|
||||||
Iter: Iterator<Item=Token>,
|
|
||||||
> de::MapVisitor<MyDeserializer<Iter>, Error> for MyMapVisitor<'a, Iter> {
|
|
||||||
fn visit_key<
|
|
||||||
K: Deserialize<MyDeserializer<Iter>, Error>,
|
|
||||||
>(&mut self) -> Result<option::Option<K>, Error> {
|
|
||||||
use serde2::de::Error;
|
|
||||||
|
|
||||||
match self.d.peek() {
|
|
||||||
Some(&Token::End) => {
|
|
||||||
self.d.next();
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
self.len -= 1;
|
|
||||||
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.d))))
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
Err(Error::syntax_error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_value<
|
|
||||||
V: Deserialize<MyDeserializer<Iter>, Error>,
|
|
||||||
>(&mut self) -> Result<V, Error> {
|
|
||||||
Ok(try!(Deserialize::deserialize(self.d)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
use serde2::de::Error;
|
|
||||||
|
|
||||||
match self.d.next() {
|
|
||||||
Some(Token::End) => Ok(()),
|
|
||||||
Some(_) => Err(Error::syntax_error()),
|
|
||||||
None => Err(Error::end_of_stream_error()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (uint, option::Option<uint>) {
|
|
||||||
(self.len, Some(self.len))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
mod json {
|
|
||||||
use std::collections::BTreeMap;
|
|
||||||
use serde2::de;
|
|
||||||
|
|
||||||
#[derive(Show)]
|
|
||||||
pub enum Value {
|
|
||||||
Null,
|
|
||||||
//Bool(bool),
|
|
||||||
Int(int),
|
|
||||||
//String(String),
|
|
||||||
List(Vec<Value>),
|
|
||||||
Map(BTreeMap<String, Value>),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<
|
|
||||||
D: de::Deserializer<E>,
|
|
||||||
E: de::Error,
|
|
||||||
> de::Deserialize<D, E> for Value {
|
|
||||||
fn deserialize(d: &mut D) -> Result<Value, E> {
|
|
||||||
struct Visitor;
|
|
||||||
|
|
||||||
impl<
|
|
||||||
D: de::Deserializer<E>,
|
|
||||||
E: de::Error,
|
|
||||||
> de::Visitor<D, Value, E> for Visitor {
|
|
||||||
fn visit_null(&mut self) -> Result<Value, E> {
|
|
||||||
Ok(Value::Null)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_int(&mut self, v: int) -> Result<Value, E> {
|
|
||||||
Ok(Value::Int(v))
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
fn visit_string(&mut self, _d: &mut D, v: String) -> Result<Value, E> {
|
|
||||||
Ok(Value::String(v))
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
fn visit_option<
|
|
||||||
Visitor: de::OptionVisitor<D, E>,
|
|
||||||
>(&mut self, mut visitor: Visitor) -> Result<Value, E> {
|
|
||||||
match try!(visitor.visit()) {
|
|
||||||
Some(value) => Ok(value),
|
|
||||||
None => Ok(Value::Null),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_seq<
|
|
||||||
Visitor: de::SeqVisitor<D, E>,
|
|
||||||
>(&mut self, mut visitor: Visitor) -> Result<Value, E> {
|
|
||||||
let (len, _) = visitor.size_hint();
|
|
||||||
let mut values = Vec::with_capacity(len);
|
|
||||||
|
|
||||||
loop {
|
|
||||||
match try!(visitor.visit()) {
|
|
||||||
Some(value) => {
|
|
||||||
values.push(value);
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Value::List(values))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_map<
|
|
||||||
Visitor: de::MapVisitor<D, E>,
|
|
||||||
>(&mut self, mut visitor: Visitor) -> Result<Value, E> {
|
|
||||||
let mut values = BTreeMap::new();
|
|
||||||
|
|
||||||
loop {
|
|
||||||
match try!(visitor.visit()) {
|
|
||||||
Some((key, value)) => {
|
|
||||||
values.insert(key, value);
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Value::Map(values))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
d.visit(&mut Visitor)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
pub fn main() {
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::SeqStart(2),
|
|
||||||
Token::Int(1),
|
|
||||||
Token::Int(2),
|
|
||||||
Token::End,
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<Vec<int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("vec: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::SeqStart(2),
|
|
||||||
Token::Int(3),
|
|
||||||
Token::Int(4),
|
|
||||||
Token::End,
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<(int, int), Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("tuple: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::SeqStart(2),
|
|
||||||
Token::Int(5),
|
|
||||||
Token::Int(6),
|
|
||||||
Token::End,
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("value: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Option(true),
|
|
||||||
Token::Int(7),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<option::Option<int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("optiony: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Option(false),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<option::Option<int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("optiony: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Option(true),
|
|
||||||
Token::Int(8),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("optiony value: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Option(false),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("optiony value: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Int(9),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<option::Option<int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("option: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Null,
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<option::Option<int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("option: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Int(10),
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("option value: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::Null,
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("option value: {:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::MapStart(2),
|
|
||||||
Token::String("a".to_string()),
|
|
||||||
Token::Int(1),
|
|
||||||
Token::String("b".to_string()),
|
|
||||||
Token::Int(2),
|
|
||||||
Token::End
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<HashMap<string::String, int>, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("{:?}", v);
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let tokens = vec!(
|
|
||||||
Token::MapStart(2),
|
|
||||||
Token::String("a".to_string()),
|
|
||||||
Token::Int(1),
|
|
||||||
Token::String("b".to_string()),
|
|
||||||
Token::Int(2),
|
|
||||||
Token::End
|
|
||||||
);
|
|
||||||
let mut state = MyDeserializer::new(tokens.into_iter());
|
|
||||||
|
|
||||||
let v: Result<json::Value, Error> = Deserialize::deserialize(&mut state);
|
|
||||||
println!("{:?}", v);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
use std::collections::BTreeMap;
|
|
||||||
use serde::{Serialize, GatherTokens};
|
|
||||||
use serde::json;
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct Foo {
|
|
||||||
x: int,
|
|
||||||
y: int,
|
|
||||||
z: &'static str,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: serde::VisitorState<R>, R> serde::Serialize<S, R> for Foo {
|
|
||||||
fn serialize(&self, state: &mut S) -> R {
|
|
||||||
state.visit_named_map("Foo", FooSerialize {
|
|
||||||
value: self,
|
|
||||||
state: 0,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct FooSerialize<'a> {
|
|
||||||
value: &'a Foo,
|
|
||||||
state: uint,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, S: serde::VisitorState<R>, R> serde::Visitor<S, R> for FooSerialize<'a> {
|
|
||||||
fn visit(&mut self, state: &mut S) -> Option<R> {
|
|
||||||
match self.state {
|
|
||||||
0 => {
|
|
||||||
self.state += 1;
|
|
||||||
Some(state.visit_map_elt(true, "x", &self.value.x))
|
|
||||||
}
|
|
||||||
1 => {
|
|
||||||
self.state += 1;
|
|
||||||
Some(state.visit_map_elt(false, "y", &self.value.y))
|
|
||||||
}
|
|
||||||
2 => {
|
|
||||||
self.state += 1;
|
|
||||||
Some(state.visit_map_elt(false, "z", &self.value.z))
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
||||||
let size = 3 - self.state;
|
|
||||||
(size, Some(size))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
let value = 5i;
|
|
||||||
|
|
||||||
let mut s = GatherTokens::new();
|
|
||||||
value.serialize(&mut s);
|
|
||||||
println!("tokens: {:?}", s.unwrap());
|
|
||||||
|
|
||||||
println!("json: {:?}", json::to_string(&value).unwrap().unwrap());
|
|
||||||
println!("");
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let value = vec!(1i, 2, 3);
|
|
||||||
|
|
||||||
let mut s = GatherTokens::new();
|
|
||||||
value.serialize(&mut s);
|
|
||||||
println!("tokens: {:?}", s.unwrap());
|
|
||||||
|
|
||||||
println!("json: {:?}", json::to_string(&value).unwrap().unwrap());
|
|
||||||
println!("");
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let mut value = BTreeMap::new();
|
|
||||||
value.insert("a", 1i);
|
|
||||||
value.insert("b", 2);
|
|
||||||
value.insert("c", 3);
|
|
||||||
|
|
||||||
let mut s = GatherTokens::new();
|
|
||||||
value.serialize(&mut s);
|
|
||||||
println!("tokens: {:?}", s.unwrap());
|
|
||||||
|
|
||||||
println!("json: {:?}", json::to_string(&value).unwrap().unwrap());
|
|
||||||
println!("");
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
/*
|
|
||||||
println!("{:?}", to_format_vec(&5i));
|
|
||||||
println!("{:?}", to_format_string(&5i));
|
|
||||||
*/
|
|
||||||
|
|
||||||
let value = Foo { x: 1, y: 2, z: "abc" };
|
|
||||||
|
|
||||||
let mut s = GatherTokens::new();
|
|
||||||
value.serialize(&mut s);
|
|
||||||
println!("tokens: {:?}", s.unwrap());
|
|
||||||
|
|
||||||
println!("json: {:?}", json::to_string(&value).unwrap().unwrap());
|
|
||||||
println!("");
|
|
||||||
|
|
||||||
////
|
|
||||||
|
|
||||||
let value = (1i, "abc");
|
|
||||||
|
|
||||||
let mut s = GatherTokens::new();
|
|
||||||
value.serialize(&mut s);
|
|
||||||
println!("tokens: {:?}", s.unwrap());
|
|
||||||
|
|
||||||
println!("json: {:?}", json::to_string(&value).unwrap().unwrap());
|
|
||||||
println!("");
|
|
||||||
}
|
|
||||||
*/
|
|
@ -213,9 +213,8 @@ pub trait Visitor {
|
|||||||
pub trait SeqVisitor {
|
pub trait SeqVisitor {
|
||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
fn visit<
|
fn visit<T>(&mut self) -> Result<Option<T>, Self::Error>
|
||||||
T: Deserialize,
|
where T: Deserialize;
|
||||||
>(&mut self) -> Result<Option<T>, Self::Error>;
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Self::Error>;
|
fn end(&mut self) -> Result<(), Self::Error>;
|
||||||
|
|
||||||
@ -229,10 +228,10 @@ pub trait MapVisitor {
|
|||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn visit<
|
fn visit<K, V>(&mut self) -> Result<Option<(K, V)>, Self::Error>
|
||||||
K: Deserialize,
|
where K: Deserialize,
|
||||||
V: Deserialize,
|
V: Deserialize,
|
||||||
>(&mut self) -> Result<Option<(K, V)>, Self::Error> {
|
{
|
||||||
match try!(self.visit_key()) {
|
match try!(self.visit_key()) {
|
||||||
Some(key) => {
|
Some(key) => {
|
||||||
let value = try!(self.visit_value());
|
let value = try!(self.visit_value());
|
||||||
@ -242,13 +241,11 @@ pub trait MapVisitor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_key<
|
fn visit_key<K>(&mut self) -> Result<Option<K>, Self::Error>
|
||||||
K: Deserialize,
|
where K: Deserialize;
|
||||||
>(&mut self) -> Result<Option<K>, Self::Error>;
|
|
||||||
|
|
||||||
fn visit_value<
|
fn visit_value<V>(&mut self) -> Result<V, Self::Error>
|
||||||
V: Deserialize,
|
where V: Deserialize;
|
||||||
>(&mut self) -> Result<V, Self::Error>;
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Self::Error>;
|
fn end(&mut self) -> Result<(), Self::Error>;
|
||||||
|
|
||||||
|
@ -4,10 +4,9 @@ use unicode::str::Utf16Item;
|
|||||||
use std::str;
|
use std::str;
|
||||||
|
|
||||||
use de;
|
use de;
|
||||||
use de::Deserializer;
|
|
||||||
use super::error::{Error, ErrorCode};
|
use super::error::{Error, ErrorCode};
|
||||||
|
|
||||||
pub struct Parser<Iter> {
|
pub struct Deserializer<Iter> {
|
||||||
rdr: Iter,
|
rdr: Iter,
|
||||||
ch: Option<u8>,
|
ch: Option<u8>,
|
||||||
line: usize,
|
line: usize,
|
||||||
@ -15,11 +14,11 @@ pub struct Parser<Iter> {
|
|||||||
buf: Vec<u8>,
|
buf: Vec<u8>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Iter: Iterator<Item=u8>> Parser<Iter> {
|
impl<Iter: Iterator<Item=u8>> Deserializer<Iter> {
|
||||||
/// Creates the JSON parser.
|
/// Creates the JSON parser.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(rdr: Iter) -> Parser<Iter> {
|
pub fn new(rdr: Iter) -> Deserializer<Iter> {
|
||||||
let mut p = Parser {
|
let mut p = Deserializer {
|
||||||
rdr: rdr,
|
rdr: rdr,
|
||||||
ch: Some(b'\x00'),
|
ch: Some(b'\x00'),
|
||||||
line: 1,
|
line: 1,
|
||||||
@ -111,11 +110,17 @@ impl<Iter: Iterator<Item=u8>> Parser<Iter> {
|
|||||||
}
|
}
|
||||||
b'[' => {
|
b'[' => {
|
||||||
self.bump();
|
self.bump();
|
||||||
visitor.visit_seq(SeqVisitor { parser: self, first: true })
|
visitor.visit_seq(SeqVisitor {
|
||||||
|
de: self,
|
||||||
|
first: true,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
b'{' => {
|
b'{' => {
|
||||||
self.bump();
|
self.bump();
|
||||||
visitor.visit_map(MapVisitor { parser: self, first: true })
|
visitor.visit_map(MapVisitor {
|
||||||
|
de: self,
|
||||||
|
first: true,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
Err(self.error(ErrorCode::ExpectedSomeValue))
|
Err(self.error(ErrorCode::ExpectedSomeValue))
|
||||||
@ -377,7 +382,7 @@ impl<Iter: Iterator<Item=u8>> Parser<Iter> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Iter: Iterator<Item=u8>> Deserializer for Parser<Iter> {
|
impl<Iter: Iterator<Item=u8>> de::Deserializer for Deserializer<Iter> {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@ -389,7 +394,7 @@ impl<Iter: Iterator<Item=u8>> Deserializer for Parser<Iter> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct SeqVisitor<'a, Iter: 'a> {
|
struct SeqVisitor<'a, Iter: 'a> {
|
||||||
parser: &'a mut Parser<Iter>,
|
de: &'a mut Deserializer<Iter>,
|
||||||
first: bool,
|
first: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -399,43 +404,43 @@ impl<'a, Iter: Iterator<Item=u8>> de::SeqVisitor for SeqVisitor<'a, Iter> {
|
|||||||
fn visit<
|
fn visit<
|
||||||
T: de::Deserialize,
|
T: de::Deserialize,
|
||||||
>(&mut self) -> Result<Option<T>, Error> {
|
>(&mut self) -> Result<Option<T>, Error> {
|
||||||
self.parser.parse_whitespace();
|
self.de.parse_whitespace();
|
||||||
|
|
||||||
if self.parser.ch_is(b']') {
|
if self.de.ch_is(b']') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.first {
|
if self.first {
|
||||||
self.first = false;
|
self.first = false;
|
||||||
} else {
|
} else {
|
||||||
if self.parser.ch_is(b',') {
|
if self.de.ch_is(b',') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
} else if self.parser.eof() {
|
} else if self.de.eof() {
|
||||||
return Err(self.parser.error(ErrorCode::EOFWhileParsingList));
|
return Err(self.de.error(ErrorCode::EOFWhileParsingList));
|
||||||
} else {
|
} else {
|
||||||
return Err(self.parser.error(ErrorCode::ExpectedListCommaOrEnd));
|
return Err(self.de.error(ErrorCode::ExpectedListCommaOrEnd));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let value = try!(de::Deserialize::deserialize(self.parser));
|
let value = try!(de::Deserialize::deserialize(self.de));
|
||||||
Ok(Some(value))
|
Ok(Some(value))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
if self.parser.ch_is(b']') {
|
if self.de.ch_is(b']') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
Ok(())
|
Ok(())
|
||||||
} else if self.parser.eof() {
|
} else if self.de.eof() {
|
||||||
Err(self.parser.error(ErrorCode::EOFWhileParsingList))
|
Err(self.de.error(ErrorCode::EOFWhileParsingList))
|
||||||
} else {
|
} else {
|
||||||
Err(self.parser.error(ErrorCode::TrailingCharacters))
|
Err(self.de.error(ErrorCode::TrailingCharacters))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct MapVisitor<'a, Iter: 'a> {
|
struct MapVisitor<'a, Iter: 'a> {
|
||||||
parser: &'a mut Parser<Iter>,
|
de: &'a mut Deserializer<Iter>,
|
||||||
first: bool,
|
first: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -445,63 +450,63 @@ impl<'a, Iter: Iterator<Item=u8>> de::MapVisitor for MapVisitor<'a, Iter> {
|
|||||||
fn visit_key<
|
fn visit_key<
|
||||||
K: de::Deserialize,
|
K: de::Deserialize,
|
||||||
>(&mut self) -> Result<Option<K>, Error> {
|
>(&mut self) -> Result<Option<K>, Error> {
|
||||||
self.parser.parse_whitespace();
|
self.de.parse_whitespace();
|
||||||
|
|
||||||
if self.parser.ch_is(b'}') {
|
if self.de.ch_is(b'}') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.first {
|
if self.first {
|
||||||
self.first = false;
|
self.first = false;
|
||||||
} else {
|
} else {
|
||||||
if self.parser.ch_is(b',') {
|
if self.de.ch_is(b',') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
self.parser.parse_whitespace();
|
self.de.parse_whitespace();
|
||||||
} else if self.parser.eof() {
|
} else if self.de.eof() {
|
||||||
return Err(self.parser.error(ErrorCode::EOFWhileParsingObject));
|
return Err(self.de.error(ErrorCode::EOFWhileParsingObject));
|
||||||
} else {
|
} else {
|
||||||
return Err(self.parser.error(ErrorCode::ExpectedObjectCommaOrEnd));
|
return Err(self.de.error(ErrorCode::ExpectedObjectCommaOrEnd));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.parser.eof() {
|
if self.de.eof() {
|
||||||
return Err(self.parser.error(ErrorCode::EOFWhileParsingValue));
|
return Err(self.de.error(ErrorCode::EOFWhileParsingValue));
|
||||||
}
|
}
|
||||||
|
|
||||||
if !self.parser.ch_is(b'"') {
|
if !self.de.ch_is(b'"') {
|
||||||
return Err(self.parser.error(ErrorCode::KeyMustBeAString));
|
return Err(self.de.error(ErrorCode::KeyMustBeAString));
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Some(try!(de::Deserialize::deserialize(self.parser))))
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_value<
|
fn visit_value<
|
||||||
V: de::Deserialize,
|
V: de::Deserialize,
|
||||||
>(&mut self) -> Result<V, Error> {
|
>(&mut self) -> Result<V, Error> {
|
||||||
self.parser.parse_whitespace();
|
self.de.parse_whitespace();
|
||||||
|
|
||||||
if self.parser.ch_is(b':') {
|
if self.de.ch_is(b':') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
} else if self.parser.eof() {
|
} else if self.de.eof() {
|
||||||
return Err(self.parser.error(ErrorCode::EOFWhileParsingObject));
|
return Err(self.de.error(ErrorCode::EOFWhileParsingObject));
|
||||||
} else {
|
} else {
|
||||||
return Err(self.parser.error(ErrorCode::ExpectedColon));
|
return Err(self.de.error(ErrorCode::ExpectedColon));
|
||||||
}
|
}
|
||||||
|
|
||||||
self.parser.parse_whitespace();
|
self.de.parse_whitespace();
|
||||||
|
|
||||||
Ok(try!(de::Deserialize::deserialize(self.parser)))
|
Ok(try!(de::Deserialize::deserialize(self.de)))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
if self.parser.ch_is(b']') {
|
if self.de.ch_is(b']') {
|
||||||
self.parser.bump();
|
self.de.bump();
|
||||||
Ok(())
|
Ok(())
|
||||||
} else if self.parser.eof() {
|
} else if self.de.eof() {
|
||||||
Err(self.parser.error(ErrorCode::EOFWhileParsingList))
|
Err(self.de.error(ErrorCode::EOFWhileParsingList))
|
||||||
} else {
|
} else {
|
||||||
Err(self.parser.error(ErrorCode::TrailingCharacters))
|
Err(self.de.error(ErrorCode::TrailingCharacters))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -511,11 +516,11 @@ pub fn from_iter<I, T>(iter: I) -> Result<T, Error>
|
|||||||
where I: Iterator<Item=u8>,
|
where I: Iterator<Item=u8>,
|
||||||
T: de::Deserialize
|
T: de::Deserialize
|
||||||
{
|
{
|
||||||
let mut parser = Parser::new(iter);
|
let mut de = Deserializer::new(iter);
|
||||||
let value = try!(de::Deserialize::deserialize(&mut parser));
|
let value = try!(de::Deserialize::deserialize(&mut de));
|
||||||
|
|
||||||
// Make sure the whole stream has been consumed.
|
// Make sure the whole stream has been consumed.
|
||||||
try!(parser.end());
|
try!(de.end());
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
pub use self::ser::Writer;
|
pub use self::de::{Deserializer, from_str};
|
||||||
pub use self::ser::{to_vec, to_string};
|
pub use self::error::{Error, ErrorCode};
|
||||||
pub use self::ser::escape_str;
|
pub use self::ser::{Serializer, to_vec, to_string, escape_str};
|
||||||
|
pub use self::value::{Value, to_value, from_value};
|
||||||
pub use self::de::from_str;
|
|
||||||
|
|
||||||
pub mod builder;
|
pub mod builder;
|
||||||
pub mod de;
|
pub mod de;
|
||||||
|
@ -3,31 +3,31 @@ use std::io;
|
|||||||
use std::num::{Float, FpCategory};
|
use std::num::{Float, FpCategory};
|
||||||
use std::string::FromUtf8Error;
|
use std::string::FromUtf8Error;
|
||||||
|
|
||||||
use ser::{self, Serializer};
|
use ser;
|
||||||
|
|
||||||
/// A structure for implementing serialization to JSON.
|
/// A structure for implementing serialization to JSON.
|
||||||
pub struct Writer<W> {
|
pub struct Serializer<W> {
|
||||||
writer: W,
|
writer: W,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<W: io::Write> Writer<W> {
|
impl<W: io::Write> Serializer<W> {
|
||||||
/// Creates a new JSON visitr whose output will be written to the writer
|
/// Creates a new JSON visitr whose output will be written to the writer
|
||||||
/// specified.
|
/// specified.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(writer: W) -> Writer<W> {
|
pub fn new(writer: W) -> Serializer<W> {
|
||||||
Writer {
|
Serializer {
|
||||||
writer: writer,
|
writer: writer,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Unwrap the Writer from the Serializer.
|
/// Unwrap the `Writer` from the `Serializer`.
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn into_inner(self) -> W {
|
pub fn into_inner(self) -> W {
|
||||||
self.writer
|
self.writer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<W: io::Write> ser::Serializer for Writer<W> {
|
impl<W: io::Write> ser::Serializer for Serializer<W> {
|
||||||
type Value = ();
|
type Value = ();
|
||||||
type Error = io::Error;
|
type Error = io::Error;
|
||||||
|
|
||||||
@ -278,8 +278,8 @@ pub fn to_writer<W, T>(writer: &mut W, value: &T) -> io::Result<()>
|
|||||||
where W: io::Write,
|
where W: io::Write,
|
||||||
T: ser::Serialize,
|
T: ser::Serialize,
|
||||||
{
|
{
|
||||||
let mut writer = Writer::new(writer);
|
let mut ser = Serializer::new(writer);
|
||||||
try!(writer.visit(value));
|
try!(ser::Serializer::visit(&mut ser, value));
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,9 +1,12 @@
|
|||||||
use std::collections::BTreeMap;
|
use std::collections::{BTreeMap, btree_map};
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::str;
|
use std::str;
|
||||||
|
use std::vec;
|
||||||
|
|
||||||
use ser::{self, Serializer};
|
use de;
|
||||||
|
use ser;
|
||||||
|
use super::error::Error;
|
||||||
|
|
||||||
#[derive(PartialEq)]
|
#[derive(PartialEq)]
|
||||||
pub enum Value {
|
pub enum Value {
|
||||||
@ -22,27 +25,13 @@ impl ser::Serialize for Value {
|
|||||||
V: ser::Visitor,
|
V: ser::Visitor,
|
||||||
>(&self, visitor: &mut V) -> Result<V::Value, V::Error> {
|
>(&self, visitor: &mut V) -> Result<V::Value, V::Error> {
|
||||||
match *self {
|
match *self {
|
||||||
Value::Null => {
|
Value::Null => visitor.visit_unit(),
|
||||||
visitor.visit_unit()
|
Value::Bool(v) => visitor.visit_bool(v),
|
||||||
}
|
Value::I64(v) => visitor.visit_i64(v),
|
||||||
Value::Bool(v) => {
|
Value::F64(v) => visitor.visit_f64(v),
|
||||||
visitor.visit_bool(v)
|
Value::String(ref v) => visitor.visit_str(&v),
|
||||||
}
|
Value::Array(ref v) => v.visit(visitor),
|
||||||
Value::I64(v) => {
|
Value::Object(ref v) => v.visit(visitor),
|
||||||
visitor.visit_i64(v)
|
|
||||||
}
|
|
||||||
Value::F64(v) => {
|
|
||||||
visitor.visit_f64(v)
|
|
||||||
}
|
|
||||||
Value::String(ref v) => {
|
|
||||||
visitor.visit_str(&v)
|
|
||||||
}
|
|
||||||
Value::Array(ref v) => {
|
|
||||||
v.visit(visitor)
|
|
||||||
}
|
|
||||||
Value::Object(ref v) => {
|
|
||||||
v.visit(visitor)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -72,25 +61,19 @@ impl fmt::Debug for Value {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn to_value<T>(value: &T) -> Value where T: ser::Serialize {
|
|
||||||
let mut writer = Writer::new();
|
|
||||||
writer.visit(value).ok().unwrap();
|
|
||||||
writer.unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
enum State {
|
enum State {
|
||||||
Value(Value),
|
Value(Value),
|
||||||
Array(Vec<Value>),
|
Array(Vec<Value>),
|
||||||
Object(BTreeMap<String, Value>),
|
Object(BTreeMap<String, Value>),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Writer {
|
pub struct Serializer {
|
||||||
state: Vec<State>,
|
state: Vec<State>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Writer {
|
impl Serializer {
|
||||||
pub fn new() -> Writer {
|
pub fn new() -> Serializer {
|
||||||
Writer {
|
Serializer {
|
||||||
state: Vec::with_capacity(4),
|
state: Vec::with_capacity(4),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -103,7 +86,7 @@ impl Writer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ser::Serializer for Writer {
|
impl ser::Serializer for Serializer {
|
||||||
type Value = ();
|
type Value = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
@ -116,7 +99,7 @@ impl ser::Serializer for Writer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ser::Visitor for Writer {
|
impl ser::Visitor for Serializer {
|
||||||
type Value = ();
|
type Value = ();
|
||||||
type Error = ();
|
type Error = ();
|
||||||
|
|
||||||
@ -264,3 +247,163 @@ impl ser::Visitor for Writer {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct Deserializer {
|
||||||
|
value: Option<Value>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deserializer {
|
||||||
|
/// Creates a new deserializer instance for deserializing the specified JSON value.
|
||||||
|
pub fn new(value: Value) -> Deserializer {
|
||||||
|
Deserializer {
|
||||||
|
value: Some(value),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl de::Deserializer for Deserializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit<
|
||||||
|
V: de::Visitor,
|
||||||
|
>(&mut self, visitor: &mut V) -> Result<V::Value, Error> {
|
||||||
|
let value = match self.value.take() {
|
||||||
|
Some(value) => value,
|
||||||
|
None => { return Err(de::Error::end_of_stream_error()); }
|
||||||
|
};
|
||||||
|
|
||||||
|
match value {
|
||||||
|
Value::Null => visitor.visit_unit(),
|
||||||
|
Value::Bool(v) => visitor.visit_bool(v),
|
||||||
|
Value::I64(v) => visitor.visit_i64(v),
|
||||||
|
Value::F64(v) => visitor.visit_f64(v),
|
||||||
|
Value::String(v) => visitor.visit_string(v),
|
||||||
|
Value::Array(v) => {
|
||||||
|
let len = v.len();
|
||||||
|
visitor.visit_seq(SeqDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: v.into_iter(),
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Value::Object(v) => {
|
||||||
|
let len = v.len();
|
||||||
|
visitor.visit_map(MapDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: v.into_iter(),
|
||||||
|
value: None,
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_option<
|
||||||
|
V: de::Visitor,
|
||||||
|
>(&mut self, visitor: &mut V) -> Result<V::Value, Error> {
|
||||||
|
match self.value {
|
||||||
|
Some(Value::Null) => visitor.visit_none(),
|
||||||
|
Some(_) => visitor.visit_some(self),
|
||||||
|
None => Err(de::Error::end_of_stream_error()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct SeqDeserializer<'a> {
|
||||||
|
de: &'a mut Deserializer,
|
||||||
|
iter: vec::IntoIter<Value>,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::SeqVisitor for SeqDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some(value) => {
|
||||||
|
self.len -= 1;
|
||||||
|
self.de.value = Some(value);
|
||||||
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
if self.len == 0 {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(de::Error::end_of_stream_error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct MapDeserializer<'a> {
|
||||||
|
de: &'a mut Deserializer,
|
||||||
|
iter: btree_map::IntoIter<String, Value>,
|
||||||
|
value: Option<Value>,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::MapVisitor for MapDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_key<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some((key, value)) => {
|
||||||
|
self.len -= 1;
|
||||||
|
self.value = Some(value);
|
||||||
|
self.de.value = Some(Value::String(key));
|
||||||
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_value<T>(&mut self) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
let value = self.value.take().unwrap();
|
||||||
|
self.de.value = Some(value);
|
||||||
|
Ok(try!(de::Deserialize::deserialize(self.de)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
if self.len == 0 {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(de::Error::end_of_stream_error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Shortcut function to encode a `T` into a JSON `Value`
|
||||||
|
pub fn to_value<T>(value: &T) -> Value
|
||||||
|
where T: ser::Serialize
|
||||||
|
{
|
||||||
|
let mut ser = Serializer::new();
|
||||||
|
ser::Serializer::visit(&mut ser, value).ok().unwrap();
|
||||||
|
ser.unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Shortcut function to decode a JSON `Value` into a `T`
|
||||||
|
pub fn from_value<T>(value: Value) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
let mut de = Deserializer::new(value);
|
||||||
|
de::Deserialize::deserialize(&mut de)
|
||||||
|
}
|
||||||
|
@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
extern crate unicode;
|
extern crate unicode;
|
||||||
|
|
||||||
//pub use ser::{Serialize, Serializer};
|
pub use ser::{Serialize, Serializer};
|
||||||
//pub use ser::{Visitor, VisitorState};
|
pub use de::{Deserialize, Deserializer, Error};
|
||||||
//pub use ser::GatherTokens;
|
|
||||||
|
|
||||||
pub mod ser;
|
pub mod ser;
|
||||||
pub mod de;
|
pub mod de;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user