chainerror
provides an error backtrace without doing a real backtrace, so even after you strip
your
binaries, you still have the error backtrace.
chainerror
has no dependencies!
chainerror
uses .source()
of std::error::Error
along with #[track_caller]
and Location
to provide a nice debug error backtrace.
It encapsulates all types, which have Display + Debug
and can store the error cause internally.
Along with the ChainError<T>
struct, chainerror
comes with some useful helper macros to save a lot of typing.
Debug information is worth it!
display-cause
: turn on printing a backtrace of the errors in Display
Read the Tutorial
examples/example.rs:
rust
// […]
fn main() {
if let Err(e) = func1() {
eprintln!("\nDebug Error {{:?}}:\n{:?}", e);
eprintln!("\nAlternative Debug Error {{:#?}}:\n{:#?}\n", e);
// […]
}
}
```console $ cargo run -q --example example Debug Error {:?}: examples/example.rs:46:13: func1 error calling func2 Caused by: examples/example.rs:21:13: Func2Error(func2 error: calling func3) Caused by: examples/example.rs:14:18: Error reading 'foo.txt' Caused by: Kind(NotFound)
Alternative Debug Error {:#?}:
ChainError
```rust use chainerror::prelude::v1::*; use std::error::Error; use std::io; use std::result::Result;
fn dosomeio() -> Result<(), Box
fn func2() -> Result<(), Box
fn func1() -> Result<(), Box
if let Err(e) = func1() { #[cfg(not(windows))] assert_eq!( format!("\n{:?}\n", e), r#" src/lib.rs:21:13: func1 error Caused by: src/lib.rs:16:18: Error reading 'foo.txt' Caused by: Kind(NotFound) "# ); } ```
```rust use chainerror::prelude::v1::*; use std::error::Error; use std::io; use std::result::Result;
fn dosomeio() -> Result<(), Box
fn func3() -> Result<(), Box
derivestrcontext!(Func2Error);
fn func2() -> ChainResult<(), Func2Error> { func3().context(Func2Error("func2 error: calling func3".into()))?; Ok(()) }
enum Func1Error { Func2, IO(String), }
impl ::std::fmt::Display for Func1Error { fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result { match self { Func1Error::Func2 => write!(f, "func1 error calling func2"), Func1Error::IO(filename) => write!(f, "Error reading '{}'", filename), } } }
impl ::std::fmt::Debug for Func1Error { fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result { write!(f, "{}", self) } }
fn func1() -> ChainResult<(), Func1Error> { func2().context(Func1Error::Func2)?; let filename = String::from("bar.txt"); dosomeio().context(Func1Error::IO(filename))?; Ok(()) }
if let Err(e) = func1() { assert!(match e.kind() { Func1Error::Func2 => { eprintln!("Main Error Report: func1 error calling func2"); true } Func1Error::IO(filename) => { eprintln!("Main Error Report: func1 error reading '{}'", filename); false } });
assert!(e.find_chain_cause::<Func2Error>().is_some());
if let Some(e) = e.find_chain_cause::<Func2Error>() {
eprintln!("\nError reported by Func2Error: {}", e)
}
assert!(e.root_cause().is_some());
if let Some(e) = e.root_cause() {
let io_error = e.downcast_ref::<io::Error>().unwrap();
eprintln!("\nThe root cause was: std::io::Error: {:#?}", io_error);
}
#[cfg(not(windows))]
assert_eq!(
format!("\n{:?}\n", e),
r#"
src/lib.rs:48:13: func1 error calling func2 Caused by: src/lib.rs:23:13: Func2Error(func2 error: calling func3) Caused by: src/lib.rs:16:18: Error reading 'foo.txt' Caused by: Kind(NotFound) "# ); } ```
Licensed under either of
at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.