use activitystreams_new::primitives::XsdAnyUriError; use actix_web::{ error::{BlockingError, ResponseError}, http::StatusCode, HttpResponse, }; use deadpool::managed::{PoolError, TimeoutType}; use http_signature_normalization_actix::PrepareSignError; use log::error; use rsa_pem::KeyError; use std::{convert::Infallible, fmt::Debug, io::Error}; #[derive(Debug, thiserror::Error)] pub enum MyError { #[error("Error queueing job, {0}")] Queue(anyhow::Error), #[error("Error in configuration, {0}")] Config(#[from] config::ConfigError), #[error("Error in db, {0}")] DbError(#[from] tokio_postgres::error::Error), #[error("Couldn't parse key, {0}")] Key(#[from] KeyError), #[error("Couldn't parse URI, {0}")] Uri(#[from] XsdAnyUriError), #[error("Couldn't perform IO, {0}")] Io(#[from] Error), #[error("Couldn't sign string, {0}")] Rsa(rsa::errors::Error), #[error("Couldn't do the json thing")] Json(#[from] serde_json::Error), #[error("Couldn't build signing string, {0}")] PrepareSign(#[from] PrepareSignError), #[error("Couldn't parse the signature header")] HeaderValidation(#[from] actix_web::http::header::InvalidHeaderValue), #[error("Couldn't decode base64")] Base64(#[from] base64::DecodeError), #[error("Actor ({0}), or Actor's server, is not subscribed")] NotSubscribed(String), #[error("Actor is blocked, {0}")] Blocked(String), #[error("Actor is not whitelisted, {0}")] Whitelist(String), #[error("Cannot make decisions for foreign actor, {0}")] WrongActor(String), #[error("Actor ({0}) tried to submit another actor's ({1}) payload")] BadActor(String, String), #[error("Signature verification is required, but no signature was given")] NoSignature(String), #[error("Wrong ActivityPub kind, {0}")] Kind(String), #[error("Too many CPUs, {0}")] CpuCount(#[from] std::num::TryFromIntError), #[error("Hosts don't match, {0}, {1}")] HostMismatch(String, String), #[error("Invalid or missing content type")] ContentType, #[error("Couldn't flush buffer")] FlushBuffer, #[error("Timed out while waiting on db pool, {0:?}")] DbTimeout(TimeoutType), #[error("Invalid algorithm provided to verifier, {0}")] Algorithm(String), #[error("Object has already been relayed")] Duplicate, #[error("Couldn't send request to {0}, {1}")] SendRequest(String, String), #[error("Couldn't receive request response from {0}, {1}")] ReceiveResponse(String, String), #[error("Response has invalid status code, {0}")] Status(StatusCode), #[error("Expected an Object, found something else")] ObjectFormat, #[error("Expected a single object, found array")] ObjectCount, #[error("Input is missing a 'type' field")] MissingKind, #[error("Input is missing a 'id' field")] MissingId, #[error("URI is missing domain field")] Domain, #[error("Blocking operation was canceled")] Canceled, } impl ResponseError for MyError { fn status_code(&self) -> StatusCode { match self { MyError::Blocked(_) | MyError::Whitelist(_) | MyError::WrongActor(_) | MyError::BadActor(_, _) => StatusCode::FORBIDDEN, MyError::NotSubscribed(_) => StatusCode::UNAUTHORIZED, MyError::Duplicate => StatusCode::ACCEPTED, MyError::Kind(_) | MyError::MissingKind | MyError::MissingId | MyError::ObjectCount => { StatusCode::BAD_REQUEST } _ => StatusCode::INTERNAL_SERVER_ERROR, } } fn error_response(&self) -> HttpResponse { HttpResponse::build(self.status_code()) .header("Content-Type", "application/activity+json") .json(serde_json::json!({ "error": self.to_string(), })) } } impl From> for MyError where T: Into + Debug, { fn from(e: BlockingError) -> Self { match e { BlockingError::Error(e) => e.into(), BlockingError::Canceled => MyError::Canceled, } } } impl From> for MyError where T: Into, { fn from(e: PoolError) -> Self { match e { PoolError::Backend(e) => e.into(), PoolError::Timeout(t) => MyError::DbTimeout(t), } } } impl From for MyError { fn from(i: Infallible) -> Self { match i {} } } impl From for MyError { fn from(e: rsa::errors::Error) -> Self { MyError::Rsa(e) } }