use std::io;
use actix_connect::resolver::ResolveError;
use derive_more::{Display, From};
#[cfg(feature = "openssl")]
use actix_connect::ssl::openssl::{HandshakeError, SslError};
use crate::error::{Error, ParseError, ResponseError};
use crate::http::{Error as HttpError, StatusCode};
#[derive(Debug, Display, From)]
pub enum ConnectError {
#[display(fmt = "SSL is not supported")]
SslIsNotSupported,
#[cfg(feature = "openssl")]
#[display(fmt = "{}", _0)]
SslError(SslError),
#[cfg(feature = "openssl")]
#[display(fmt = "{}", _0)]
SslHandshakeError(String),
#[display(fmt = "Failed resolving hostname: {}", _0)]
Resolver(ResolveError),
#[display(fmt = "No dns records found for the input")]
NoRecords,
#[display(fmt = "{}", _0)]
H2(h2::Error),
#[display(fmt = "Timeout out while establishing connection")]
Timeout,
#[display(fmt = "Internal error: connector has been disconnected")]
Disconnected,
#[display(fmt = "Connector received `Connect` method with unresolved host")]
Unresolverd,
#[display(fmt = "{}", _0)]
Io(io::Error),
}
impl From<actix_connect::ConnectError> for ConnectError {
fn from(err: actix_connect::ConnectError) -> ConnectError {
match err {
actix_connect::ConnectError::Resolver(e) => ConnectError::Resolver(e),
actix_connect::ConnectError::NoRecords => ConnectError::NoRecords,
actix_connect::ConnectError::InvalidInput => panic!(),
actix_connect::ConnectError::Unresolverd => ConnectError::Unresolverd,
actix_connect::ConnectError::Io(e) => ConnectError::Io(e),
}
}
}
#[cfg(feature = "openssl")]
impl<T: std::fmt::Debug> From<HandshakeError<T>> for ConnectError {
fn from(err: HandshakeError<T>) -> ConnectError {
ConnectError::SslHandshakeError(format!("{:?}", err))
}
}
#[derive(Debug, Display, From)]
pub enum InvalidUrl {
#[display(fmt = "Missing url scheme")]
MissingScheme,
#[display(fmt = "Unknown url scheme")]
UnknownScheme,
#[display(fmt = "Missing host name")]
MissingHost,
#[display(fmt = "Url parse error: {}", _0)]
HttpError(http::Error),
}
#[derive(Debug, Display, From)]
pub enum SendRequestError {
#[display(fmt = "Invalid URL: {}", _0)]
Url(InvalidUrl),
#[display(fmt = "Failed to connect to host: {}", _0)]
Connect(ConnectError),
Send(io::Error),
Response(ParseError),
#[display(fmt = "{}", _0)]
Http(HttpError),
#[display(fmt = "{}", _0)]
H2(h2::Error),
#[display(fmt = "Timeout out while waiting for response")]
Timeout,
#[display(fmt = "Tunnels are not supported for http2 connection")]
TunnelNotSupported,
Body(Error),
}
impl ResponseError for SendRequestError {
fn status_code(&self) -> StatusCode {
match *self {
SendRequestError::Connect(ConnectError::Timeout) => {
StatusCode::GATEWAY_TIMEOUT
}
SendRequestError::Connect(_) => StatusCode::BAD_REQUEST,
_ => StatusCode::INTERNAL_SERVER_ERROR,
}
}
}
#[derive(Debug, Display, From)]
pub enum FreezeRequestError {
#[display(fmt = "Invalid URL: {}", _0)]
Url(InvalidUrl),
#[display(fmt = "{}", _0)]
Http(HttpError),
}
impl From<FreezeRequestError> for SendRequestError {
fn from(e: FreezeRequestError) -> Self {
match e {
FreezeRequestError::Url(e) => e.into(),
FreezeRequestError::Http(e) => e.into(),
}
}
}