2022-05-24 10:16:39 +00:00
|
|
|
use err_derive::Error;
|
|
|
|
use hyper::StatusCode;
|
|
|
|
|
|
|
|
use garage_util::error::Error as GarageError;
|
|
|
|
|
2024-02-05 18:27:12 +00:00
|
|
|
use garage_model::helper::error::Error as HelperError;
|
|
|
|
|
2022-05-24 10:16:39 +00:00
|
|
|
/// Errors of this crate
|
|
|
|
#[derive(Debug, Error)]
|
|
|
|
pub enum CommonError {
|
|
|
|
// ---- INTERNAL ERRORS ----
|
|
|
|
/// Error related to deeper parts of Garage
|
|
|
|
#[error(display = "Internal error: {}", _0)]
|
|
|
|
InternalError(#[error(source)] GarageError),
|
|
|
|
|
|
|
|
/// Error related to Hyper
|
|
|
|
#[error(display = "Internal error (Hyper error): {}", _0)]
|
|
|
|
Hyper(#[error(source)] hyper::Error),
|
|
|
|
|
|
|
|
/// Error related to HTTP
|
|
|
|
#[error(display = "Internal error (HTTP error): {}", _0)]
|
|
|
|
Http(#[error(source)] http::Error),
|
|
|
|
|
|
|
|
// ---- GENERIC CLIENT ERRORS ----
|
|
|
|
/// Proper authentication was not provided
|
|
|
|
#[error(display = "Forbidden: {}", _0)]
|
|
|
|
Forbidden(String),
|
|
|
|
|
|
|
|
/// Generic bad request response with custom message
|
|
|
|
#[error(display = "Bad request: {}", _0)]
|
|
|
|
BadRequest(String),
|
|
|
|
|
2024-02-05 18:27:12 +00:00
|
|
|
/// The client sent a header with invalid value
|
|
|
|
#[error(display = "Invalid header value: {}", _0)]
|
|
|
|
InvalidHeader(#[error(source)] hyper::header::ToStrError),
|
|
|
|
|
2022-05-24 10:16:39 +00:00
|
|
|
// ---- SPECIFIC ERROR CONDITIONS ----
|
|
|
|
// These have to be error codes referenced in the S3 spec here:
|
|
|
|
// https://docs.aws.amazon.com/AmazonS3/latest/API/ErrorResponses.html#ErrorCodeList
|
|
|
|
/// The bucket requested don't exists
|
|
|
|
#[error(display = "Bucket not found: {}", _0)]
|
|
|
|
NoSuchBucket(String),
|
|
|
|
|
|
|
|
/// Tried to create a bucket that already exist
|
|
|
|
#[error(display = "Bucket already exists")]
|
|
|
|
BucketAlreadyExists,
|
|
|
|
|
|
|
|
/// Tried to delete a non-empty bucket
|
|
|
|
#[error(display = "Tried to delete a non-empty bucket")]
|
|
|
|
BucketNotEmpty,
|
|
|
|
|
|
|
|
// Category: bad request
|
|
|
|
/// Bucket name is not valid according to AWS S3 specs
|
|
|
|
#[error(display = "Invalid bucket name: {}", _0)]
|
|
|
|
InvalidBucketName(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CommonError {
|
|
|
|
pub fn http_status_code(&self) -> StatusCode {
|
|
|
|
match self {
|
|
|
|
CommonError::InternalError(
|
2023-11-15 12:07:42 +00:00
|
|
|
GarageError::Timeout | GarageError::RemoteError(_) | GarageError::Quorum(..),
|
2022-05-24 10:16:39 +00:00
|
|
|
) => StatusCode::SERVICE_UNAVAILABLE,
|
|
|
|
CommonError::InternalError(_) | CommonError::Hyper(_) | CommonError::Http(_) => {
|
|
|
|
StatusCode::INTERNAL_SERVER_ERROR
|
|
|
|
}
|
|
|
|
CommonError::BadRequest(_) => StatusCode::BAD_REQUEST,
|
|
|
|
CommonError::Forbidden(_) => StatusCode::FORBIDDEN,
|
|
|
|
CommonError::NoSuchBucket(_) => StatusCode::NOT_FOUND,
|
|
|
|
CommonError::BucketNotEmpty | CommonError::BucketAlreadyExists => StatusCode::CONFLICT,
|
2024-02-05 18:27:12 +00:00
|
|
|
CommonError::InvalidBucketName(_) | CommonError::InvalidHeader(_) => {
|
|
|
|
StatusCode::BAD_REQUEST
|
|
|
|
}
|
2022-05-24 10:16:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn aws_code(&self) -> &'static str {
|
|
|
|
match self {
|
|
|
|
CommonError::Forbidden(_) => "AccessDenied",
|
|
|
|
CommonError::InternalError(
|
2023-11-15 12:07:42 +00:00
|
|
|
GarageError::Timeout | GarageError::RemoteError(_) | GarageError::Quorum(..),
|
2022-05-24 10:16:39 +00:00
|
|
|
) => "ServiceUnavailable",
|
|
|
|
CommonError::InternalError(_) | CommonError::Hyper(_) | CommonError::Http(_) => {
|
|
|
|
"InternalError"
|
|
|
|
}
|
|
|
|
CommonError::BadRequest(_) => "InvalidRequest",
|
|
|
|
CommonError::NoSuchBucket(_) => "NoSuchBucket",
|
|
|
|
CommonError::BucketAlreadyExists => "BucketAlreadyExists",
|
|
|
|
CommonError::BucketNotEmpty => "BucketNotEmpty",
|
|
|
|
CommonError::InvalidBucketName(_) => "InvalidBucketName",
|
2024-02-05 18:27:12 +00:00
|
|
|
CommonError::InvalidHeader(_) => "InvalidHeaderValue",
|
2022-05-24 10:16:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn bad_request<M: ToString>(msg: M) -> Self {
|
|
|
|
CommonError::BadRequest(msg.to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-05 18:27:12 +00:00
|
|
|
impl From<HelperError> for CommonError {
|
|
|
|
fn from(err: HelperError) -> Self {
|
|
|
|
match err {
|
|
|
|
HelperError::Internal(i) => Self::InternalError(i),
|
|
|
|
HelperError::BadRequest(b) => Self::BadRequest(b),
|
|
|
|
HelperError::InvalidBucketName(n) => Self::InvalidBucketName(n),
|
|
|
|
HelperError::NoSuchBucket(n) => Self::NoSuchBucket(n),
|
|
|
|
e => Self::bad_request(format!("{}", e)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-24 10:16:39 +00:00
|
|
|
pub trait CommonErrorDerivative: From<CommonError> {
|
|
|
|
fn internal_error<M: ToString>(msg: M) -> Self {
|
|
|
|
Self::from(CommonError::InternalError(GarageError::Message(
|
|
|
|
msg.to_string(),
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn bad_request<M: ToString>(msg: M) -> Self {
|
|
|
|
Self::from(CommonError::BadRequest(msg.to_string()))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn forbidden<M: ToString>(msg: M) -> Self {
|
|
|
|
Self::from(CommonError::Forbidden(msg.to_string()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Trait to map error to the Bad Request error code
|
|
|
|
pub trait OkOrBadRequest {
|
|
|
|
type S;
|
|
|
|
fn ok_or_bad_request<M: AsRef<str>>(self, reason: M) -> Result<Self::S, CommonError>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T, E> OkOrBadRequest for Result<T, E>
|
|
|
|
where
|
|
|
|
E: std::fmt::Display,
|
|
|
|
{
|
|
|
|
type S = T;
|
|
|
|
fn ok_or_bad_request<M: AsRef<str>>(self, reason: M) -> Result<T, CommonError> {
|
|
|
|
match self {
|
|
|
|
Ok(x) => Ok(x),
|
|
|
|
Err(e) => Err(CommonError::BadRequest(format!(
|
|
|
|
"{}: {}",
|
|
|
|
reason.as_ref(),
|
|
|
|
e
|
|
|
|
))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> OkOrBadRequest for Option<T> {
|
|
|
|
type S = T;
|
|
|
|
fn ok_or_bad_request<M: AsRef<str>>(self, reason: M) -> Result<T, CommonError> {
|
|
|
|
match self {
|
|
|
|
Some(x) => Ok(x),
|
|
|
|
None => Err(CommonError::BadRequest(reason.as_ref().to_string())),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Trait to map an error to an Internal Error code
|
|
|
|
pub trait OkOrInternalError {
|
|
|
|
type S;
|
|
|
|
fn ok_or_internal_error<M: AsRef<str>>(self, reason: M) -> Result<Self::S, CommonError>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T, E> OkOrInternalError for Result<T, E>
|
|
|
|
where
|
|
|
|
E: std::fmt::Display,
|
|
|
|
{
|
|
|
|
type S = T;
|
|
|
|
fn ok_or_internal_error<M: AsRef<str>>(self, reason: M) -> Result<T, CommonError> {
|
|
|
|
match self {
|
|
|
|
Ok(x) => Ok(x),
|
|
|
|
Err(e) => Err(CommonError::InternalError(GarageError::Message(format!(
|
|
|
|
"{}: {}",
|
|
|
|
reason.as_ref(),
|
|
|
|
e
|
|
|
|
)))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> OkOrInternalError for Option<T> {
|
|
|
|
type S = T;
|
|
|
|
fn ok_or_internal_error<M: AsRef<str>>(self, reason: M) -> Result<T, CommonError> {
|
|
|
|
match self {
|
|
|
|
Some(x) => Ok(x),
|
|
|
|
None => Err(CommonError::InternalError(GarageError::Message(
|
|
|
|
reason.as_ref().to_string(),
|
|
|
|
))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|