jmserver/src/v1/error.rs

52 lines
1.4 KiB
Rust
Raw Normal View History

2021-12-29 18:33:31 +01:00
use std::convert::Infallible;
use axum::{
body::{Bytes, Full},
2022-01-02 17:25:23 +01:00
extract::multipart::MultipartError,
2021-12-29 18:33:31 +01:00
response::IntoResponse,
Json,
};
use reqwest::StatusCode;
2022-01-02 17:25:23 +01:00
use thiserror::Error;
2021-12-29 18:33:31 +01:00
use super::models::ErrorResponse;
2022-01-02 17:25:23 +01:00
#[derive(Error, Debug)]
pub enum APIError {
#[error("SQL error: {0}")]
Sql(#[from] sqlx::Error),
#[error("Multipart form error: {0}")]
Multipart(#[from] MultipartError),
#[error("Bad request: {0}")]
BadRequest(String),
}
2021-12-29 18:33:31 +01:00
impl ErrorResponse {
fn new(status: StatusCode, message: Option<String>) -> Self {
2022-01-02 17:25:23 +01:00
let reason = status.canonical_reason().unwrap_or_default();
Self {
2021-12-29 18:33:31 +01:00
status,
2022-01-02 17:25:23 +01:00
error: message.unwrap_or(reason.to_string()),
2021-12-29 18:33:31 +01:00
}
}
}
2022-01-02 17:25:23 +01:00
impl IntoResponse for APIError {
2021-12-29 18:33:31 +01:00
type Body = Full<Bytes>;
type BodyError = Infallible;
fn into_response(self) -> axum::http::Response<Self::Body> {
2022-01-02 17:25:23 +01:00
let res = match self {
APIError::Sql(err) => match err {
sqlx::Error::RowNotFound => ErrorResponse::new(StatusCode::NOT_FOUND, None),
_ => ErrorResponse::new(StatusCode::INTERNAL_SERVER_ERROR, None),
},
APIError::Multipart(_) => ErrorResponse::new(StatusCode::INTERNAL_SERVER_ERROR, None),
APIError::BadRequest(err) => ErrorResponse::new(StatusCode::BAD_REQUEST, Some(err)),
};
let status = res.status.clone();
(status, Json(res)).into_response()
2021-12-29 18:33:31 +01:00
}
}