use std::future::Future;
use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, Poll};
use axum::http::{Request, StatusCode};
use axum::response::{IntoResponse, Response};
use pin_project_lite::pin_project;
use tower::{Layer, Service};
#[derive(Clone, Debug)]
pub struct AutumnErrorInfo {
pub status: StatusCode,
pub message: String,
pub details: Option<std::collections::HashMap<String, Vec<String>>>,
}
impl AutumnErrorInfo {
#[must_use]
pub fn into_default_response(self) -> Response {
let body = serde_json::json!({
"error": {
"status": self.status.as_u16(),
"message": self.message,
"details": self.details,
}
});
(self.status, axum::Json(body)).into_response()
}
}
pub trait ExceptionFilter: Send + Sync + 'static {
fn filter(&self, error: &AutumnErrorInfo, response: Response) -> Response;
}
#[derive(Clone)]
pub struct ExceptionFilterLayer {
filters: Arc<Vec<Arc<dyn ExceptionFilter>>>,
}
impl ExceptionFilterLayer {
#[must_use]
pub fn new(filters: Vec<Arc<dyn ExceptionFilter>>) -> Self {
Self {
filters: Arc::new(filters),
}
}
}
impl<S> Layer<S> for ExceptionFilterLayer {
type Service = ExceptionFilterService<S>;
fn layer(&self, inner: S) -> Self::Service {
ExceptionFilterService {
inner,
filters: Arc::clone(&self.filters),
}
}
}
#[derive(Clone)]
pub struct ExceptionFilterService<S> {
inner: S,
filters: Arc<Vec<Arc<dyn ExceptionFilter>>>,
}
impl<S, ReqBody> Service<Request<ReqBody>> for ExceptionFilterService<S>
where
S: Service<Request<ReqBody>, Response = Response>,
{
type Response = Response;
type Error = S::Error;
type Future = ExceptionFilterFuture<S::Future>;
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
self.inner.poll_ready(cx)
}
fn call(&mut self, req: Request<ReqBody>) -> Self::Future {
ExceptionFilterFuture {
inner: self.inner.call(req),
filters: Arc::clone(&self.filters),
}
}
}
pin_project! {
pub struct ExceptionFilterFuture<F> {
#[pin]
inner: F,
filters: Arc<Vec<Arc<dyn ExceptionFilter>>>,
}
}
impl<F, E> Future for ExceptionFilterFuture<F>
where
F: Future<Output = Result<Response, E>>,
{
type Output = Result<Response, E>;
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let this = self.project();
match this.inner.poll(cx) {
Poll::Ready(Ok(response)) => {
if let Some(error_info) = response.extensions().get::<AutumnErrorInfo>().cloned() {
let mut response = response;
let filters = this.filters;
for filter in filters.iter() {
response = filter.filter(&error_info, response);
}
Poll::Ready(Ok(response))
} else {
Poll::Ready(Ok(response))
}
}
Poll::Ready(Err(e)) => Poll::Ready(Err(e)),
Poll::Pending => Poll::Pending,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use axum::Router;
use axum::body::Body;
use axum::routing::get;
use http::Request;
use tower::ServiceExt;
use crate::error::AutumnError;
#[tokio::test]
async fn filter_receives_error_info() {
use std::sync::atomic::{AtomicBool, Ordering};
static CALLED: AtomicBool = AtomicBool::new(false);
struct TestFilter;
impl ExceptionFilter for TestFilter {
fn filter(&self, error: &AutumnErrorInfo, response: Response) -> Response {
assert_eq!(error.status, StatusCode::NOT_FOUND);
assert_eq!(error.message, "not here");
CALLED.store(true, Ordering::SeqCst);
response
}
}
let app = Router::new()
.route(
"/",
get(|| async {
Err::<String, AutumnError>(AutumnError::not_found_msg("not here"))
}),
)
.layer(ExceptionFilterLayer::new(vec![Arc::new(TestFilter)]));
let response = app
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(response.status(), StatusCode::NOT_FOUND);
assert!(CALLED.load(Ordering::SeqCst));
}
#[tokio::test]
async fn filter_can_replace_response() {
struct ReplaceFilter;
impl ExceptionFilter for ReplaceFilter {
fn filter(&self, _error: &AutumnErrorInfo, _response: Response) -> Response {
(StatusCode::SERVICE_UNAVAILABLE, "custom error page").into_response()
}
}
let app = Router::new()
.route(
"/",
get(|| async { Err::<String, AutumnError>(AutumnError::not_found_msg("gone")) }),
)
.layer(ExceptionFilterLayer::new(vec![Arc::new(ReplaceFilter)]));
let response = app
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(response.status(), StatusCode::SERVICE_UNAVAILABLE);
let body = axum::body::to_bytes(response.into_body(), usize::MAX)
.await
.unwrap();
assert_eq!(&body[..], b"custom error page");
}
#[tokio::test]
async fn success_responses_bypass_filters() {
use std::sync::atomic::{AtomicBool, Ordering};
static CALLED: AtomicBool = AtomicBool::new(false);
struct NeverFilter;
impl ExceptionFilter for NeverFilter {
fn filter(&self, _error: &AutumnErrorInfo, response: Response) -> Response {
CALLED.store(true, Ordering::SeqCst);
response
}
}
let app = Router::new()
.route("/", get(|| async { "ok" }))
.layer(ExceptionFilterLayer::new(vec![Arc::new(NeverFilter)]));
let response = app
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(response.status(), StatusCode::OK);
assert!(!CALLED.load(Ordering::SeqCst));
}
#[tokio::test]
async fn multiple_filters_run_in_order() {
use std::sync::atomic::{AtomicU32, Ordering};
static COUNTER: AtomicU32 = AtomicU32::new(0);
struct OrderFilter(u32);
impl ExceptionFilter for OrderFilter {
fn filter(&self, _error: &AutumnErrorInfo, response: Response) -> Response {
let current = COUNTER.fetch_add(1, Ordering::SeqCst);
assert_eq!(current, self.0, "filters should run in registration order");
response
}
}
COUNTER.store(0, Ordering::SeqCst);
let app = Router::new()
.route(
"/",
get(|| async { Err::<String, AutumnError>(AutumnError::bad_request_msg("oops")) }),
)
.layer(ExceptionFilterLayer::new(vec![
Arc::new(OrderFilter(0)),
Arc::new(OrderFilter(1)),
Arc::new(OrderFilter(2)),
]));
app.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(COUNTER.load(Ordering::SeqCst), 3);
}
#[test]
fn error_info_into_default_response() {
let info = AutumnErrorInfo {
status: StatusCode::NOT_FOUND,
message: "not found".into(),
details: None,
};
let response = info.into_default_response();
assert_eq!(response.status(), StatusCode::NOT_FOUND);
}
}