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
use std::fmt::{self, Debug, Display};
#[derive(Debug, Copy, Clone)]
#[allow(clippy::exhaustive_structs)]
pub struct Report<E>(pub E)
where
E: AsRef<dyn std::error::Error>;
impl<E> Display for Report<E>
where
E: AsRef<dyn std::error::Error>,
{
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fn inner(mut e: &dyn std::error::Error, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "error")?;
let mut last = String::new();
loop {
let this = e.to_string();
if !last.contains(&this) {
write!(f, ": {}", &this)?;
}
last = this;
if let Some(ne) = e.source() {
e = ne;
} else {
break;
}
}
Ok(())
}
inner(self.0.as_ref(), f)
}
}
#[allow(clippy::print_stderr)]
pub fn report_and_exit<E, R>(e: E) -> R
where
E: AsRef<dyn std::error::Error>,
{
fn eprint_progname() {
if let Some(progname) = std::env::args().next() {
eprint!("{}: ", progname);
}
}
eprint_progname();
eprintln!("{}", Report(e));
std::process::exit(127)
}
#[cfg(test)]
mod test {
use super::*;
use std::error::Error as StdError;
use std::io;
use thiserror::Error;
#[derive(Error, Debug)]
#[error("terse")]
struct TerseError {
#[from]
source: Box<dyn StdError>,
}
#[derive(Error, Debug)]
#[error("verbose - {source}")]
struct VerboseError {
#[from]
source: Box<dyn StdError>,
}
#[derive(Error, Debug)]
#[error("shallow")]
struct ShallowError;
fn chk<E: StdError + 'static>(e: E, expected: &str) {
let e: Box<dyn StdError> = Box::new(e);
let got = Report(&e).to_string();
assert_eq!(got, expected, "\nmismatch: {:?}", &e);
}
#[test]
#[rustfmt::skip]
fn test() {
chk(ShallowError,
"error: shallow");
let terse_1 = || TerseError { source: ShallowError.into() };
chk(terse_1(),
"error: terse: shallow");
let verbose_1 = || VerboseError { source: ShallowError.into() };
chk(verbose_1(),
"error: verbose - shallow");
chk(VerboseError { source: terse_1().into() },
"error: verbose - terse: shallow");
chk(TerseError { source: verbose_1().into() },
"error: terse: verbose - shallow");
chk(io::Error::new(io::ErrorKind::Other, ShallowError),
"error: shallow");
}
}