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
use failure::Fail;

pub type Result<T> = std::result::Result<T, failure::Error>;

pub trait Clamp {
    fn clamp(self, lower: Self, upper: Self) -> Self;
}
impl Clamp for f32 {
    fn clamp(self, lower: f32, upper: f32) -> f32 {
        self.max(lower).min(upper)
    }
}

#[derive(Fail, Debug)]
pub enum EngineError {
    #[fail(display = "Parse Error: {}", error)]
    ParseError {
        error: combine::error::StringStreamError,
    },
}

impl From<combine::error::StringStreamError> for EngineError {
    fn from(error: combine::error::StringStreamError) -> Self {
        EngineError::ParseError { error }
    }
}

#[cfg(test)]
macro_rules! assert_parse {
    ($parser:expr, $input:expr, $output:expr, $remaining:expr) => {
        assert_eq!($parser.parse($input), Ok(($output, $remaining)));
    };
    ($parser:expr, $input:expr, $output:expr) => {
        assert_eq!($parser.parse($input), Ok(($output, "")));
    };
}

#[cfg(test)]
macro_rules! assert_parse_fail {
    ($parser:expr, $input:expr) => {
        assert!($parser.parse($input).is_err());
    };
}

macro_rules! def_parser {
    (pub fn $parser:ident() -> $type:ty $body:block) => {
        parser! {
            pub fn $parser[I]()(I) -> $type
            where
                [I: Stream<Item = char>]
                $body
        }
    };
    (fn $parser:ident() -> $type:ty $body:block) => {
        parser! {
            fn $parser[I]()(I) -> $type
            where
                [I: Stream<Item = char>]
                $body
        }
    };
}