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
use std::{error::Error, fmt};
use crate::parser::{LexError, ParseError};
use super::LeftoverError;
#[derive(Debug, PartialEq)]
pub enum SyntaxError<T> {
LexError(LexError),
ParseError(ParseError),
Leftover(LeftoverError<T>),
}
impl<T> SyntaxError<T> {
pub fn map_parsed<T2>(self, map: impl Fn(T) -> T2) -> SyntaxError<T2> {
match self {
Self::LexError(err) => SyntaxError::LexError(err),
Self::ParseError(err) => SyntaxError::ParseError(err),
Self::Leftover(err) => SyntaxError::Leftover(err.map_parsed(map)),
}
}
pub fn recover(self) -> Result<T, Self> {
match self {
Self::Leftover(err) => Ok(err.recover()),
err => Err(err),
}
}
}
impl<T: fmt::Debug + 'static> fmt::Display for SyntaxError<T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if f.alternate() {
write!(f, "error while parsing: {:#}", self.source().unwrap())
} else {
write!(f, "error while parsing: {}", self.source().unwrap())
}
}
}
impl<T> From<LexError> for SyntaxError<T> {
fn from(err: LexError) -> Self {
Self::LexError(err)
}
}
impl<T> From<ParseError> for SyntaxError<T> {
fn from(err: ParseError) -> Self {
Self::ParseError(err)
}
}
impl<T> From<LeftoverError<T>> for SyntaxError<T> {
fn from(err: LeftoverError<T>) -> Self {
Self::Leftover(err)
}
}
impl<T: fmt::Debug + 'static> Error for SyntaxError<T> {
fn source(&self) -> Option<&(dyn Error + 'static)> {
match self {
Self::LexError(err) => Some(err),
Self::ParseError(err) => Some(err),
Self::Leftover(err) => Some(err),
}
}
}