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
106
107
108
109
110
111
112
113
114
115
116
use crate::parsers::MapParser;
use crate::{ParseContext, ParseIter, Parser, Reported, Result};
#[allow(non_camel_case_types)]
#[derive(Clone, Copy)]
pub struct CharParser {
noun: &'static str,
predicate: fn(char) -> bool,
}
pub struct CharParseIter {
c: char,
end: usize,
}
impl Parser for CharParser {
type Output = char;
type RawOutput = (char,);
type Iter<'parse> = CharParseIter;
fn parse_iter<'parse>(
&'parse self,
context: &mut ParseContext<'parse>,
start: usize,
) -> Result<Self::Iter<'parse>, Reported> {
match context.source()[start..].chars().next() {
Some(c) if (self.predicate)(c) => Ok(CharParseIter {
c,
end: start + c.len_utf8(),
}),
_ => Err(context.error_expected(start, self.noun)),
}
}
}
impl<'parse> ParseIter<'parse> for CharParseIter {
type RawOutput = (char,);
fn match_end(&self) -> usize {
self.end
}
fn backtrack(&mut self, _context: &mut ParseContext<'parse>) -> Result<(), Reported> {
Err(Reported)
}
fn into_raw_output(self) -> (char,) {
(self.c,)
}
}
#[allow(non_upper_case_globals)]
pub const alpha: CharParser = CharParser {
noun: "letter",
predicate: char::is_alphabetic,
};
#[allow(non_upper_case_globals)]
pub const alnum: CharParser = CharParser {
noun: "letter or digit",
predicate: char::is_alphanumeric,
};
#[allow(non_upper_case_globals)]
pub const upper: CharParser = CharParser {
noun: "uppercase letter",
predicate: char::is_uppercase,
};
#[allow(non_upper_case_globals)]
pub const lower: CharParser = CharParser {
noun: "lowercase letter",
predicate: char::is_lowercase,
};
#[allow(non_upper_case_globals)]
pub const any_char: CharParser = CharParser {
noun: "any character",
predicate: |_| true,
};
#[allow(non_upper_case_globals)]
pub const digit: MapParser<CharParser, fn(char) -> usize> = MapParser {
parser: CharParser {
noun: "decimal digit",
predicate: |c| c.is_ascii_digit(),
},
mapper: |c| c.to_digit(10).unwrap() as usize,
};
#[allow(non_upper_case_globals)]
pub const digit_bin: MapParser<CharParser, fn(char) -> usize> = MapParser {
parser: CharParser {
noun: "hexadecimal digit",
predicate: |c| c.is_digit(2),
},
mapper: |c| c.to_digit(2).unwrap() as usize,
};
#[allow(non_upper_case_globals, clippy::is_digit_ascii_radix)]
pub const digit_hex: MapParser<CharParser, fn(char) -> usize> = MapParser {
parser: CharParser {
noun: "hexadecimal digit",
predicate: |c| c.is_digit(16),
},
mapper: |c| c.to_digit(16).unwrap() as usize,
};