use prax_query::QueryError;
use thiserror::Error;
pub type SqlxResult<T> = Result<T, SqlxError>;
#[derive(Error, Debug)]
pub enum SqlxError {
#[error("Database error: {0}")]
Sqlx(#[from] sqlx::Error),
#[error("Configuration error: {0}")]
Config(String),
#[error("Connection error: {0}")]
Connection(String),
#[error("Query error: {0}")]
Query(String),
#[error("Deserialization error: {0}")]
Deserialization(String),
#[error("Type conversion error: {0}")]
TypeConversion(String),
#[error("Pool error: {0}")]
Pool(String),
#[error("Operation timed out after {0}ms")]
Timeout(u64),
#[error("Migration error: {0}")]
Migration(String),
#[error("Internal error: {0}")]
Internal(String),
}
impl From<SqlxError> for QueryError {
fn from(err: SqlxError) -> Self {
match err {
SqlxError::Sqlx(e) => {
let msg = e.to_string();
if msg.contains("connection") {
QueryError::connection(msg)
} else if msg.contains("constraint") || msg.contains("duplicate") {
QueryError::constraint_violation("unknown", msg)
} else {
QueryError::database(msg)
}
}
SqlxError::Config(msg) => QueryError::connection(msg),
SqlxError::Connection(msg) => QueryError::connection(msg),
SqlxError::Query(msg) => QueryError::database(msg),
SqlxError::Deserialization(msg) => QueryError::serialization(msg),
SqlxError::TypeConversion(msg) => QueryError::serialization(msg),
SqlxError::Pool(msg) => QueryError::connection(msg),
SqlxError::Timeout(ms) => QueryError::timeout(ms),
SqlxError::Migration(msg) => QueryError::database(msg),
SqlxError::Internal(msg) => QueryError::internal(msg),
}
}
}
impl SqlxError {
pub fn config(msg: impl Into<String>) -> Self {
Self::Config(msg.into())
}
pub fn connection(msg: impl Into<String>) -> Self {
Self::Connection(msg.into())
}
pub fn query(msg: impl Into<String>) -> Self {
Self::Query(msg.into())
}
pub fn pool(msg: impl Into<String>) -> Self {
Self::Pool(msg.into())
}
pub fn timeout(ms: u64) -> Self {
Self::Timeout(ms)
}
pub fn type_conversion(msg: impl Into<String>) -> Self {
Self::TypeConversion(msg.into())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_error_creation() {
let err = SqlxError::config("test config error");
assert!(matches!(err, SqlxError::Config(_)));
let err = SqlxError::connection("test connection error");
assert!(matches!(err, SqlxError::Connection(_)));
let err = SqlxError::timeout(5000);
assert!(matches!(err, SqlxError::Timeout(5000)));
}
#[test]
fn test_error_to_query_error() {
let err = SqlxError::connection("connection failed");
let query_err: QueryError = err.into();
assert!(query_err.to_string().contains("connection"));
let err = SqlxError::timeout(1000);
let query_err: QueryError = err.into();
assert!(
query_err.to_string().contains("timeout") || query_err.to_string().contains("1000")
);
}
}