• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 use internals::respan::respan;
2 use proc_macro2::Span;
3 use quote::ToTokens;
4 use std::mem;
5 use syn::punctuated::Punctuated;
6 use syn::{
7     parse_quote, Data, DeriveInput, Expr, ExprPath, GenericArgument, GenericParam, Generics, Macro,
8     Path, PathArguments, QSelf, ReturnType, Type, TypeParamBound, TypePath, WherePredicate,
9 };
10 
replace_receiver(input: &mut DeriveInput)11 pub fn replace_receiver(input: &mut DeriveInput) {
12     let self_ty = {
13         let ident = &input.ident;
14         let ty_generics = input.generics.split_for_impl().1;
15         parse_quote!(#ident #ty_generics)
16     };
17     let mut visitor = ReplaceReceiver(&self_ty);
18     visitor.visit_generics_mut(&mut input.generics);
19     visitor.visit_data_mut(&mut input.data);
20 }
21 
22 struct ReplaceReceiver<'a>(&'a TypePath);
23 
24 impl ReplaceReceiver<'_> {
self_ty(&self, span: Span) -> TypePath25     fn self_ty(&self, span: Span) -> TypePath {
26         let tokens = self.0.to_token_stream();
27         let respanned = respan(tokens, span);
28         syn::parse2(respanned).unwrap()
29     }
30 
self_to_qself(&self, qself: &mut Option<QSelf>, path: &mut Path)31     fn self_to_qself(&self, qself: &mut Option<QSelf>, path: &mut Path) {
32         if path.leading_colon.is_some() || path.segments[0].ident != "Self" {
33             return;
34         }
35 
36         if path.segments.len() == 1 {
37             self.self_to_expr_path(path);
38             return;
39         }
40 
41         let span = path.segments[0].ident.span();
42         *qself = Some(QSelf {
43             lt_token: Token![<](span),
44             ty: Box::new(Type::Path(self.self_ty(span))),
45             position: 0,
46             as_token: None,
47             gt_token: Token![>](span),
48         });
49 
50         path.leading_colon = Some(**path.segments.pairs().next().unwrap().punct().unwrap());
51 
52         let segments = mem::replace(&mut path.segments, Punctuated::new());
53         path.segments = segments.into_pairs().skip(1).collect();
54     }
55 
self_to_expr_path(&self, path: &mut Path)56     fn self_to_expr_path(&self, path: &mut Path) {
57         let self_ty = self.self_ty(path.segments[0].ident.span());
58         let variant = mem::replace(path, self_ty.path);
59         for segment in &mut path.segments {
60             if let PathArguments::AngleBracketed(bracketed) = &mut segment.arguments {
61                 if bracketed.colon2_token.is_none() && !bracketed.args.is_empty() {
62                     bracketed.colon2_token = Some(<Token![::]>::default());
63                 }
64             }
65         }
66         if variant.segments.len() > 1 {
67             path.segments.push_punct(<Token![::]>::default());
68             path.segments.extend(variant.segments.into_pairs().skip(1));
69         }
70     }
71 }
72 
73 impl ReplaceReceiver<'_> {
74     // `Self` -> `Receiver`
visit_type_mut(&mut self, ty: &mut Type)75     fn visit_type_mut(&mut self, ty: &mut Type) {
76         let span = if let Type::Path(node) = ty {
77             if node.qself.is_none() && node.path.is_ident("Self") {
78                 node.path.segments[0].ident.span()
79             } else {
80                 self.visit_type_path_mut(node);
81                 return;
82             }
83         } else {
84             self.visit_type_mut_impl(ty);
85             return;
86         };
87         *ty = self.self_ty(span).into();
88     }
89 
90     // `Self::Assoc` -> `<Receiver>::Assoc`
visit_type_path_mut(&mut self, ty: &mut TypePath)91     fn visit_type_path_mut(&mut self, ty: &mut TypePath) {
92         if ty.qself.is_none() {
93             self.self_to_qself(&mut ty.qself, &mut ty.path);
94         }
95         self.visit_type_path_mut_impl(ty);
96     }
97 
98     // `Self::method` -> `<Receiver>::method`
visit_expr_path_mut(&mut self, expr: &mut ExprPath)99     fn visit_expr_path_mut(&mut self, expr: &mut ExprPath) {
100         if expr.qself.is_none() {
101             self.self_to_qself(&mut expr.qself, &mut expr.path);
102         }
103         self.visit_expr_path_mut_impl(expr);
104     }
105 
106     // Everything below is simply traversing the syntax tree.
107 
visit_type_mut_impl(&mut self, ty: &mut Type)108     fn visit_type_mut_impl(&mut self, ty: &mut Type) {
109         match ty {
110             Type::Array(ty) => {
111                 self.visit_type_mut(&mut ty.elem);
112                 self.visit_expr_mut(&mut ty.len);
113             }
114             Type::BareFn(ty) => {
115                 for arg in &mut ty.inputs {
116                     self.visit_type_mut(&mut arg.ty);
117                 }
118                 self.visit_return_type_mut(&mut ty.output);
119             }
120             Type::Group(ty) => self.visit_type_mut(&mut ty.elem),
121             Type::ImplTrait(ty) => {
122                 for bound in &mut ty.bounds {
123                     self.visit_type_param_bound_mut(bound);
124                 }
125             }
126             Type::Macro(ty) => self.visit_macro_mut(&mut ty.mac),
127             Type::Paren(ty) => self.visit_type_mut(&mut ty.elem),
128             Type::Path(ty) => {
129                 if let Some(qself) = &mut ty.qself {
130                     self.visit_type_mut(&mut qself.ty);
131                 }
132                 self.visit_path_mut(&mut ty.path);
133             }
134             Type::Ptr(ty) => self.visit_type_mut(&mut ty.elem),
135             Type::Reference(ty) => self.visit_type_mut(&mut ty.elem),
136             Type::Slice(ty) => self.visit_type_mut(&mut ty.elem),
137             Type::TraitObject(ty) => {
138                 for bound in &mut ty.bounds {
139                     self.visit_type_param_bound_mut(bound);
140                 }
141             }
142             Type::Tuple(ty) => {
143                 for elem in &mut ty.elems {
144                     self.visit_type_mut(elem);
145                 }
146             }
147 
148             Type::Infer(_) | Type::Never(_) | Type::Verbatim(_) => {}
149 
150             #[cfg_attr(all(test, exhaustive), deny(non_exhaustive_omitted_patterns))]
151             _ => {}
152         }
153     }
154 
visit_type_path_mut_impl(&mut self, ty: &mut TypePath)155     fn visit_type_path_mut_impl(&mut self, ty: &mut TypePath) {
156         if let Some(qself) = &mut ty.qself {
157             self.visit_type_mut(&mut qself.ty);
158         }
159         self.visit_path_mut(&mut ty.path);
160     }
161 
visit_expr_path_mut_impl(&mut self, expr: &mut ExprPath)162     fn visit_expr_path_mut_impl(&mut self, expr: &mut ExprPath) {
163         if let Some(qself) = &mut expr.qself {
164             self.visit_type_mut(&mut qself.ty);
165         }
166         self.visit_path_mut(&mut expr.path);
167     }
168 
visit_path_mut(&mut self, path: &mut Path)169     fn visit_path_mut(&mut self, path: &mut Path) {
170         for segment in &mut path.segments {
171             self.visit_path_arguments_mut(&mut segment.arguments);
172         }
173     }
174 
visit_path_arguments_mut(&mut self, arguments: &mut PathArguments)175     fn visit_path_arguments_mut(&mut self, arguments: &mut PathArguments) {
176         match arguments {
177             PathArguments::None => {}
178             PathArguments::AngleBracketed(arguments) => {
179                 for arg in &mut arguments.args {
180                     match arg {
181                         GenericArgument::Type(arg) => self.visit_type_mut(arg),
182                         GenericArgument::Binding(arg) => self.visit_type_mut(&mut arg.ty),
183                         GenericArgument::Lifetime(_)
184                         | GenericArgument::Constraint(_)
185                         | GenericArgument::Const(_) => {}
186                     }
187                 }
188             }
189             PathArguments::Parenthesized(arguments) => {
190                 for argument in &mut arguments.inputs {
191                     self.visit_type_mut(argument);
192                 }
193                 self.visit_return_type_mut(&mut arguments.output);
194             }
195         }
196     }
197 
visit_return_type_mut(&mut self, return_type: &mut ReturnType)198     fn visit_return_type_mut(&mut self, return_type: &mut ReturnType) {
199         match return_type {
200             ReturnType::Default => {}
201             ReturnType::Type(_, output) => self.visit_type_mut(output),
202         }
203     }
204 
visit_type_param_bound_mut(&mut self, bound: &mut TypeParamBound)205     fn visit_type_param_bound_mut(&mut self, bound: &mut TypeParamBound) {
206         match bound {
207             TypeParamBound::Trait(bound) => self.visit_path_mut(&mut bound.path),
208             TypeParamBound::Lifetime(_) => {}
209         }
210     }
211 
visit_generics_mut(&mut self, generics: &mut Generics)212     fn visit_generics_mut(&mut self, generics: &mut Generics) {
213         for param in &mut generics.params {
214             match param {
215                 GenericParam::Type(param) => {
216                     for bound in &mut param.bounds {
217                         self.visit_type_param_bound_mut(bound);
218                     }
219                 }
220                 GenericParam::Lifetime(_) | GenericParam::Const(_) => {}
221             }
222         }
223         if let Some(where_clause) = &mut generics.where_clause {
224             for predicate in &mut where_clause.predicates {
225                 match predicate {
226                     WherePredicate::Type(predicate) => {
227                         self.visit_type_mut(&mut predicate.bounded_ty);
228                         for bound in &mut predicate.bounds {
229                             self.visit_type_param_bound_mut(bound);
230                         }
231                     }
232                     WherePredicate::Lifetime(_) | WherePredicate::Eq(_) => {}
233                 }
234             }
235         }
236     }
237 
visit_data_mut(&mut self, data: &mut Data)238     fn visit_data_mut(&mut self, data: &mut Data) {
239         match data {
240             Data::Struct(data) => {
241                 for field in &mut data.fields {
242                     self.visit_type_mut(&mut field.ty);
243                 }
244             }
245             Data::Enum(data) => {
246                 for variant in &mut data.variants {
247                     for field in &mut variant.fields {
248                         self.visit_type_mut(&mut field.ty);
249                     }
250                 }
251             }
252             Data::Union(_) => {}
253         }
254     }
255 
visit_expr_mut(&mut self, expr: &mut Expr)256     fn visit_expr_mut(&mut self, expr: &mut Expr) {
257         match expr {
258             Expr::Binary(expr) => {
259                 self.visit_expr_mut(&mut expr.left);
260                 self.visit_expr_mut(&mut expr.right);
261             }
262             Expr::Call(expr) => {
263                 self.visit_expr_mut(&mut expr.func);
264                 for arg in &mut expr.args {
265                     self.visit_expr_mut(arg);
266                 }
267             }
268             Expr::Cast(expr) => {
269                 self.visit_expr_mut(&mut expr.expr);
270                 self.visit_type_mut(&mut expr.ty);
271             }
272             Expr::Field(expr) => self.visit_expr_mut(&mut expr.base),
273             Expr::Index(expr) => {
274                 self.visit_expr_mut(&mut expr.expr);
275                 self.visit_expr_mut(&mut expr.index);
276             }
277             Expr::Paren(expr) => self.visit_expr_mut(&mut expr.expr),
278             Expr::Path(expr) => self.visit_expr_path_mut(expr),
279             Expr::Unary(expr) => self.visit_expr_mut(&mut expr.expr),
280             _ => {}
281         }
282     }
283 
visit_macro_mut(&mut self, _mac: &mut Macro)284     fn visit_macro_mut(&mut self, _mac: &mut Macro) {}
285 }
286