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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
use proc_macro2::Span;
use syn::parse::{self, Parse, ParseStream};

mod kw {
    syn::custom_keyword!(when);
    syn::custom_keyword!(then);
    syn::custom_keyword!(given);
    syn::custom_keyword!(case);
    syn::custom_keyword!(section);
}

#[derive(Debug, Clone, PartialEq, Eq)]
pub enum SectionKeyword {
    When,
    Then,
    Given,
    Case,
    Section,
}

impl SectionKeyword {
    pub fn to_name(&self) -> String {
        match self {
            Self::When => "when".to_string(),
            Self::Then => "then".to_string(),
            Self::Given => "given".to_string(),
            Self::Case => "case".to_string(),
            Self::Section => "section".to_string(),
        }
    }
}

impl SectionKeyword {
    pub fn peek(i: ParseStream) -> bool {
        let lk = i.lookahead1();

        let mut test = false;

        test |= lk.peek(kw::when);
        test |= lk.peek(kw::given);
        test |= lk.peek(kw::section);
        test |= lk.peek(kw::then);
        test |= lk.peek(kw::case);

        test
    }
}

impl Parse for SectionKeyword {
    fn parse(input: ParseStream) -> parse::Result<Self> {
        let lk = input.lookahead1();

        if lk.peek(kw::when) {
            input.parse::<kw::when>()?;

            Ok(Self::When)
        } else if lk.peek(kw::then) {
            input.parse::<kw::then>()?;

            Ok(Self::Then)
        } else if lk.peek(kw::given) {
            input.parse::<kw::given>()?;

            Ok(Self::Given)
        } else if lk.peek(kw::case) {
            input.parse::<kw::case>()?;

            Ok(Self::Case)
        } else if lk.peek(kw::section) {
            input.parse::<kw::section>()?;

            Ok(Self::Section)
        } else {
            Err(parse::Error::new(
                Span::call_site(),
                "Invalid section keyword",
            ))
        }
    }
}

#[cfg(test)]
#[cfg(test)]
mod tests {
    use super::*;
    use test_case::test_case;

    #[test_case("when"      => SectionKeyword::When     ; "when")]
    #[test_case("given"     => SectionKeyword::Given    ; "given")]
    #[test_case("section"   => SectionKeyword::Section  ; "section")]
    #[test_case("then"      => SectionKeyword::Then     ; "then")]
    #[test_case("case"      => SectionKeyword::Case     ; "case")]
    fn parse(s: &str) -> SectionKeyword {
        syn::parse_str(s).unwrap()
    }

    #[test_case(SectionKeyword::When,    "when"     ; "when")]
    #[test_case(SectionKeyword::Given,   "given"    ; "given")]
    #[test_case(SectionKeyword::Section, "section"  ; "section")]
    #[test_case(SectionKeyword::Then,    "then"     ; "then")]
    #[test_case(SectionKeyword::Case,    "case"     ; "case")]
    fn to_name(kw: SectionKeyword, exp: &str) {
        assert_eq!(&kw.to_name(), exp);
    }
}