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
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
use proc_macro2::{Ident, TokenStream};
use quote::quote;
use syn::{parse::Parse, Expr, ItemFn, Token};

pub fn parse(item: ItemFn, attr: TokenStream) -> syn::Result<TokenStream> {
    // grab threads if specified
    let MetricList { ptr, fn_name, data, scope } = syn::parse2(attr)?;

    let attrs = item.attrs;
    let vis = item.vis;
    let sig = item.sig;
    let block = item.block;

    if scope {
        Ok(quote!(
            #(#attrs)*
            #vis
            #sig
            {
                if let Some(metrics) = self.#ptr.clone() {
                    metrics.#fn_name(#(#data),*, || #block)
                } else {
                    #block
                }
            }
        ))
    } else {
        Ok(quote!(
            #(#attrs)*
            #vis
            #sig
            {
                let result = #block;
                self.#ptr.as_ref().inspect(|m| m.#fn_name(#(#data),*));
                result
            }
        ))
    }
}

pub struct MetricList {
    // ptr to metric in struct
    ptr:     Ident,
    scope:   bool,
    // recorder name
    fn_name: Ident,
    data:    Vec<Expr>,
}

impl Parse for MetricList {
    fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
        let ptr: Ident = input.parse()?;
        if ptr != "ptr" {
            return Err(syn::Error::new(ptr.span(), "first field must be ptr=location"))
        }
        input.parse::<Token![=]>()?;
        let ptr_value: Ident = input.parse()?;

        input.parse::<Token![,]>()?;
        let scope: Ident = input.parse()?;

        let (fn_name, scope) = if scope == "scope" {
            input.parse::<Token![,]>()?;
            let fn_name: Ident = input.parse()?;
            (fn_name, true)
        } else {
            (scope, false)
        };

        let mut data = Vec::new();
        // take out all args
        while input.peek(Token![,]) {
            input.parse::<Token![,]>()?;
            data.push(input.parse()?);
        }

        Ok(Self { ptr: ptr_value, fn_name, data, scope })
    }
}