darling_core/codegen/
from_meta_impl.rs
1use proc_macro2::TokenStream;
2use quote::{quote, ToTokens};
3
4use crate::ast::{Data, Fields, Style};
5use crate::codegen::{Field, OuterFromImpl, TraitImpl, Variant};
6
7pub struct FromMetaImpl<'a> {
8 pub base: TraitImpl<'a>,
9}
10
11impl<'a> ToTokens for FromMetaImpl<'a> {
12 fn to_tokens(&self, tokens: &mut TokenStream) {
13 let base = &self.base;
14
15 let impl_block = match base.data {
16 Data::Struct(ref vd) if vd.style.is_unit() => {
18 let ty_ident = base.ident;
19 quote!(
20 fn from_word() -> ::darling::Result<Self> {
21 ::darling::export::Ok(#ty_ident)
22 }
23 )
24 }
25
26 Data::Struct(Fields {
28 ref fields,
29 style: Style::Tuple,
30 ..
31 }) if fields.len() == 1 => {
32 let ty_ident = base.ident;
33 quote!(
34 fn from_meta(__item: &::darling::export::syn::Meta) -> ::darling::Result<Self> {
35 ::darling::FromMeta::from_meta(__item)
36 .map_err(|e| e.with_span(&__item))
37 .map(#ty_ident)
38 }
39 )
40 }
41 Data::Struct(Fields {
42 style: Style::Tuple,
43 ..
44 }) => {
45 panic!("Multi-field tuples are not supported");
46 }
47 Data::Struct(ref data) => {
48 let inits = data.fields.iter().map(Field::as_initializer);
49 let declare_errors = base.declare_errors();
50 let require_fields = base.require_fields();
51 let check_errors = base.check_errors();
52 let decls = base.local_declarations();
53 let core_loop = base.core_loop();
54 let default = base.fallback_decl();
55 let post_transform = base.post_transform_call();
56
57 quote!(
58 fn from_list(__items: &[::darling::export::NestedMeta]) -> ::darling::Result<Self> {
59
60 #decls
61
62 #declare_errors
63
64 #core_loop
65
66 #require_fields
67
68 #check_errors
69
70 #default
71
72 ::darling::export::Ok(Self {
73 #(#inits),*
74 }) #post_transform
75 }
76 )
77 }
78 Data::Enum(ref variants) => {
79 let unit_arms = variants.iter().map(Variant::as_unit_match_arm);
80
81 let unknown_variant_err = if !variants.is_empty() {
82 let names = variants.iter().map(Variant::as_name);
83 quote! {
84 unknown_field_with_alts(__other, &[#(#names),*])
85 }
86 } else {
87 quote! {
88 unknown_field(__other)
89 }
90 };
91
92 let word_or_err = variants
93 .iter()
94 .find_map(|variant| {
95 if variant.word {
96 let ty_ident = variant.ty_ident;
97 let variant_ident = variant.variant_ident;
98 Some(quote!(::darling::export::Ok(#ty_ident::#variant_ident)))
99 } else {
100 None
101 }
102 })
103 .unwrap_or_else(|| {
104 quote!(::darling::export::Err(
105 ::darling::Error::unsupported_format("word")
106 ))
107 });
108
109 quote!(
110 fn from_list(__outer: &[::darling::export::NestedMeta]) -> ::darling::Result<Self> {
111 match __outer.len() {
114 0 => ::darling::export::Err(::darling::Error::too_few_items(1)),
115 1 => {
116 if let ::darling::export::NestedMeta::Meta(ref __nested) = __outer[0] {
117 match ::darling::util::path_to_string(__nested.path()).as_ref() {
118 #(#variants)*
119 __other => ::darling::export::Err(::darling::Error::#unknown_variant_err.with_span(__nested))
120 }
121 } else {
122 ::darling::export::Err(::darling::Error::unsupported_format("literal"))
123 }
124 }
125 _ => ::darling::export::Err(::darling::Error::too_many_items(1)),
126 }
127 }
128
129 fn from_string(lit: &str) -> ::darling::Result<Self> {
130 match lit {
131 #(#unit_arms)*
132 __other => ::darling::export::Err(::darling::Error::unknown_value(__other))
133 }
134 }
135
136 fn from_word() -> ::darling::Result<Self> {
137 #word_or_err
138 }
139 )
140 }
141 };
142
143 self.wrap(impl_block, tokens);
144 }
145}
146
147impl<'a> OuterFromImpl<'a> for FromMetaImpl<'a> {
148 fn trait_path(&self) -> syn::Path {
149 path!(::darling::FromMeta)
150 }
151
152 fn base(&'a self) -> &'a TraitImpl<'a> {
153 &self.base
154 }
155}