use crate::http::{transport::BuildError, StatusCode};
use std::{error, fmt, io};
#[derive(Debug)]
pub struct Error {
kind: Kind,
}
#[derive(Debug)]
enum Kind {
Build(BuildError),
Lib(String),
Http(reqwest::Error),
Io(io::Error),
Json(serde_json::error::Error),
}
impl From<io::Error> for Error {
fn from(err: io::Error) -> Error {
Error {
kind: Kind::Io(err),
}
}
}
impl From<reqwest::Error> for Error {
fn from(err: reqwest::Error) -> Error {
Error {
kind: Kind::Http(err),
}
}
}
impl From<serde_json::error::Error> for Error {
fn from(err: serde_json::error::Error) -> Error {
Error {
kind: Kind::Json(err),
}
}
}
impl From<url::ParseError> for Error {
fn from(err: url::ParseError) -> Error {
Error {
kind: Kind::Lib(err.to_string()),
}
}
}
impl From<BuildError> for Error {
fn from(err: BuildError) -> Error {
Error {
kind: Kind::Build(err),
}
}
}
pub(crate) fn lib(err: impl Into<String>) -> Error {
Error {
kind: Kind::Lib(err.into()),
}
}
impl Error {
pub fn status_code(&self) -> Option<StatusCode> {
match &self.kind {
Kind::Http(err) => err.status(),
_ => None,
}
}
pub fn is_timeout(&self) -> bool {
match &self.kind {
Kind::Http(err) => err.is_timeout(),
_ => false,
}
}
pub fn is_json(&self) -> bool {
matches!(&self.kind, Kind::Json(_))
}
}
impl error::Error for Error {
fn source(&self) -> Option<&(dyn error::Error + 'static)> {
match &self.kind {
Kind::Build(err) => Some(err),
Kind::Lib(_) => None,
Kind::Http(err) => Some(err),
Kind::Io(err) => Some(err),
Kind::Json(err) => Some(err),
}
}
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match &self.kind {
Kind::Build(err) => err.fmt(f),
Kind::Lib(err) => err.fmt(f),
Kind::Http(err) => err.fmt(f),
Kind::Io(err) => err.fmt(f),
Kind::Json(err) => err.fmt(f),
}
}
}