1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
//! Lib3h_protocol custom error definition.

use serde::de::value::Error as DeserializeError;
use std::{error::Error as StdError, fmt, io, result};
use url::ParseError;

/// A type alias for `Result<T, Lib3hProtocolError>`.
pub type Lib3hProtocolResult<T> = result::Result<T, Lib3hProtocolError>;

/// An error that can occur when interacting with the algorithm.
#[derive(Debug)]
pub struct Lib3hProtocolError(Box<ErrorKind>);

impl Lib3hProtocolError {
    /// A crate private constructor for `Error`.
    pub fn new(kind: ErrorKind) -> Self {
        Lib3hProtocolError(Box::new(kind))
    }

    /// Return the specific type of this error.
    pub fn kind(&self) -> &ErrorKind {
        &self.0
    }

    /// Unwrap this error into its underlying type.
    pub fn into_kind(self) -> ErrorKind {
        *self.0
    }
}

/// The specific type of an error.
#[derive(Debug)]
pub enum ErrorKind {
    /// An I/O error that occurred while processing a data stream.
    Io(io::Error),
    /// Error occuring when using `transport`.
    TransportError(String),
    /// An error occuring whiling trying to deserialize stuff during gossiping for example.
    DeserializeError(DeserializeError),
    /// Error occuring in [Lib3h](https://github.com/holochain/lib3h/). This is kind of a hacky way
    /// to do it but it's a viable option to avoid circular dependency.
    Lib3hError(String, Option<backtrace::Backtrace>),
    /// Error occurred while parsing URLs
    UrlError(ParseError),
    /// Yet undefined error.
    Other(String),
    /// Hints that destructuring should not be exhaustive.
    ///
    /// This enum may grow additional variants, so this makes sure clients
    /// don't count on exhaustive matching. (Otherwise, adding a new variant
    /// could break existing code.)
    #[doc(hidden)]
    __Nonexhaustive,
}

impl StdError for Lib3hProtocolError {
    /// The lower-level source of this error, if any.
    fn source(&self) -> Option<&(dyn StdError + 'static)> {
        match *self.0 {
            ErrorKind::Io(ref err) => Some(err),
            ErrorKind::DeserializeError(ref err) => Some(err),
            ErrorKind::Lib3hError(ref _s, ref _bt) => None,
            ErrorKind::UrlError(ref err) => Some(err),
            ErrorKind::Other(ref _s) | ErrorKind::TransportError(ref _s) => None,
            _ => unreachable!(),
        }
    }
}

impl fmt::Display for Lib3hProtocolError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match *self.0 {
            ErrorKind::Io(ref err) => err.fmt(f),
            ErrorKind::TransportError(ref s) => write!(f, "TransportError: '{}'.", s),
            ErrorKind::DeserializeError(ref err) => err.fmt(f),
            ErrorKind::Lib3hError(ref s, ref _bt) => write!(f, "Li3hError encountered: '{}'.", s),
            ErrorKind::UrlError(ref s) => write!(f, "Url ParseError: '{}'.", s),
            ErrorKind::Other(ref s) => write!(f, "Unknown error encountered: '{}'.", s),
            _ => unreachable!(),
        }
    }
}

impl From<io::Error> for Lib3hProtocolError {
    fn from(err: io::Error) -> Self {
        Lib3hProtocolError::new(ErrorKind::Io(err))
    }
}

impl From<DeserializeError> for Lib3hProtocolError {
    fn from(err: DeserializeError) -> Self {
        Lib3hProtocolError::new(ErrorKind::DeserializeError(err))
    }
}

impl From<ParseError> for Lib3hProtocolError {
    fn from(err: ParseError) -> Self {
        Lib3hProtocolError::new(ErrorKind::UrlError(err))
    }
}