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
use proc_macro2::TokenStream;
use quote::quote;

fn is_reflect_ignore(attribute: &syn::Attribute) -> bool {
    if !attribute.path().is_ident("reflect") {
        return false;
    }

    let mut ignore = false;
    let _ = attribute.parse_nested_meta(|meta| {
        ignore = meta.path.is_ident("ignore");
        Ok(())
    });
    ignore
}
pub fn is_reflect_ignore_field(field: &syn::Field) -> bool {
    field.attrs.iter().any(is_reflect_ignore)
}

pub enum InspectorAttribute {
    Assignment(syn::Member, syn::Expr),
    Tag(syn::Member),
}

impl InspectorAttribute {
    pub fn lhs(&self) -> &syn::Member {
        match self {
            InspectorAttribute::Assignment(member, _) => member,
            InspectorAttribute::Tag(member) => member,
        }
    }

    pub fn rhs(&self) -> TokenStream {
        match self {
            InspectorAttribute::Assignment(_, expr) => quote! { #expr },
            InspectorAttribute::Tag(_) => quote! { true },
        }
    }
}

fn parse_inspectable_attributes(
    input: syn::parse::ParseStream,
) -> syn::Result<impl Iterator<Item = InspectorAttribute>> {
    let parse_attribute = |input: syn::parse::ParseStream| {
        let ident: syn::Member = input.parse()?;
        if input.peek(syn::Token![=]) {
            let _eq_token: syn::Token![=] = input.parse()?;
            let expr: syn::Expr = input.parse()?;
            Ok(InspectorAttribute::Assignment(ident, expr))
        } else {
            Ok(InspectorAttribute::Tag(ident))
        }
    };

    input
        .parse_terminated(parse_attribute, syn::Token![,])
        .map(IntoIterator::into_iter)
}

pub fn extract_inspector_attributes(
    attrs: &[syn::Attribute],
) -> syn::Result<Vec<InspectorAttribute>> {
    Ok(attrs
        .iter()
        .filter(|attr| attr.path().get_ident().map_or(false, |p| p == "inspector"))
        .map(|attr| attr.parse_args_with(parse_inspectable_attributes))
        .collect::<syn::Result<Vec<_>>>()?
        .into_iter()
        .flatten()
        .collect())
}