use log::error;
use std::fmt::Display;
#[derive(Debug)]
pub(super) enum Error {
NonFatal(anyhow::Error),
Fatal(anyhow::Error),
}
impl Error {
#[track_caller]
pub(super) fn accept_non_fatal(self) -> Result<(), anyhow::Error> {
match self {
Self::NonFatal(e) => {
accept_error(e);
Ok(())
}
Self::Fatal(e) => Err(e),
}
}
}
#[track_caller]
pub(super) fn accept_error(e: anyhow::Error) {
let l = std::panic::Location::caller();
error!("{}:{}: {:?}", l.file(), l.line(), e);
}
pub(super) trait ContextExt {
fn context<C>(self, context: C) -> Self
where
C: Display + Send + Sync + 'static;
fn with_context<C, F>(self, f: F) -> Self
where
C: Display + Send + Sync + 'static,
F: FnOnce() -> C;
}
impl ContextExt for Error {
fn context<C>(self, context: C) -> Error
where
C: Display + Send + Sync + 'static,
{
match self {
Error::NonFatal(e) => Error::NonFatal(e.context(context)),
Error::Fatal(e) => Error::Fatal(e.context(context)),
}
}
fn with_context<C, F>(self, f: F) -> Error
where
C: Display + Send + Sync + 'static,
F: FnOnce() -> C,
{
self.context(f())
}
}
impl<T, E: ContextExt> ContextExt for Result<T, E> {
fn context<C>(self, context: C) -> Result<T, E>
where
C: Display + Send + Sync + 'static,
{
self.map_err(|e| e.context(context))
}
fn with_context<C, F>(self, f: F) -> Self
where
C: Display + Send + Sync + 'static,
F: FnOnce() -> C,
{
self.map_err(|e| e.with_context(f))
}
}