use ruma::{ api::client::{ error::{Error as RumaError, ErrorKind}, r0::uiaa::UiaaInfo, }, ServerName, }; use thiserror::Error; use tracing::warn; #[cfg(feature = "persy")] use persy::PersyError; #[cfg(feature = "conduit_bin")] use { crate::RumaResponse, http::StatusCode, rocket::{ response::{self, Responder}, Request, }, ruma::api::client::r0::uiaa::UiaaResponse, tracing::error, }; pub type Result = std::result::Result; #[derive(Error, Debug)] pub enum Error { #[cfg(feature = "sled")] #[error("There was a problem with the connection to the sled database.")] SledError { #[from] source: sled::Error, }, #[cfg(feature = "sqlite")] #[error("There was a problem with the connection to the sqlite database: {source}")] SqliteError { #[from] source: rusqlite::Error, }, #[cfg(feature = "persy")] #[error("There was a problem with the connection to the persy database.")] PersyError { source: PersyError }, #[cfg(feature = "heed")] #[error("There was a problem with the connection to the heed database: {error}")] HeedError { error: String }, #[cfg(feature = "rocksdb")] #[error("There was a problem with the connection to the rocksdb database: {source}")] RocksDbError { #[from] source: rocksdb::Error, }, #[error("Could not generate an image.")] ImageError { #[from] source: image::error::ImageError, }, #[error("Could not connect to server: {source}")] ReqwestError { #[from] source: reqwest::Error, }, #[error("{0}")] FederationError(Box, RumaError), #[error("Could not do this io: {source}")] IoError { #[from] source: std::io::Error, }, #[error("{0}")] BadServerResponse(&'static str), #[error("{0}")] BadConfig(&'static str), #[error("{0}")] /// Don't create this directly. Use Error::bad_database instead. BadDatabase(&'static str), #[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) } } impl Error { pub fn to_response(&self) -> RumaResponse { if let Self::Uiaa(uiaainfo) = self { return RumaResponse(UiaaResponse::AuthResponse(uiaainfo.clone())); } if let Self::FederationError(origin, error) = self { let mut error = error.clone(); error.message = format!("Answer from {}: {}", origin, error.message); return RumaResponse(UiaaResponse::MatrixError(error)); } let message = format!("{}", self); use ErrorKind::*; let (kind, status_code) = match self { Self::BadRequest(kind, _) => ( kind.clone(), match kind { Forbidden | GuestAccessForbidden | ThreepidAuthFailed | ThreepidDenied => { StatusCode::FORBIDDEN } Unauthorized | UnknownToken { .. } | MissingToken => StatusCode::UNAUTHORIZED, NotFound => StatusCode::NOT_FOUND, 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(UiaaResponse::MatrixError(RumaError { kind, message, status_code, })) } } #[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> { self.to_response().respond_to(r) } } #[cfg(feature = "persy")] impl> From> for Error { fn from(err: persy::PE) -> Self { Error::PersyError { source: err.error().into(), } } }