conduit/src/error.rs

118 lines
3.3 KiB
Rust
Raw Normal View History

use log::{error, warn};
use ruma::{
api::client::{
error::{Error as RumaError, ErrorKind},
r0::uiaa::UiaaInfo,
},
ServerName,
};
use thiserror::Error;
#[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 {
2020-06-09 14:13:17 +01:00
#[error("There was a problem with the connection to the database.")]
SledError {
#[from]
source: sled::Error,
},
2020-06-09 14:13:17 +01:00
#[error("Could not generate an image.")]
2020-05-19 17:31:34 +01:00
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),
2020-06-09 14:13:17 +01:00
#[error("{0}")]
BadConfig(&'static str),
#[error("{0}")]
/// Don't create this directly. Use Error::bad_database instead.
BadDatabase(&'static str),
2020-06-09 14:13:17 +01:00
#[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);
2020-06-09 14:13:17 +01:00
}
let message = format!("{}", self);
use ErrorKind::*;
let (kind, status_code) = match self {
Self::BadRequest(kind, _) => (
2020-09-08 16:32:03 +01:00
kind.clone(),
2020-06-09 14:13:17 +01:00
match kind {
Forbidden | GuestAccessForbidden | ThreepidAuthFailed | ThreepidDenied => {
StatusCode::FORBIDDEN
}
2020-09-08 16:32:03 +01:00
Unauthorized | UnknownToken { .. } | MissingToken => StatusCode::UNAUTHORIZED,
2020-06-09 14:13:17 +01:00
NotFound => StatusCode::NOT_FOUND,
2020-09-08 16:32:03 +01:00
LimitExceeded { .. } => StatusCode::TOO_MANY_REQUESTS,
2020-06-09 14:13:17 +01:00
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);
2020-06-09 14:13:17 +01:00
RumaResponse::from(RumaError {
kind,
message,
status_code,
})
.respond_to(r)
}
}