Redirect on HxRequest guard failures

This commit is contained in:
Rob Wagner 2023-07-29 16:04:51 -04:00
parent 35f86927e7
commit a07426695a
No known key found for this signature in database
GPG key ID: 53CCB4497B15CF61

View file

@ -8,7 +8,7 @@ use std::{
}; };
use axum::{ use axum::{
http::{Request, StatusCode}, http::{header::LOCATION, Request, StatusCode},
response::Response, response::Response,
}; };
use futures_core::ready; use futures_core::ready;
@ -22,41 +22,50 @@ use crate::HX_REQUEST;
/// ///
/// This can be used to protect routes that should only be accessed via htmx /// This can be used to protect routes that should only be accessed via htmx
/// requests. /// requests.
#[derive(Default, Debug, Clone)] #[derive(Debug, Clone)]
pub struct HxRequestGuardLayer; pub struct HxRequestGuardLayer<'a> {
redirect_to: &'a str,
}
impl HxRequestGuardLayer { impl<'a> HxRequestGuardLayer<'a> {
#[allow(clippy::default_constructed_unit_structs)] pub fn new(redirect_to: &'a str) -> Self {
pub fn new() -> Self { Self { redirect_to }
Self::default()
} }
} }
impl<S> Layer<S> for HxRequestGuardLayer { impl Default for HxRequestGuardLayer<'_> {
type Service = HxRequestGuard<S>; fn default() -> Self {
Self { redirect_to: "/" }
}
}
impl<'a, S> Layer<S> for HxRequestGuardLayer<'a> {
type Service = HxRequestGuard<'a, S>;
fn layer(&self, inner: S) -> Self::Service { fn layer(&self, inner: S) -> Self::Service {
HxRequestGuard { HxRequestGuard {
inner, inner,
hx_request: false, hx_request: false,
layer: self.clone(),
} }
} }
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct HxRequestGuard<S> { pub struct HxRequestGuard<'a, S> {
inner: S, inner: S,
hx_request: bool, hx_request: bool,
layer: HxRequestGuardLayer<'a>,
} }
impl<S, T, U> Service<Request<T>> for HxRequestGuard<S> impl<'a, S, T, U> Service<Request<T>> for HxRequestGuard<'a, S>
where where
S: Service<Request<T>, Response = Response<U>>, S: Service<Request<T>, Response = Response<U>>,
U: Default, U: Default,
{ {
type Response = S::Response; type Response = S::Response;
type Error = S::Error; type Error = S::Error;
type Future = ResponseFuture<S::Future>; type Future = ResponseFuture<'a, S::Future>;
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> { fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
self.inner.poll_ready(cx) self.inner.poll_ready(cx)
@ -73,19 +82,21 @@ where
ResponseFuture { ResponseFuture {
response_future, response_future,
hx_request: self.hx_request, hx_request: self.hx_request,
layer: self.layer.clone(),
} }
} }
} }
pin_project! { pin_project! {
pub struct ResponseFuture<F> { pub struct ResponseFuture<'a, F> {
#[pin] #[pin]
response_future: F, response_future: F,
hx_request: bool, hx_request: bool,
layer: HxRequestGuardLayer<'a>,
} }
} }
impl<F, B, E> Future for ResponseFuture<F> impl<'a, F, B, E> Future for ResponseFuture<'a, F>
where where
F: Future<Output = Result<Response<B>, E>>, F: Future<Output = Result<Response<B>, E>>,
B: Default, B: Default,
@ -99,8 +110,11 @@ where
match *this.hx_request { match *this.hx_request {
true => Poll::Ready(Ok(response)), true => Poll::Ready(Ok(response)),
false => { false => {
let mut res = Response::new(B::default()); let res = Response::builder()
*res.status_mut() = StatusCode::FORBIDDEN; .status(StatusCode::SEE_OTHER)
.header(LOCATION, this.layer.redirect_to)
.body(B::default())
.expect("failed to build response");
Poll::Ready(Ok(res)) Poll::Ready(Ok(res))
} }