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
use nom::error::ParseError;
use quick_error::quick_error;
quick_error! {
#[derive(Debug)]
#[allow(missing_docs)]
pub enum Error {
ParseIntegerError(msg: &'static str, number: crate::BString, err: btoi::ParseIntegerError) {
display("{}: {:?}", msg, number)
source(err)
}
Nom(err_msg: String) {
display("{}", err_msg)
}
NomDetail(input: crate::BString, msg: &'static str) {
display("{}: '{}' could not be parsed", msg, input)
}
ParseKindError(err: crate::types::Error) {
display("{}", err)
source(err)
}
ObjectKind(err: crate::Error) {
from()
source(err)
}
}
}
impl Error {
fn set_parse_context(mut self, ctx: &'static str) -> Self {
if let Error::NomDetail(_, ref mut message) = self {
*message = ctx
}
self
}
pub(crate) fn context(msg: &'static str) -> impl Fn(nom::Err<Self>) -> nom::Err<Self> {
move |e: nom::Err<Self>| e.map(|e| e.set_parse_context(msg))
}
}
impl ParseError<&[u8]> for Error {
fn from_error_kind(input: &[u8], _kind: nom::error::ErrorKind) -> Self {
Error::NomDetail(input.into(), "parse error")
}
fn append(_: &[u8], _: nom::error::ErrorKind, other: Self) -> Self {
other
}
}
impl From<nom::Err<Error>> for Error {
fn from(e: nom::Err<Error>) -> Self {
match e {
nom::Err::Error(err) | nom::Err::Failure(err) => Error::Nom(err.to_string()),
nom::Err::Incomplete(_) => unreachable!("we do not implement streaming parsers"),
}
}
}