Support more unusual reference types in Outgoing derive
This commit is contained in:
		
							parent
							
								
									16dd20d64e
								
							
						
					
					
						commit
						3853a36ff6
					
				| @ -240,37 +240,43 @@ fn strip_lifetimes(field_type: &mut Type) -> bool { | ||||
| 
 | ||||
|             has_lifetimes || is_lifetime_generic | ||||
|         } | ||||
|         Type::Reference(TypeReference { elem, .. }) => match &mut **elem { | ||||
|             Type::Path(ty_path) => { | ||||
|                 let TypePath { path, .. } = ty_path; | ||||
|                 let segs = path | ||||
|                     .segments | ||||
|                     .clone() | ||||
|                     .into_iter() | ||||
|                     .map(|seg| seg.ident.to_string()) | ||||
|                     .collect::<Vec<_>>(); | ||||
|         Type::Reference(TypeReference { elem, .. }) => { | ||||
|             let special_replacement = match &mut **elem { | ||||
|                 Type::Path(ty) => { | ||||
|                     let path = &ty.path; | ||||
|                     let last_seg = path.segments.last().unwrap(); | ||||
| 
 | ||||
|                 if path.is_ident("str") { | ||||
|                     if last_seg.ident == "str" { | ||||
|                         // &str -> String
 | ||||
|                     *field_type = parse_quote! { ::std::string::String }; | ||||
|                 } else if segs.contains(&"DeviceId".into()) || segs.contains(&"ServerName".into()) { | ||||
|                         Some(parse_quote! { ::std::string::String }) | ||||
|                     } else if last_seg.ident == "DeviceId" || last_seg.ident == "ServerName" { | ||||
|                         // The identifiers that need to be boxed `Box<T>` since they are DST's.
 | ||||
|                     *field_type = parse_quote! { ::std::boxed::Box<#path> }; | ||||
|                         Some(parse_quote! { ::std::boxed::Box<#path> }) | ||||
|                     } else { | ||||
|                     // &T -> T
 | ||||
|                     *field_type = Type::Path(ty_path.clone()); | ||||
|                         None | ||||
|                     } | ||||
|                 true | ||||
|                 } | ||||
|                 // &[T] -> Vec<T>
 | ||||
|                 Type::Slice(TypeSlice { elem, .. }) => { | ||||
|                     // Recursively strip the lifetimes of the slice's elements.
 | ||||
|                     strip_lifetimes(&mut *elem); | ||||
|                 *field_type = parse_quote! { Vec<#elem> }; | ||||
|                     Some(parse_quote! { Vec<#elem> }) | ||||
|                 } | ||||
|                 _ => None, | ||||
|             }; | ||||
| 
 | ||||
|             *field_type = match special_replacement { | ||||
|                 Some(ty) => ty, | ||||
|                 None => { | ||||
|                     // Strip lifetimes of `elem`.
 | ||||
|                     strip_lifetimes(elem); | ||||
|                     // Replace reference with `elem`.
 | ||||
|                     (**elem).clone() | ||||
|                 } | ||||
|             }; | ||||
| 
 | ||||
|             true | ||||
|         } | ||||
|             _ => false, | ||||
|         }, | ||||
|         Type::Tuple(syn::TypeTuple { elems, .. }) => { | ||||
|             let mut has_lifetime = false; | ||||
|             for elem in elems { | ||||
|  | ||||
| @ -32,6 +32,9 @@ pub struct FakeRequest<'a, T> { | ||||
|     pub option: Option<&'a [u8]>, | ||||
|     pub depth: Option<&'a [(&'a str, &'a str)]>, | ||||
|     pub arc_type: std::sync::Arc<&'a ::ruma_identifiers::ServerName>, | ||||
|     pub thing_ref: &'a Thing<'a, T>, | ||||
|     pub double_ref: &'a &'a u8, | ||||
|     pub triple_ref: &'a &'a &'a str, | ||||
| } | ||||
| 
 | ||||
| #[derive(Outgoing)] | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user