use std::sync::PoisonError; use thiserror::Error; use tokio::task::JoinError; use tonic::Status; /// Errors related to communication with the store. #[derive(Debug, Error)] pub enum Error { #[error("invalid request: {0}")] InvalidRequest(String), #[error("internal storage error: {0}")] StorageError(String), } impl From> for Error { fn from(value: PoisonError) -> Self { Error::StorageError(value.to_string()) } } impl From for Error { fn from(value: JoinError) -> Self { Error::StorageError(value.to_string()) } } impl From for Status { fn from(value: Error) -> Self { match value { Error::InvalidRequest(msg) => Status::invalid_argument(msg), Error::StorageError(msg) => Status::data_loss(format!("storage error: {}", msg)), } } } impl From for Error { fn from(value: crate::tonic::Error) -> Self { Self::StorageError(value.to_string()) } } impl From for Error { fn from(value: std::io::Error) -> Self { if value.kind() == std::io::ErrorKind::InvalidInput { Error::InvalidRequest(value.to_string()) } else { Error::StorageError(value.to_string()) } } } // TODO: this should probably go somewhere else? impl From for std::io::Error { fn from(value: Error) -> Self { match value { Error::InvalidRequest(msg) => Self::new(std::io::ErrorKind::InvalidInput, msg), Error::StorageError(msg) => Self::new(std::io::ErrorKind::Other, msg), } } }