/*
* meli - error module
*
* Copyright 2017 Manos Pitsidianakis
*
* This file is part of meli.
*
* meli is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* meli is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with meli. If not, see .
*/
/*!
* An error object for `melib`
*/
use std::borrow::Cow;
use std::error::Error;
use std::fmt;
use std::io;
use std::result;
use std::str;
use std::string;
use std::sync::Arc;
pub type Result = result::Result;
#[derive(Debug, Clone)]
pub struct MeliError {
pub summary: Option>,
pub details: Cow<'static, str>,
pub source: Option>,
}
pub trait IntoMeliError {
fn set_err_summary(self, msg: M) -> MeliError
where
M: Into>;
}
pub trait ResultIntoMeliError {
fn chain_err_summary(self, msg_fn: F) -> Result
where
F: Fn() -> M,
M: Into>;
}
impl> IntoMeliError for I {
#[inline]
fn set_err_summary(self, msg: M) -> MeliError
where
M: Into>,
{
let err: MeliError = self.into();
err.set_summary(msg)
}
}
impl> ResultIntoMeliError for std::result::Result {
#[inline]
fn chain_err_summary(self, msg_fn: F) -> Result
where
F: Fn() -> M,
M: Into>,
{
self.map_err(|err| err.set_err_summary(msg_fn()))
}
}
impl MeliError {
pub fn new(msg: M) -> MeliError
where
M: Into>,
{
MeliError {
summary: None,
details: msg.into(),
source: None,
}
}
pub fn set_summary(mut self, summary: M) -> MeliError
where
M: Into>,
{
if let Some(old_summary) = self.summary.take() {
self.summary = Some(format!("{}. {}", old_summary, summary.into()).into());
} else {
self.summary = Some(summary.into());
}
self
}
pub fn set_source(
mut self,
new_val: Option>,
) -> MeliError {
self.source = new_val;
self
}
}
impl fmt::Display for MeliError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if let Some(summary) = self.summary.as_ref() {
writeln!(f, "Summary: {}", summary)?;
}
let ret = write!(f, "{}", self.details)?;
if let Some(source) = self.source.as_ref() {
write!(f, "\nCaused by: {}", source)?;
}
Ok(ret)
}
}
impl Error for MeliError {
fn source(&self) -> Option<&(dyn Error + 'static)> {
self.source.as_ref().map(|s| &(*(*s)) as _)
}
}
impl From for MeliError {
#[inline]
fn from(kind: io::Error) -> MeliError {
MeliError::new(kind.to_string()).set_source(Some(Arc::new(kind)))
}
}
impl<'a> From> for MeliError {
#[inline]
fn from(kind: Cow<'_, str>) -> MeliError {
MeliError::new(format!("{:?}", kind))
}
}
impl From for MeliError {
#[inline]
fn from(kind: string::FromUtf8Error) -> MeliError {
MeliError::new(format!("{:?}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From for MeliError {
#[inline]
fn from(kind: str::Utf8Error) -> MeliError {
MeliError::new(format!("{:?}", kind)).set_source(Some(Arc::new(kind)))
}
}
//use std::option;
//impl From for MeliError {
// #[inline]
// fn from(kind: option::NoneError) -> MeliError {
// MeliError::new(format!("{:?}", kind))
// }
//}
impl From> for MeliError {
#[inline]
fn from(kind: std::sync::PoisonError) -> MeliError {
MeliError::new(format!("{}", kind))
}
}
#[cfg(feature = "imap_backend")]
impl From>
for MeliError
{
#[inline]
fn from(kind: native_tls::HandshakeError) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
#[cfg(feature = "imap_backend")]
impl From for MeliError {
#[inline]
fn from(kind: native_tls::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From for MeliError {
#[inline]
fn from(kind: std::num::ParseIntError) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
#[cfg(feature = "jmap_backend")]
impl From for MeliError {
#[inline]
fn from(kind: reqwest::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
#[cfg(feature = "jmap_backend")]
impl From for MeliError {
#[inline]
fn from(kind: serde_json::error::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From> for MeliError {
#[inline]
fn from(kind: Box) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(kind.into()))
}
}
impl From for MeliError {
#[inline]
fn from(kind: std::ffi::NulError) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From for MeliError {
#[inline]
fn from(kind: nix::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
#[cfg(feature = "sqlite3")]
impl From for MeliError {
#[inline]
fn from(kind: rusqlite::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From for MeliError {
#[inline]
fn from(kind: libloading::Error) -> MeliError {
MeliError::new(format!("{}", kind)).set_source(Some(Arc::new(kind)))
}
}
impl From<&str> for MeliError {
#[inline]
fn from(kind: &str) -> MeliError {
MeliError::new(kind.to_string())
}
}
impl From for MeliError {
#[inline]
fn from(kind: String) -> MeliError {
MeliError::new(kind)
}
}
impl From> for MeliError {
#[inline]
fn from(kind: nom::Err<(&[u8], nom::error::ErrorKind)>) -> MeliError {
MeliError::new("Parsing error")
.set_source(Some(Arc::new(MeliError::new(format!("{}", kind)))))
}
}
impl From> for MeliError {
#[inline]
fn from(kind: nom::Err<(&str, nom::error::ErrorKind)>) -> MeliError {
MeliError::new("Parsing error")
.set_source(Some(Arc::new(MeliError::new(format!("{}", kind)))))
}
}
impl<'a> From<&'a mut MeliError> for MeliError {
#[inline]
fn from(kind: &'a mut MeliError) -> MeliError {
kind.clone()
}
}