forked from mirrors/relay
177 lines
4.6 KiB
Rust
177 lines
4.6 KiB
Rust
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<T> From<BlockingError<T>> for MyError
|
|
where
|
|
T: Into<MyError> + Debug,
|
|
{
|
|
fn from(e: BlockingError<T>) -> Self {
|
|
match e {
|
|
BlockingError::Error(e) => e.into(),
|
|
BlockingError::Canceled => MyError::Canceled,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> From<PoolError<T>> for MyError
|
|
where
|
|
T: Into<MyError>,
|
|
{
|
|
fn from(e: PoolError<T>) -> Self {
|
|
match e {
|
|
PoolError::Backend(e) => e.into(),
|
|
PoolError::Timeout(t) => MyError::DbTimeout(t),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<Infallible> for MyError {
|
|
fn from(i: Infallible) -> Self {
|
|
match i {}
|
|
}
|
|
}
|
|
|
|
impl From<rsa::errors::Error> for MyError {
|
|
fn from(e: rsa::errors::Error) -> Self {
|
|
MyError::Rsa(e)
|
|
}
|
|
}
|