- Run cargo fmt on all Rust crates for consistent formatting - Update CLAUDE.md with WASM plugin commands and dev.ps1 instructions - Update wiki: add WASM plugin architecture, rewrite dev environment docs - Minor frontend cleanup (unused imports)
98 lines
2.9 KiB
Rust
98 lines
2.9 KiB
Rust
use axum::Json;
|
||
use axum::http::StatusCode;
|
||
use axum::response::{IntoResponse, Response};
|
||
use serde::Serialize;
|
||
|
||
/// 统一错误响应格式
|
||
#[derive(Debug, Serialize)]
|
||
pub struct ErrorResponse {
|
||
pub error: String,
|
||
pub message: String,
|
||
#[serde(skip_serializing_if = "Option::is_none")]
|
||
pub details: Option<serde_json::Value>,
|
||
}
|
||
|
||
/// 平台级错误类型
|
||
#[derive(Debug, thiserror::Error)]
|
||
pub enum AppError {
|
||
#[error("资源未找到: {0}")]
|
||
NotFound(String),
|
||
|
||
#[error("验证失败: {0}")]
|
||
Validation(String),
|
||
|
||
#[error("未授权")]
|
||
Unauthorized,
|
||
|
||
#[error("禁止访问: {0}")]
|
||
Forbidden(String),
|
||
|
||
#[error("冲突: {0}")]
|
||
Conflict(String),
|
||
|
||
#[error("版本冲突: 数据已被其他操作修改,请刷新后重试")]
|
||
VersionMismatch,
|
||
|
||
#[error("请求过于频繁,请稍后重试")]
|
||
TooManyRequests,
|
||
|
||
#[error("内部错误: {0}")]
|
||
Internal(String),
|
||
}
|
||
|
||
impl IntoResponse for AppError {
|
||
fn into_response(self) -> Response {
|
||
let (status, message) = match &self {
|
||
AppError::NotFound(_) => (StatusCode::NOT_FOUND, self.to_string()),
|
||
AppError::Validation(_) => (StatusCode::BAD_REQUEST, self.to_string()),
|
||
AppError::Unauthorized => (StatusCode::UNAUTHORIZED, "未授权".to_string()),
|
||
AppError::Forbidden(_) => (StatusCode::FORBIDDEN, self.to_string()),
|
||
AppError::Conflict(_) => (StatusCode::CONFLICT, self.to_string()),
|
||
AppError::VersionMismatch => (StatusCode::CONFLICT, self.to_string()),
|
||
AppError::TooManyRequests => (StatusCode::TOO_MANY_REQUESTS, self.to_string()),
|
||
AppError::Internal(_) => (StatusCode::INTERNAL_SERVER_ERROR, "内部错误".to_string()),
|
||
};
|
||
|
||
let body = ErrorResponse {
|
||
error: status.canonical_reason().unwrap_or("Error").to_string(),
|
||
message,
|
||
details: None,
|
||
};
|
||
|
||
(status, Json(body)).into_response()
|
||
}
|
||
}
|
||
|
||
impl From<anyhow::Error> for AppError {
|
||
fn from(err: anyhow::Error) -> Self {
|
||
AppError::Internal(err.to_string())
|
||
}
|
||
}
|
||
|
||
impl From<sea_orm::DbErr> for AppError {
|
||
fn from(err: sea_orm::DbErr) -> Self {
|
||
match err {
|
||
sea_orm::DbErr::RecordNotFound(msg) => AppError::NotFound(msg),
|
||
sea_orm::DbErr::Query(sea_orm::RuntimeErr::SqlxError(e))
|
||
if e.to_string().contains("duplicate key") =>
|
||
{
|
||
AppError::Conflict("记录已存在".to_string())
|
||
}
|
||
_ => AppError::Internal(err.to_string()),
|
||
}
|
||
}
|
||
}
|
||
|
||
pub type AppResult<T> = Result<T, AppError>;
|
||
|
||
/// 检查乐观锁版本是否匹配。
|
||
///
|
||
/// 返回下一个版本号(actual + 1),或 VersionMismatch 错误。
|
||
pub fn check_version(expected: i32, actual: i32) -> AppResult<i32> {
|
||
if expected == actual {
|
||
Ok(actual + 1)
|
||
} else {
|
||
Err(AppError::VersionMismatch)
|
||
}
|
||
}
|