Skip to content
Snippets Groups Projects
error.rs 3.29 KiB
Newer Older
  • Learn to ignore specific revisions
  • use log::{error, warn};
    
    use ruma::{
        api::client::{
            error::{Error as RumaError, ErrorKind},
            r0::uiaa::UiaaInfo,
        },
        ServerName,
    };
    
    #[cfg(feature = "conduit_bin")]
    use {
        crate::RumaResponse,
        http::StatusCode,
        rocket::{
            response::{self, Responder},
            Request,
        },
    
        ruma::api::client::r0::uiaa::UiaaResponse,
    
    pub type Result<T> = std::result::Result<T, Error>;
    
    #[derive(Error, Debug)]
    pub enum Error {
    
        #[error("There was a problem with the connection to the database.")]
    
        #[error("Could not generate an image.")]
    
    timokoesters's avatar
    timokoesters committed
        ImageError {
            #[from]
            source: image::error::ImageError,
        },
    
        #[error("Could not connect to server: {source}")]
    
        ReqwestError {
            #[from]
            source: reqwest::Error,
        },
        #[error("{0}")]
    
        FederationError(Box<ServerName>, RumaError),
        #[error("{0}")]
    
        BadServerResponse(&'static str),
    
        #[error("{0}")]
        BadConfig(&'static str),
        #[error("{0}")]
    
        /// Don't create this directly. Use Error::bad_database instead.
    
        #[error("uiaa")]
        Uiaa(UiaaInfo),
    
        #[error("{0}: {1}")]
        BadRequest(ErrorKind, &'static str),
        #[error("{0}")]
        Conflict(&'static str), // This is only needed for when a room alias already exists
    }
    
    
    impl Error {
        pub fn bad_database(message: &'static str) -> Self {
            error!("BadDatabase: {}", message);
            Self::BadDatabase(message)
        }
    
    
        pub fn bad_config(message: &'static str) -> Self {
            error!("BadConfig: {}", message);
            Self::BadConfig(message)
        }
    
    #[cfg(feature = "conduit_bin")]
    
    impl<'r, 'o> Responder<'r, 'o> for Error
    where
        'o: 'r,
    {
        fn respond_to(self, r: &'r Request<'_>) -> response::Result<'o> {
    
            if let Self::Uiaa(uiaainfo) = self {
                return RumaResponse::from(UiaaResponse::AuthResponse(uiaainfo)).respond_to(r);
            }
    
            if let Self::FederationError(origin, mut error) = self {
                error.message = format!("Answer from {}: {}", origin, error.message);
                return RumaResponse::from(error).respond_to(r);
    
            }
    
            let message = format!("{}", self);
    
            use ErrorKind::*;
            let (kind, status_code) = match self {
                Self::BadRequest(kind, _) => (
    
    Timo Kösters's avatar
    Timo Kösters committed
                    kind.clone(),
    
                    match kind {
                        Forbidden | GuestAccessForbidden | ThreepidAuthFailed | ThreepidDenied => {
                            StatusCode::FORBIDDEN
                        }
    
    Timo Kösters's avatar
    Timo Kösters committed
                        Unauthorized | UnknownToken { .. } | MissingToken => StatusCode::UNAUTHORIZED,
    
                        NotFound => StatusCode::NOT_FOUND,
    
    Timo Kösters's avatar
    Timo Kösters committed
                        LimitExceeded { .. } => StatusCode::TOO_MANY_REQUESTS,
    
                        UserDeactivated => StatusCode::FORBIDDEN,
                        TooLarge => StatusCode::PAYLOAD_TOO_LARGE,
                        _ => StatusCode::BAD_REQUEST,
                    },
                ),
                Self::Conflict(_) => (Unknown, StatusCode::CONFLICT),
                _ => (Unknown, StatusCode::INTERNAL_SERVER_ERROR),
            };
    
    
            warn!("{}: {}", status_code, message);
    
    
            RumaResponse::from(RumaError {
                kind,
                message,
                status_code,
            })
            .respond_to(r)
        }