1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
use proc_macro::TokenStream;
use syn::{parse_macro_input, ItemImpl, ImplItem, Generics};
use quote::quote;

#[proc_macro_attribute]
pub fn implicit_trait(_args: TokenStream, input: TokenStream) -> TokenStream {
    let item = parse_macro_input!(input as ItemImpl);
    let trait_name = &item.trait_.as_ref().unwrap().1;
    let declarations = item.items.iter().map(|item| {
        match item {
            ImplItem::Method(method) => {
                let sig = &method.sig;
                quote! {
                    fn #sig;
                }
            }
            _ => panic!("Only methods are allowed in an implicit trait"),
        }
    });
    (quote! {
        pub trait #trait_name {
            #(#declarations)*
        }
        #item
    }).into()
}