Replace some aster with quasi in serde_macros
This commit is contained in:
parent
9ca1e2a8f7
commit
61317f5935
@ -17,7 +17,6 @@ use syntax::ast::{
|
|||||||
EnumDef,
|
EnumDef,
|
||||||
};
|
};
|
||||||
use syntax::ast;
|
use syntax::ast;
|
||||||
use syntax::ast_util;
|
|
||||||
use syntax::codemap::{Span, respan};
|
use syntax::codemap::{Span, respan};
|
||||||
use syntax::ext::base::{ExtCtxt, Decorator, ItemDecorator};
|
use syntax::ext::base::{ExtCtxt, Decorator, ItemDecorator};
|
||||||
use syntax::ext::build::AstBuilder;
|
use syntax::ext::build::AstBuilder;
|
||||||
@ -178,7 +177,6 @@ fn serialize_substructure(
|
|||||||
(&ast::ItemEnum(_, ref generics), &EnumMatching(_idx, variant, ref fields)) => {
|
(&ast::ItemEnum(_, ref generics), &EnumMatching(_idx, variant, ref fields)) => {
|
||||||
serialize_enum(
|
serialize_enum(
|
||||||
cx,
|
cx,
|
||||||
span,
|
|
||||||
&builder,
|
&builder,
|
||||||
serializer,
|
serializer,
|
||||||
substr.type_ident,
|
substr.type_ident,
|
||||||
@ -362,7 +360,6 @@ fn serialize_struct(
|
|||||||
|
|
||||||
fn serialize_enum(
|
fn serialize_enum(
|
||||||
cx: &ExtCtxt,
|
cx: &ExtCtxt,
|
||||||
span: Span,
|
|
||||||
builder: &aster::AstBuilder,
|
builder: &aster::AstBuilder,
|
||||||
serializer: P<Expr>,
|
serializer: P<Expr>,
|
||||||
type_ident: Ident,
|
type_ident: Ident,
|
||||||
@ -383,7 +380,6 @@ fn serialize_enum(
|
|||||||
} else {
|
} else {
|
||||||
serialize_variant(
|
serialize_variant(
|
||||||
cx,
|
cx,
|
||||||
span,
|
|
||||||
builder,
|
builder,
|
||||||
serializer,
|
serializer,
|
||||||
type_name,
|
type_name,
|
||||||
@ -396,7 +392,6 @@ fn serialize_enum(
|
|||||||
|
|
||||||
fn serialize_variant(
|
fn serialize_variant(
|
||||||
cx: &ExtCtxt,
|
cx: &ExtCtxt,
|
||||||
span: Span,
|
|
||||||
builder: &aster::AstBuilder,
|
builder: &aster::AstBuilder,
|
||||||
serializer: P<ast::Expr>,
|
serializer: P<ast::Expr>,
|
||||||
type_name: P<ast::Expr>,
|
type_name: P<ast::Expr>,
|
||||||
@ -405,7 +400,7 @@ fn serialize_variant(
|
|||||||
variant: &ast::Variant,
|
variant: &ast::Variant,
|
||||||
fields: &[FieldInfo],
|
fields: &[FieldInfo],
|
||||||
) -> P<Expr> {
|
) -> P<Expr> {
|
||||||
let generics = builder.from_generics(generics.clone())
|
let trait_generics = builder.from_generics(generics.clone())
|
||||||
.add_lifetime_bound("'__a")
|
.add_lifetime_bound("'__a")
|
||||||
.add_ty_param_bound(
|
.add_ty_param_bound(
|
||||||
builder.path().global().ids(&["serde", "ser", "Serialize"]).build()
|
builder.path().global().ids(&["serde", "ser", "Serialize"]).build()
|
||||||
@ -413,11 +408,13 @@ fn serialize_variant(
|
|||||||
.lifetime_name("'__a")
|
.lifetime_name("'__a")
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
let (
|
let where_clause = &generics.where_clause;
|
||||||
trait_name,
|
|
||||||
visitor_method_name,
|
let type_generics = builder.from_generics(trait_generics.clone())
|
||||||
tys,
|
.strip_bounds()
|
||||||
): (Ident, Ident, Vec<P<ast::Ty>>) = match variant.node.kind {
|
.build();
|
||||||
|
|
||||||
|
let (trait_name, method, tys,): (_, _, Vec<_>) = match variant.node.kind {
|
||||||
ast::TupleVariantKind(ref args) => {
|
ast::TupleVariantKind(ref args) => {
|
||||||
(
|
(
|
||||||
cx.ident_of("SeqVisitor"),
|
cx.ident_of("SeqVisitor"),
|
||||||
@ -449,32 +446,21 @@ fn serialize_variant(
|
|||||||
}))
|
}))
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
let visitor_ident = builder.id("__Visitor");
|
let value_expr = builder.expr().tuple()
|
||||||
|
.with_exprs(
|
||||||
let visitor_struct = builder.item().struct_(visitor_ident)
|
fields.iter().map(|field| {
|
||||||
.generics().with(generics.clone()).build()
|
builder.expr()
|
||||||
.field("state").usize()
|
.addr_of()
|
||||||
.field("value").build(value_ty)
|
.build(field.self_.clone())
|
||||||
|
})
|
||||||
|
)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
let visitor_expr = builder.expr().struct_path(visitor_ident)
|
|
||||||
.field("state").usize(0)
|
|
||||||
.field("value").tuple()
|
|
||||||
.with_exprs(
|
|
||||||
fields.iter().map(|field| {
|
|
||||||
builder.expr()
|
|
||||||
.addr_of()
|
|
||||||
.build(field.self_.clone())
|
|
||||||
})
|
|
||||||
)
|
|
||||||
.build()
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let mut first = true;
|
|
||||||
|
|
||||||
let visitor_arms: Vec<ast::Arm> = fields.iter()
|
let visitor_arms: Vec<ast::Arm> = fields.iter()
|
||||||
.enumerate()
|
.enumerate()
|
||||||
.map(|(state, field)| {
|
.map(|(state, field)| {
|
||||||
|
let first = state == 0;
|
||||||
|
|
||||||
let field_expr = builder.expr()
|
let field_expr = builder.expr()
|
||||||
.tup_field(state)
|
.tup_field(state)
|
||||||
.field("value").self_();
|
.field("value").self_();
|
||||||
@ -503,8 +489,6 @@ fn serialize_variant(
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
first = false;
|
|
||||||
|
|
||||||
quote_arm!(cx,
|
quote_arm!(cx,
|
||||||
$state => {
|
$state => {
|
||||||
self.state += 1;
|
self.state += 1;
|
||||||
@ -514,70 +498,36 @@ fn serialize_variant(
|
|||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let trait_path = builder.path()
|
|
||||||
.global()
|
|
||||||
.ids(&["serde", "ser"]).id(trait_name)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let trait_ref = cx.trait_ref(trait_path);
|
|
||||||
let opt_trait_ref = Some(trait_ref);
|
|
||||||
|
|
||||||
let self_ty = builder.ty()
|
|
||||||
.path()
|
|
||||||
.segment("__Visitor")
|
|
||||||
.with_generics(generics.clone())
|
|
||||||
.build()
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let len = fields.len();
|
let len = fields.len();
|
||||||
let impl_ident = ast_util::impl_pretty_name(&opt_trait_ref, Some(&self_ty));
|
|
||||||
|
|
||||||
let methods = vec![
|
|
||||||
ast::MethodImplItem(
|
|
||||||
quote_method!(cx,
|
|
||||||
fn visit<V>(&mut self, serializer: &mut V) -> Result<Option<()>, V::Error>
|
|
||||||
where V: ::serde::ser::Serializer,
|
|
||||||
{
|
|
||||||
match self.state {
|
|
||||||
$visitor_arms
|
|
||||||
_ => Ok(None),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
),
|
|
||||||
|
|
||||||
ast::MethodImplItem(
|
|
||||||
quote_method!(cx,
|
|
||||||
fn len(&self) -> Option<usize> {
|
|
||||||
Some($len)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
),
|
|
||||||
];
|
|
||||||
|
|
||||||
let visitor_impl = cx.item(
|
|
||||||
span,
|
|
||||||
impl_ident,
|
|
||||||
vec![],
|
|
||||||
ast::ItemImpl(
|
|
||||||
ast::Unsafety::Normal,
|
|
||||||
ast::ImplPolarity::Positive,
|
|
||||||
generics,
|
|
||||||
opt_trait_ref,
|
|
||||||
self_ty,
|
|
||||||
methods,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
quote_expr!(cx, {
|
quote_expr!(cx, {
|
||||||
$visitor_struct
|
struct __Visitor $trait_generics $where_clause {
|
||||||
$visitor_impl
|
state: usize,
|
||||||
|
value: $value_ty,
|
||||||
|
}
|
||||||
|
|
||||||
::serde::ser::Serializer::$visitor_method_name(
|
impl $trait_generics ::serde::ser::$trait_name for __Visitor $type_generics $where_clause {
|
||||||
|
fn visit<V>(&mut self, serializer: &mut V) -> Result<Option<()>, V::Error>
|
||||||
|
where V: ::serde::ser::Serializer,
|
||||||
|
{
|
||||||
|
match self.state {
|
||||||
|
$visitor_arms
|
||||||
|
_ => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn len(&self) -> Option<usize> {
|
||||||
|
Some($len)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
::serde::ser::Serializer::$method(
|
||||||
$serializer,
|
$serializer,
|
||||||
$type_name,
|
$type_name,
|
||||||
$variant_name,
|
$variant_name,
|
||||||
$visitor_expr,
|
__Visitor {
|
||||||
|
state: 0,
|
||||||
|
value: $value_expr,
|
||||||
|
},
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -789,12 +739,18 @@ fn deserialize_struct_unnamed_fields(
|
|||||||
state: P<ast::Expr>,
|
state: P<ast::Expr>,
|
||||||
generics: &ast::Generics,
|
generics: &ast::Generics,
|
||||||
) -> P<ast::Expr> {
|
) -> P<ast::Expr> {
|
||||||
let visitor_impl_generics = builder.from_generics(generics.clone())
|
let trait_generics = builder.from_generics(generics.clone())
|
||||||
.add_ty_param_bound(
|
.add_ty_param_bound(
|
||||||
builder.path().global().ids(&["serde", "de", "Deserialize"]).build()
|
builder.path().global().ids(&["serde", "de", "Deserialize"]).build()
|
||||||
)
|
)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
|
let where_clause = &trait_generics.where_clause;
|
||||||
|
|
||||||
|
let type_generics = builder.from_generics(trait_generics.clone())
|
||||||
|
.strip_bounds()
|
||||||
|
.build();
|
||||||
|
|
||||||
let field_names: Vec<ast::Ident> = (0 .. fields.len())
|
let field_names: Vec<ast::Ident> = (0 .. fields.len())
|
||||||
.map(|i| builder.id(&format!("__field{}", i)))
|
.map(|i| builder.id(&format!("__field{}", i)))
|
||||||
.collect();
|
.collect();
|
||||||
@ -835,10 +791,6 @@ fn deserialize_struct_unnamed_fields(
|
|||||||
|
|
||||||
let struct_name = builder.expr().str(struct_ident);
|
let struct_name = builder.expr().str(struct_ident);
|
||||||
|
|
||||||
let visitor_ty = builder.ty().path()
|
|
||||||
.segment("__Visitor").with_generics(generics.clone()).build()
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let value_ty = builder.ty().path()
|
let value_ty = builder.ty().path()
|
||||||
.segment(type_ident).with_generics(generics.clone()).build()
|
.segment(type_ident).with_generics(generics.clone()).build()
|
||||||
.build();
|
.build();
|
||||||
@ -846,7 +798,7 @@ fn deserialize_struct_unnamed_fields(
|
|||||||
quote_expr!(cx, {
|
quote_expr!(cx, {
|
||||||
$visitor_struct;
|
$visitor_struct;
|
||||||
|
|
||||||
impl $visitor_impl_generics ::serde::de::Visitor for $visitor_ty {
|
impl $trait_generics ::serde::de::Visitor for __Visitor $type_generics $where_clause {
|
||||||
type Value = $value_ty;
|
type Value = $value_ty;
|
||||||
|
|
||||||
fn visit_seq<__V>(&mut self, mut visitor: __V) -> Result<$value_ty, __V::Error>
|
fn visit_seq<__V>(&mut self, mut visitor: __V) -> Result<$value_ty, __V::Error>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user