mirror of
https://github.com/robertwayne/axum-htmx
synced 2024-11-23 20:02:50 +01:00
Draft
This commit is contained in:
parent
535d57a3b5
commit
f622f12bba
4 changed files with 151 additions and 0 deletions
|
@ -20,6 +20,8 @@ serde = ["dep:serde", "dep:serde_json"]
|
||||||
axum-core = "0.4"
|
axum-core = "0.4"
|
||||||
http = { version = "1.0", default-features = false }
|
http = { version = "1.0", default-features = false }
|
||||||
async-trait = "0.1"
|
async-trait = "0.1"
|
||||||
|
axum = "0.7" # TODO: remove
|
||||||
|
tokio = { version = "1", features = ["sync"] } # TODO: hide behind a feature?
|
||||||
|
|
||||||
# Optional dependencies required for the `guards` feature.
|
# Optional dependencies required for the `guards` feature.
|
||||||
tower = { version = "0.4", default-features = false, optional = true }
|
tower = { version = "0.4", default-features = false, optional = true }
|
||||||
|
|
|
@ -137,6 +137,12 @@ where
|
||||||
type Rejection = std::convert::Infallible;
|
type Rejection = std::convert::Infallible;
|
||||||
|
|
||||||
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
||||||
|
use crate::vary_middleware::{HxRequestExtracted, Notifier};
|
||||||
|
parts
|
||||||
|
.extensions
|
||||||
|
.get_mut::<HxRequestExtracted>()
|
||||||
|
.map(Notifier::notify);
|
||||||
|
|
||||||
if parts.headers.contains_key(HX_REQUEST) {
|
if parts.headers.contains_key(HX_REQUEST) {
|
||||||
return Ok(HxRequest(true));
|
return Ok(HxRequest(true));
|
||||||
} else {
|
} else {
|
||||||
|
@ -164,6 +170,12 @@ where
|
||||||
type Rejection = std::convert::Infallible;
|
type Rejection = std::convert::Infallible;
|
||||||
|
|
||||||
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
async fn from_request_parts(parts: &mut Parts, _: &S) -> Result<Self, Self::Rejection> {
|
||||||
|
use crate::vary_middleware::{HxTargetExtracted, Notifier};
|
||||||
|
parts
|
||||||
|
.extensions
|
||||||
|
.get_mut::<HxTargetExtracted>()
|
||||||
|
.map(Notifier::notify);
|
||||||
|
|
||||||
if let Some(target) = parts.headers.get(HX_TARGET) {
|
if let Some(target) = parts.headers.get(HX_TARGET) {
|
||||||
if let Ok(target) = target.to_str() {
|
if let Ok(target) = target.to_str() {
|
||||||
return Ok(HxTarget(Some(target.to_string())));
|
return Ok(HxTarget(Some(target.to_string())));
|
||||||
|
|
|
@ -22,3 +22,6 @@ pub use guard::*;
|
||||||
pub use headers::*;
|
pub use headers::*;
|
||||||
#[doc(inline)]
|
#[doc(inline)]
|
||||||
pub use responders::*;
|
pub use responders::*;
|
||||||
|
|
||||||
|
pub(crate) mod vary_middleware;
|
||||||
|
pub use vary_middleware::vary_middleware;
|
||||||
|
|
134
src/vary_middleware.rs
Normal file
134
src/vary_middleware.rs
Normal file
|
@ -0,0 +1,134 @@
|
||||||
|
use crate::{
|
||||||
|
headers::{HX_REQUEST_STR, HX_TARGET_STR},
|
||||||
|
HxError,
|
||||||
|
};
|
||||||
|
use axum::{extract::Request, middleware::Next, response::Response};
|
||||||
|
use axum_core::response::IntoResponse;
|
||||||
|
use http::{
|
||||||
|
header::{HeaderValue, VARY},
|
||||||
|
Extensions,
|
||||||
|
};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use tokio::sync::oneshot::{self, Receiver, Sender};
|
||||||
|
|
||||||
|
const MIDDLEWARE_DOUBLE_USE: &str =
|
||||||
|
"Configuration error: `axum_httpx::vary_middleware` is used twice";
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub(crate) struct HxRequestExtracted(Option<Arc<Sender<()>>>);
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub(crate) struct HxTargetExtracted(Option<Arc<Sender<()>>>);
|
||||||
|
|
||||||
|
pub trait Notifier {
|
||||||
|
fn sender(&mut self) -> Option<Sender<()>>;
|
||||||
|
|
||||||
|
fn notify(&mut self) {
|
||||||
|
if let Some(sender) = self.sender().take() {
|
||||||
|
sender.send(()).ok();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Notifier for HxRequestExtracted {
|
||||||
|
fn sender(&mut self) -> Option<Sender<()>> {
|
||||||
|
self.0.take().and_then(Arc::into_inner)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Notifier for HxTargetExtracted {
|
||||||
|
fn sender(&mut self) -> Option<Sender<()>> {
|
||||||
|
self.0.take().and_then(Arc::into_inner)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HxRequestExtracted {
|
||||||
|
fn insert_into_extensions(extensions: &mut Extensions) -> Receiver<()> {
|
||||||
|
let (tx, rx) = oneshot::channel();
|
||||||
|
if extensions.insert(Self(Some(Arc::new(tx)))).is_some() {
|
||||||
|
panic!("{}", MIDDLEWARE_DOUBLE_USE);
|
||||||
|
}
|
||||||
|
rx
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HxTargetExtracted {
|
||||||
|
fn insert_into_extensions(extensions: &mut Extensions) -> Receiver<()> {
|
||||||
|
let (tx, rx) = oneshot::channel();
|
||||||
|
if extensions.insert(Self(Some(Arc::new(tx)))).is_some() {
|
||||||
|
panic!("{}", MIDDLEWARE_DOUBLE_USE);
|
||||||
|
}
|
||||||
|
rx
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn vary_middleware(mut request: Request, next: Next) -> Response {
|
||||||
|
let hx_request_rx = HxRequestExtracted::insert_into_extensions(request.extensions_mut());
|
||||||
|
let hx_target_rx = HxTargetExtracted::insert_into_extensions(request.extensions_mut());
|
||||||
|
|
||||||
|
let mut response = next.run(request).await;
|
||||||
|
|
||||||
|
let mut used = Vec::with_capacity(4);
|
||||||
|
if hx_request_rx.await.is_ok() {
|
||||||
|
used.push(HX_REQUEST_STR)
|
||||||
|
}
|
||||||
|
if hx_target_rx.await.is_ok() {
|
||||||
|
used.push(HX_TARGET_STR)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !used.is_empty() {
|
||||||
|
let value = match HeaderValue::from_str(&used.join(", ")) {
|
||||||
|
Ok(x) => x,
|
||||||
|
Err(e) => return HxError::from(e).into_response(),
|
||||||
|
};
|
||||||
|
if let Err(e) = response.headers_mut().try_append(VARY, value) {
|
||||||
|
return HxError::from(e).into_response();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
response
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use crate::{HxRequest, HxTarget};
|
||||||
|
use axum::{routing::get, Router};
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
fn vary_headers(resp: &axum_test::TestResponse) -> Vec<HeaderValue> {
|
||||||
|
resp.iter_headers_by_name("vary").cloned().collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn multiple_headers() {
|
||||||
|
let app = Router::new()
|
||||||
|
.route("/no-extractors", get(|| async { () }))
|
||||||
|
.route("/single-extractor", get(|_: HxRequest| async { () }))
|
||||||
|
// Extractors can be used multiple times e.g. in middlewares
|
||||||
|
.route(
|
||||||
|
"/repeated-extractor",
|
||||||
|
get(|_: HxRequest, _: HxRequest| async { () }),
|
||||||
|
)
|
||||||
|
.route(
|
||||||
|
"/multiple-extractors",
|
||||||
|
get(|_: HxRequest, _: HxTarget| async { () }),
|
||||||
|
)
|
||||||
|
.layer(axum::middleware::from_fn(vary_middleware));
|
||||||
|
let server = axum_test::TestServer::new(app).unwrap();
|
||||||
|
|
||||||
|
assert!(vary_headers(&server.get("/no-extractors").await).is_empty());
|
||||||
|
assert_eq!(
|
||||||
|
vary_headers(&server.get("/single-extractor").await),
|
||||||
|
[HX_REQUEST_STR]
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
vary_headers(&server.get("/repeated-extractor").await),
|
||||||
|
[HX_REQUEST_STR]
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
vary_headers(&server.get("/multiple-extractors").await),
|
||||||
|
[format!("{HX_REQUEST_STR}, {HX_TARGET_STR}")]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue