1
1
//! Macros `BonsaiDb`.
2

            
3
#![forbid(unsafe_code)]
4
#![warn(
5
    clippy::cargo,
6
    missing_docs,
7
    // clippy::missing_docs_in_private_items,
8
    clippy::nursery,
9
    clippy::pedantic,
10
    future_incompatible,
11
    rust_2018_idioms,
12
)]
13
#![allow(clippy::option_if_let_else)]
14

            
15
use proc_macro::TokenStream;
16
use proc_macro_error::{abort, abort_call_site, proc_macro_error};
17
use quote::quote;
18
use syn::{parse_macro_input, Data, DeriveInput};
19

            
20
/// Derives the `bonsaidb_core::permissions::Action` trait.
21
#[proc_macro_error]
22
#[proc_macro_derive(Action)]
23
pub fn permissions_action_derive(input: TokenStream) -> TokenStream {
24
    let input = parse_macro_input!(input as DeriveInput);
25

            
26
    let name = input.ident;
27

            
28
    let mut fields = Vec::new();
29
    match input.data {
30
        Data::Enum(data) => {
31
            for variant in data.variants.iter() {
32
                let ident = variant.ident.clone();
33
                let ident_as_string = ident.to_string();
34
                match variant.fields.len() {
35
                    0 => {
36
                        fields.push(quote! { Self::#ident => ActionName(vec![::std::borrow::Cow::Borrowed(#ident_as_string)]) });
37
                    }
38
                    1 => {
39
                        fields.push(quote! {
40
                            Self::#ident(subaction) => {
41
                                let mut name = Action::name(subaction);
42
                                name.0.insert(0, ::std::borrow::Cow::Borrowed(#ident_as_string));
43
                                name
44
                            }
45
                        });
46
                    }
47
                    _ => {
48
                        abort!(
49
                            variant.ident,
50
                            "For derive(Action), all enum variants may have at most 1 field"
51
                        )
52
                    }
53
                }
54
            }
55
        }
56
        _ => abort_call_site!("Action can only be derived for an enum."),
57
    }
58

            
59
    let expanded = quote! {
60
        impl Action for #name {
61
            fn name(&self) -> ActionName {
62
                match self {
63
                    #(
64
                        #fields
65
                    ),*
66
                }
67
            }
68
        }
69
    };
70

            
71
    TokenStream::from(expanded)
72
}