summaryrefslogtreecommitdiff
path: root/src/settings.rs
blob: 0b7ec766f6ced17902956f843ae3ba04436522cb (plain)
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
use crate::{Error, Result};
use proc_macro2::{Ident, Span, TokenStream, TokenTree};

macro_rules! decl_settings {
    ($($val:expr => $variant:ident),+ $(,)*) => {
        #[derive(PartialEq)]
        pub(crate) enum Setting {
            $($variant),*
        }

        fn ident_to_setting(ident: Ident) -> Result<Setting> {
            match &*ident.to_string() {
                $($val => Ok(Setting::$variant),)*
                _ => {
                    let possible_vals = [$($val),*]
                        .iter()
                        .map(|v| format!("`{}`", v))
                        .collect::<Vec<_>>()
                        .join(", ");

                    Err(Error::new(
                        ident.span(),
                        format!("unknown setting `{}`, expected one of {}", ident, possible_vals)))
                }
            }
        }
    };
}

decl_settings! {
    "assert_unwind_safe" => AssertUnwindSafe,
    "allow_not_macro"    => AllowNotMacro,
    "proc_macro_hack"    => ProcMacroHack,
}

pub(crate) fn parse_settings(input: TokenStream) -> Result<Settings> {
    let mut input = input.into_iter();
    let mut res = Settings(Vec::new());
    loop {
        match input.next() {
            Some(TokenTree::Ident(ident)) => {
                res.0.push(ident_to_setting(ident)?);
            }
            None => return Ok(res),
            other => {
                let span = other.map_or(Span::call_site(), |tt| tt.span());
                return Err(Error::new(span, "expected identifier".to_string()));
            }
        }

        match input.next() {
            Some(TokenTree::Punct(ref punct)) if punct.as_char() == ',' => {}
            None => return Ok(res),
            other => {
                let span = other.map_or(Span::call_site(), |tt| tt.span());
                return Err(Error::new(span, "expected `,`".to_string()));
            }
        }
    }
}

pub(crate) struct Settings(Vec<Setting>);

impl Settings {
    pub(crate) fn is_set(&self, setting: Setting) -> bool {
        self.0.iter().any(|s| *s == setting)
    }

    pub(crate) fn set(&mut self, setting: Setting) {
        self.0.push(setting)
    }
}