pub fn format(error: impl AsRef<dyn std::error::Error>) -> String {
fn format_impl(error: &dyn std::error::Error) -> String {
let mut string = error.to_string();
for source in std::iter::successors(error.source(), |error| error.source()) {
string.push_str(" -> ");
string.push_str(&source.to_string());
}
string
}
format_impl(error.as_ref())
}
#[test]
fn test_format() {
let err = anyhow::format_err!("root_cause")
.context("inner_context")
.context("outer_context");
assert_eq!(err.to_string(), "outer_context"); assert_eq!(format(&err), "outer_context -> inner_context -> root_cause");
}
pub trait ResultExt<T> {
fn warn_on_err_once(self, msg: impl std::fmt::Display) -> Option<T>;
}
impl<T, E: std::fmt::Display> ResultExt<T> for Result<T, E> {
fn warn_on_err_once(self, msg: impl std::fmt::Display) -> Option<T> {
match self {
Ok(value) => Some(value),
Err(err) => {
re_log::warn_once!("{msg}: {err}");
None
}
}
}
}