educe/trait_handlers/partial_eq/models/
field_attribute.rs1use syn::{punctuated::Punctuated, Attribute, Meta, Path, Token};
2
3use crate::{
4 common::{
5 ident_bool::{meta_2_bool_allow_path, meta_name_value_2_bool},
6 path::meta_2_path,
7 },
8 panic,
9 supported_traits::Trait,
10};
11
12pub(crate) struct FieldAttribute {
13 pub(crate) ignore: bool,
14 pub(crate) method: Option<Path>,
15}
16
17pub(crate) struct FieldAttributeBuilder {
18 pub(crate) enable_ignore: bool,
19 pub(crate) enable_method: bool,
20}
21
22impl FieldAttributeBuilder {
23 pub(crate) fn build_from_partial_eq_meta(&self, meta: &Meta) -> syn::Result<FieldAttribute> {
24 debug_assert!(meta.path().is_ident("PartialEq") || meta.path().is_ident("Eq"));
25
26 let mut ignore = false;
27 let mut method = None;
28
29 let correct_usage_for_partial_eq_attribute = {
30 let mut usage = vec![];
31
32 if self.enable_ignore {
33 usage.push(stringify!(#[educe(PartialEq = false)]));
34 usage.push(stringify!(#[educe(PartialEq(ignore))]));
35 }
36
37 if self.enable_method {
38 usage.push(stringify!(#[educe(PartialEq(method(path_to_method)))]));
39 }
40
41 usage
42 };
43
44 match meta {
45 Meta::Path(_) => {
46 return Err(panic::attribute_incorrect_format(
47 meta.path().get_ident().unwrap(),
48 &correct_usage_for_partial_eq_attribute,
49 ));
50 },
51 Meta::NameValue(name_value) => {
52 if self.enable_ignore {
53 ignore = !meta_name_value_2_bool(name_value)?;
54 } else {
55 return Err(panic::attribute_incorrect_format(
56 meta.path().get_ident().unwrap(),
57 &correct_usage_for_partial_eq_attribute,
58 ));
59 }
60 },
61 Meta::List(list) => {
62 let result =
63 list.parse_args_with(Punctuated::<Meta, Token![,]>::parse_terminated)?;
64
65 let mut ignore_is_set = false;
66 let mut method_is_set = false;
67
68 let mut handler = |meta: Meta| -> syn::Result<bool> {
69 if let Some(ident) = meta.path().get_ident() {
70 match ident.to_string().as_str() {
71 "ignore" => {
72 if !self.enable_ignore {
73 return Ok(false);
74 }
75
76 let v = meta_2_bool_allow_path(&meta)?;
77
78 if ignore_is_set {
79 return Err(panic::parameter_reset(ident));
80 }
81
82 ignore_is_set = true;
83
84 ignore = v;
85
86 return Ok(true);
87 },
88 "method" => {
89 if !self.enable_method {
90 return Ok(false);
91 }
92
93 let v = meta_2_path(&meta)?;
94
95 if method_is_set {
96 return Err(panic::parameter_reset(ident));
97 }
98
99 method_is_set = true;
100
101 method = Some(v);
102
103 return Ok(true);
104 },
105 _ => (),
106 }
107 }
108
109 Ok(false)
110 };
111
112 for p in result {
113 if !handler(p)? {
114 return Err(panic::attribute_incorrect_format(
115 meta.path().get_ident().unwrap(),
116 &correct_usage_for_partial_eq_attribute,
117 ));
118 }
119 }
120 },
121 }
122
123 Ok(FieldAttribute {
124 ignore,
125 method,
126 })
127 }
128
129 pub(crate) fn build_from_attributes(
130 &self,
131 attributes: &[Attribute],
132 traits: &[Trait],
133 ) -> syn::Result<FieldAttribute> {
134 let mut output = None;
135
136 for attribute in attributes.iter() {
137 let path = attribute.path();
138
139 if path.is_ident("educe") {
140 if let Meta::List(list) = &attribute.meta {
141 let result =
142 list.parse_args_with(Punctuated::<Meta, Token![,]>::parse_terminated)?;
143
144 for meta in result {
145 let path = meta.path();
146
147 let t = match Trait::from_path(path) {
148 Some(t) => t,
149 None => return Err(panic::unsupported_trait(meta.path())),
150 };
151
152 if !traits.contains(&t) {
153 return Err(panic::trait_not_used(path.get_ident().unwrap()));
154 }
155
156 if t == Trait::PartialEq {
157 if output.is_some() {
158 return Err(panic::reuse_a_trait(path.get_ident().unwrap()));
159 }
160
161 output = Some(self.build_from_partial_eq_meta(&meta)?);
162 }
163
164 #[cfg(feature = "Eq")]
165 if traits.contains(&Trait::Eq) && t == Trait::Eq {
166 if output.is_some() {
167 return Err(panic::reuse_a_trait(path.get_ident().unwrap()));
168 }
169
170 output = Some(self.build_from_partial_eq_meta(&meta)?);
171 }
172 }
173 }
174 }
175 }
176
177 Ok(output.unwrap_or(FieldAttribute {
178 ignore: false, method: None
179 }))
180 }
181}