parent
a515ac47ae
commit
f1f3a30ac7
@ -0,0 +1,260 @@
|
|||||||
|
//! Error handling stuffs
|
||||||
|
use std::{
|
||||||
|
error,
|
||||||
|
fmt,
|
||||||
|
};
|
||||||
|
|
||||||
|
//TODO:
|
||||||
|
// this was a complete failure, look at eyre instead or something
|
||||||
|
|
||||||
|
/// Context for an error
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Ord, PartialOrd, Hash)]
|
||||||
|
pub struct Context
|
||||||
|
{
|
||||||
|
line: Option<u32>,
|
||||||
|
column: Option<u32>,
|
||||||
|
file: &'static str,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Context
|
||||||
|
{
|
||||||
|
pub fn new(line: Option<u32>, column: Option<u32>, file: &'static str) -> Self
|
||||||
|
{
|
||||||
|
Self {
|
||||||
|
line,column,file
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for Context
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{} ", self.file)?;
|
||||||
|
if let Some(line) = self.line {
|
||||||
|
write!(f, "{}:", line)?;
|
||||||
|
} else {
|
||||||
|
write!(f, "?:")?;
|
||||||
|
}
|
||||||
|
if let Some(column) = self.column {
|
||||||
|
write!(f, "{}", column)?;
|
||||||
|
} else {
|
||||||
|
write!(f, "?")?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A wrapper over an error with a nicer error message
|
||||||
|
///
|
||||||
|
/// Like doushio's `Muggle`.
|
||||||
|
pub struct Wrap<E,D>(E, D);
|
||||||
|
|
||||||
|
pub trait WrappedError
|
||||||
|
{
|
||||||
|
fn long_msg(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result;
|
||||||
|
fn short_msg(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E,D> WrappedError for Wrap<E,D>
|
||||||
|
where D: fmt::Display,
|
||||||
|
E: error::Error + 'static
|
||||||
|
{
|
||||||
|
fn long_msg(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", self.long_message())
|
||||||
|
}
|
||||||
|
fn short_msg(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E,D> Wrap<E,D>
|
||||||
|
where D: fmt::Display,
|
||||||
|
E: error::Error + 'static
|
||||||
|
{
|
||||||
|
/// Get all sources of the error
|
||||||
|
pub fn all_sources(&self) -> Vec<&(dyn error::Error + 'static)>
|
||||||
|
{
|
||||||
|
use std::error::Error;
|
||||||
|
std::iter::successors(self.source(), |x| x.source()).collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Into the inner error
|
||||||
|
pub fn into_inner(self) -> E
|
||||||
|
{
|
||||||
|
self.0
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the full and long error message
|
||||||
|
pub fn long_message(&self) -> impl fmt::Display + '_
|
||||||
|
{
|
||||||
|
struct LongOutput<'a, E,D>(&'a Wrap<E,D>);
|
||||||
|
impl<'a, E,D> fmt::Display for LongOutput<'a, E,D>
|
||||||
|
where Wrap<E,D>: error::Error
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
writeln!(f, "{}", self.0)?;
|
||||||
|
use std::error::Error;
|
||||||
|
for (i, spec) in (1..).zip(std::iter::successors(self.0.source(), |x| x.source()))
|
||||||
|
{
|
||||||
|
writeln!(f, " [{}] -> {}", i, spec)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LongOutput(&self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait WrapExt: Sized
|
||||||
|
{
|
||||||
|
/// Wrap this error in another
|
||||||
|
fn wrap<T>(self, msg: T) -> Wrap<Self, T>
|
||||||
|
where T: fmt::Display;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E> WrapExt for E where E: error::Error
|
||||||
|
{
|
||||||
|
fn wrap<T>(self, msg: T) -> Wrap<Self, T>
|
||||||
|
where T: fmt::Display
|
||||||
|
{
|
||||||
|
Wrap(self, msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait WrapErrExt<T,E>: Sized
|
||||||
|
{
|
||||||
|
/// Wrap the error from this result with a nicer error message
|
||||||
|
fn wrap_err<U>(self, msg: U) -> Result<T,Wrap<E, U>>
|
||||||
|
where U: fmt::Display;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T,E> WrapErrExt<T,E> for Result<T,E>
|
||||||
|
{
|
||||||
|
fn wrap_err<U>(self, msg: U) -> Result<T,Wrap<E, U>>
|
||||||
|
where U: fmt::Display
|
||||||
|
{
|
||||||
|
self.map_err(|e| Wrap(e, msg))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
impl<E,D> fmt::Debug for Wrap<E,D>
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", std::any::type_name::<Self>())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E,D> fmt::Display for Wrap<E,D>
|
||||||
|
where D: fmt::Display
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", self.1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E,D> error::Error for Wrap<E,D>
|
||||||
|
where E: error::Error + 'static,
|
||||||
|
D: fmt::Display
|
||||||
|
{
|
||||||
|
fn source(&self) -> Option<&(dyn error::Error + 'static)>
|
||||||
|
{
|
||||||
|
Some(&self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An error with a context
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct ContextError<E>(pub E, pub Context)
|
||||||
|
where E: fmt::Debug;
|
||||||
|
|
||||||
|
impl<E> WrappedError for ContextError<E>
|
||||||
|
where E: fmt::Debug + fmt::Display
|
||||||
|
{
|
||||||
|
fn long_msg(&self,f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", self)
|
||||||
|
}
|
||||||
|
fn short_msg(&self,f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}", self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E> fmt::Display for ContextError<E>
|
||||||
|
where E: fmt::Display + fmt::Debug
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f, "{}: {}", self.1, self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<E> error::Error for ContextError<E>
|
||||||
|
where E: fmt::Display + fmt::Debug{}
|
||||||
|
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! context {
|
||||||
|
() => ($crate::error::Context::new(Some(line!()), Some(column!()), file!()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Construct a new error with context
|
||||||
|
#[macro_export] macro_rules! error {
|
||||||
|
($err:expr) => ($crate::error::ContextError($err, $crate::context!()));
|
||||||
|
($fmt:literal $($tt:tt)*) => ($crate::error!(format!($fmt $($tt)*)));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ErrorStack(Box<dyn WrappedError>);
|
||||||
|
|
||||||
|
impl<T: WrappedError + 'static> From<T> for ErrorStack
|
||||||
|
{
|
||||||
|
fn from(from: T) -> Self
|
||||||
|
{
|
||||||
|
Self(Box::new(from))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for ErrorStack
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
self.0.long_msg(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for ErrorStack
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
write!(f,"<opaque stack>")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl error::Error for ErrorStack{}
|
||||||
|
|
||||||
|
impl ErrorStack
|
||||||
|
{
|
||||||
|
/// Get the short message from this stack
|
||||||
|
pub fn short_message(&self) -> impl fmt::Display + '_
|
||||||
|
{
|
||||||
|
pub struct ShortMessage<'a>(&'a ErrorStack);
|
||||||
|
|
||||||
|
impl<'a> fmt::Display for ShortMessage<'a>
|
||||||
|
{
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result
|
||||||
|
{
|
||||||
|
self.0.0.short_msg(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ShortMessage(self)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue