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
27
28
29
30
31
32
33
34
35
36
37
use proc_macro::TokenStream;
use quote::quote;
use syn::parse_macro_input;
use syn::ItemFn;

#[proc_macro_attribute]
pub fn elapsed(_attr: TokenStream, func: TokenStream) -> TokenStream {
    let func = parse_macro_input!(func as ItemFn);
    let vis = &func.vis;
    let block = &func.block;

    let signature = func.sig;
    let asyncness = &signature.asyncness;
    let constness = &signature.constness;
    let unsafety = &signature.unsafety;
    let abi = &signature.abi;
    let ident = &signature.ident;
    let name_str = ident.to_string();
    let generics = &signature.generics;
    let inputs = &signature.inputs;
    let output = &signature.output;
    let where_clause = &signature.generics.where_clause;

    let new_fn = quote! {

        #vis #constness #asyncness #unsafety #abi fn #ident #generics(#inputs) #output #where_clause {
            use std::time;

            let start = time::Instant::now();
            let fn_return_value = #block;
            println!("fn {} costs {}", #name_str, human_time::human_time(start.elapsed()));
            fn_return_value
        }
    };

    new_fn.into()
}