2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2024-11-10 06:25:00 +00:00
pict-rs/src/error.rs

176 lines
4.6 KiB
Rust
Raw Normal View History

2021-06-19 19:39:41 +00:00
use actix_web::{http::StatusCode, HttpResponse, ResponseError};
2021-09-14 01:22:42 +00:00
use tracing_error::SpanTrace;
pub(crate) struct Error {
context: SpanTrace,
kind: UploadError,
}
impl std::fmt::Debug for Error {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
writeln!(f, "{}", self.kind)
2021-09-14 01:22:42 +00:00
}
}
impl std::fmt::Display for Error {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
writeln!(f, "{}", self.kind)?;
2021-09-14 01:22:42 +00:00
std::fmt::Display::fmt(&self.context, f)
}
}
impl std::error::Error for Error {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
self.kind.source()
}
}
impl<T> From<T> for Error
where
UploadError: From<T>,
{
fn from(error: T) -> Self {
Error {
kind: UploadError::from(error),
context: SpanTrace::capture(),
}
}
}
impl From<sled::transaction::TransactionError<Error>> for Error {
fn from(e: sled::transaction::TransactionError<Error>) -> Self {
match e {
sled::transaction::TransactionError::Abort(t) => t,
sled::transaction::TransactionError::Storage(e) => e.into(),
}
}
}
2020-06-06 21:41:17 +00:00
#[derive(Debug, thiserror::Error)]
2020-06-11 16:46:00 +00:00
pub(crate) enum UploadError {
2020-06-06 21:41:17 +00:00
#[error("Couln't upload file, {0}")]
2021-09-14 01:50:51 +00:00
Upload(#[from] actix_form_data::Error),
2020-06-06 21:41:17 +00:00
#[error("Error in DB, {0}")]
Db(#[from] sled::Error),
#[error("Error parsing string, {0}")]
ParseString(#[from] std::string::FromUtf8Error),
2021-01-14 01:08:04 +00:00
#[error("Error parsing request, {0}")]
ParseReq(String),
2020-06-07 00:29:15 +00:00
#[error("Error interacting with filesystem, {0}")]
Io(#[from] std::io::Error),
2021-10-19 01:29:06 +00:00
#[error(transparent)]
PathGenerator(#[from] storage_path_generator::PathError),
#[error(transparent)]
StripPrefix(#[from] std::path::StripPrefixError),
2021-10-23 04:48:56 +00:00
#[error(transparent)]
FileStore(#[from] crate::store::file_store::FileError),
#[cfg(feature = "object-storage")]
#[error(transparent)]
ObjectStore(#[from] crate::store::object_store::ObjectError),
2021-10-21 02:36:18 +00:00
#[error("Provided process path is invalid")]
ParsePath,
#[error("Failed to acquire the semaphore")]
Semaphore,
2020-06-06 21:41:17 +00:00
#[error("Panic in blocking operation")]
Canceled,
#[error("No files present in upload")]
NoFiles,
2020-06-06 22:43:33 +00:00
#[error("Requested a file that doesn't exist")]
MissingAlias,
#[error("Alias directed to missing file")]
MissingFile,
2020-06-07 00:29:15 +00:00
#[error("Provided token did not match expected token")]
InvalidToken,
2020-06-07 01:44:26 +00:00
#[error("Unsupported image format")]
UnsupportedFormat,
2021-09-14 01:22:42 +00:00
#[error("Invalid media dimensions")]
Dimensions,
#[error("Unable to download image, bad response {0}")]
Download(actix_web::http::StatusCode),
#[error("Unable to download image, {0}")]
2021-03-10 02:51:03 +00:00
Payload(#[from] awc::error::PayloadError),
#[error("Unable to send request, {0}")]
SendRequest(String),
#[error("No filename provided in request")]
MissingFilename,
#[error("Error converting Path to String")]
Path,
#[error("Tried to save an image with an already-taken name")]
DuplicateAlias,
2020-06-11 16:46:00 +00:00
#[error("{0}")]
Json(#[from] serde_json::Error),
#[error("Range header not satisfiable")]
Range,
2021-10-20 23:58:32 +00:00
#[error(transparent)]
Limit(#[from] super::LimitError),
}
2021-03-10 02:51:03 +00:00
impl From<awc::error::SendRequestError> for UploadError {
fn from(e: awc::error::SendRequestError) -> Self {
UploadError::SendRequest(e.to_string())
}
2020-06-07 00:29:15 +00:00
}
2021-02-10 22:57:42 +00:00
impl From<actix_web::error::BlockingError> for UploadError {
fn from(_: actix_web::error::BlockingError) -> Self {
UploadError::Canceled
2020-06-06 21:41:17 +00:00
}
}
impl From<tokio::sync::AcquireError> for UploadError {
fn from(_: tokio::sync::AcquireError) -> Self {
UploadError::Semaphore
}
}
2021-09-14 01:22:42 +00:00
impl ResponseError for Error {
2020-06-06 21:41:17 +00:00
fn status_code(&self) -> StatusCode {
2021-09-14 01:22:42 +00:00
match self.kind {
UploadError::DuplicateAlias
2021-10-20 23:58:32 +00:00
| UploadError::Limit(_)
| UploadError::NoFiles
2021-01-14 01:08:04 +00:00
| UploadError::Upload(_)
| UploadError::ParseReq(_) => StatusCode::BAD_REQUEST,
UploadError::MissingAlias | UploadError::MissingFilename => StatusCode::NOT_FOUND,
2020-06-07 00:29:15 +00:00
UploadError::InvalidToken => StatusCode::FORBIDDEN,
UploadError::Range => StatusCode::RANGE_NOT_SATISFIABLE,
2020-06-06 21:41:17 +00:00
_ => StatusCode::INTERNAL_SERVER_ERROR,
}
}
2021-06-19 19:39:41 +00:00
fn error_response(&self) -> HttpResponse {
HttpResponse::build(self.status_code())
2021-04-17 19:59:42 +00:00
.content_type("application/json")
.body(
2021-09-14 01:22:42 +00:00
serde_json::to_string(&serde_json::json!({ "msg": self.kind.to_string() }))
.unwrap_or_else(|_| r#"{"msg":"Request failed"}"#.to_string()),
2021-04-17 19:59:42 +00:00
)
2020-06-06 21:41:17 +00:00
}
}