1extern crate proc_macro;
2
3use proc_macro::TokenStream;
4use quote::quote;
5
6use syn::*;
7
8#[proc_macro_attribute]
9pub fn clonable(_attrs: TokenStream, item: TokenStream) -> TokenStream {
10 let mut item_trait = parse_macro_input!(item as ItemTrait);
11
12 let item_trait_ident = &item_trait.ident;
13
14 let cloneish_paths = &[quote!(Clone), quote!(std::clone::Clone), quote!(::std::clone::Clone)];
15
16 if let Some(path) = item_trait
17 .supertraits
18 .iter_mut()
19 .filter_map(|x| match x {
20 TypeParamBound::Trait(y) => Some(y),
21 _ => None
22 })
23 .map(|x| &mut x.path)
24 .find(|x| {
25 let s = quote!(#x).to_string();
26 cloneish_paths.iter().any(|y| y.to_string() == s)
27 })
28 {
29 *path = parse_quote!(dyn_clonable::dyn_clone::DynClone);
30 } else {
31 panic!("`Clone` must be present in trait supertrait list");
32 }
33
34 let (impl_generics, ty_generics, where_clause) = item_trait.generics.split_for_impl();
35
36 (quote! {
37 #item_trait
38 dyn_clonable::dyn_clone::clone_trait_object!(#impl_generics #item_trait_ident #ty_generics #where_clause);
39 })
40 .into()
41}