1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
extern crate syn;
#[macro_use]
extern crate quote;
extern crate proc_macro;
extern crate proc_macro2;
use proc_macro::TokenStream;
use proc_macro2::TokenStream as TokenStream2;
#[doc(hidden)]
#[proc_macro_derive(FromValue)]
pub fn derive_from_value(input: TokenStream) -> TokenStream {
let ast = syn::parse::<syn::DeriveInput>(input).unwrap();
let name = &ast.ident;
let body = match ast.data {
syn::Data::Struct(ref s) => impl_derive_from_value_struct(&name, s),
syn::Data::Enum(ref e) => impl_derive_from_value_enum(&name, e),
_ => unreachable!(),
};
let gen = quote! {
impl FromValue for #name {
fn from_value(mut v: Value) -> Result<Self, String> {
#body
}
}
};
gen.into()
}
fn impl_derive_from_value_struct(name: &syn::Ident, data: &syn::DataStruct) -> TokenStream2 {
match data.fields {
syn::Fields::Named(_) => impl_derive_from_value_struct_named(name, &data.fields),
syn::Fields::Unnamed(_) => impl_derive_from_value_struct_unnamed(name, &data.fields),
syn::Fields::Unit => impl_derive_from_value_struct_unit(name),
}
}
fn impl_derive_from_value_struct_named(name: &syn::Ident, fields: &syn::Fields) -> TokenStream2 {
let fields: TokenStream2 = fields
.iter()
.fold(quote!{}, |q, f| {
let n = f.ident.as_ref().unwrap();
quote! {
#q
#n: table.remove(stringify!(#n)).map_or_else(
|| convert_no_value()
.map_err(|e| format!("Cannot convert to {}. Field {}: {}", stringify!(#name), stringify!(#n), e)),
convert
)?,
}
}).into();
quote! {
#[inline]
fn convert<T: FromValue>(v: Value) -> Result<T, String> {
T::from_value(v)
}
#[inline]
fn convert_no_value<T: FromValue>() -> Result<T, String> {
T::from_no_value()
}
let table = v.as_table_mut().ok_or_else(
|| format!("Cannot convert to {}: not a Table", stringify!(#name))
)?;
Ok( #name {
#fields
})
}
}
fn impl_derive_from_value_struct_unnamed(
_name: &syn::Ident,
_fields: &syn::Fields,
) -> TokenStream2 {
unimplemented!()
}
fn impl_derive_from_value_struct_unit(_name: &syn::Ident) -> TokenStream2 {
unimplemented!()
}
fn impl_derive_from_value_enum(_name: &syn::Ident, _data: &syn::DataEnum) -> TokenStream2 {
unimplemented!()
}