use crate::{HeaderValue, Request, Response, header::HeaderName, headers::HeaderEncode};
use rama_core::{Layer, Service};
use rama_http_headers::TypedHeader;
use rama_utils::macros::define_inner_service_accessors;
use std::fmt;
mod header;
use header::InsertHeaderMode;
pub use header::{
BoxMakeHeaderValueFactoryFn, BoxMakeHeaderValueFn, MakeHeaderValue, MakeHeaderValueDefault,
MakeHeaderValueFactory, MakeHeaderValueFactoryFn, MakeHeaderValueFn, TypedHeaderAsMaker,
};
#[derive(Clone)]
pub struct SetResponseHeaderLayer<M> {
header_name: HeaderName,
make: M,
mode: InsertHeaderMode,
}
impl<M> fmt::Debug for SetResponseHeaderLayer<M> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("SetResponseHeaderLayer")
.field("header_name", &self.header_name)
.field("mode", &self.mode)
.field("make", &std::any::type_name::<M>())
.finish()
}
}
impl<M> SetResponseHeaderLayer<M> {
#[inline(always)]
pub fn overriding(header_name: HeaderName, make: M) -> Self {
Self::new(header_name, make, InsertHeaderMode::Override)
}
#[inline(always)]
pub fn appending(header_name: HeaderName, make: M) -> Self {
Self::new(header_name, make, InsertHeaderMode::Append)
}
#[inline(always)]
pub fn if_not_present(header_name: HeaderName, make: M) -> Self {
Self::new(header_name, make, InsertHeaderMode::IfNotPresent)
}
#[inline(always)]
fn new(header_name: HeaderName, make: M, mode: InsertHeaderMode) -> Self {
Self {
make,
header_name,
mode,
}
}
}
impl SetResponseHeaderLayer<Option<HeaderValue>> {
#[inline(always)]
pub fn overriding_typed<H: HeaderEncode>(header: H) -> Self {
Self::overriding(H::name().clone(), header.encode_to_value())
}
#[inline(always)]
pub fn appending_typed<H: HeaderEncode>(header: H) -> Self {
Self::appending(H::name().clone(), header.encode_to_value())
}
#[inline(always)]
pub fn if_not_present_typed<H: HeaderEncode>(header: H) -> Self {
Self::if_not_present(H::name().clone(), header.encode_to_value())
}
}
impl<F, A> SetResponseHeaderLayer<BoxMakeHeaderValueFactoryFn<F, A>> {
#[inline(always)]
pub fn overriding_fn(header_name: HeaderName, make_fn: F) -> Self {
Self::new(
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::Override,
)
}
#[inline(always)]
pub fn appending_fn(header_name: HeaderName, make_fn: F) -> Self {
Self::new(
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::Append,
)
}
#[inline(always)]
pub fn if_not_present_fn(header_name: HeaderName, make_fn: F) -> Self {
Self::new(
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<M> SetResponseHeaderLayer<M> {
#[inline(always)]
pub fn overriding_default(
header_name: HeaderName,
) -> SetResponseHeaderLayer<MakeHeaderValueDefault<M>> {
SetResponseHeaderLayer::new(
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::Override,
)
}
#[inline(always)]
pub fn appending_default(
header_name: HeaderName,
) -> SetResponseHeaderLayer<MakeHeaderValueDefault<M>> {
SetResponseHeaderLayer::new(
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::Append,
)
}
#[inline(always)]
pub fn if_not_present_default(
header_name: HeaderName,
) -> SetResponseHeaderLayer<MakeHeaderValueDefault<M>> {
SetResponseHeaderLayer::new(
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<M: TypedHeader> SetResponseHeaderLayer<M> {
#[inline(always)]
#[must_use]
pub fn overriding_default_typed()
-> SetResponseHeaderLayer<MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeaderLayer::new(
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::Override,
)
}
#[inline(always)]
#[must_use]
pub fn appending_default_typed()
-> SetResponseHeaderLayer<MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeaderLayer::new(
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::Append,
)
}
#[inline(always)]
#[must_use]
pub fn if_not_present_default_typed()
-> SetResponseHeaderLayer<MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeaderLayer::new(
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<S, M> Layer<S> for SetResponseHeaderLayer<M>
where
M: Clone,
{
type Service = SetResponseHeader<S, M>;
fn layer(&self, inner: S) -> Self::Service {
SetResponseHeader {
inner,
header_name: self.header_name.clone(),
make: self.make.clone(),
mode: self.mode,
}
}
fn into_layer(self, inner: S) -> Self::Service {
SetResponseHeader {
inner,
header_name: self.header_name,
make: self.make,
mode: self.mode,
}
}
}
#[derive(Clone)]
pub struct SetResponseHeader<S, M> {
inner: S,
header_name: HeaderName,
make: M,
mode: InsertHeaderMode,
}
impl<S, H: HeaderEncode> SetResponseHeader<S, TypedHeaderAsMaker<H>> {
pub fn overriding_typed(inner: S, header: H) -> Self {
Self::overriding(inner, H::name().clone(), TypedHeaderAsMaker(header))
}
pub fn appending_typed(inner: S, header: H) -> Self {
Self::appending(inner, H::name().clone(), TypedHeaderAsMaker(header))
}
pub fn if_not_present_typed(inner: S, header: H) -> Self {
Self::if_not_present(inner, H::name().clone(), TypedHeaderAsMaker(header))
}
}
impl<S, M> SetResponseHeader<S, M> {
#[inline(always)]
pub fn overriding(inner: S, header_name: HeaderName, make: M) -> Self {
Self::new(inner, header_name, make, InsertHeaderMode::Override)
}
#[inline(always)]
pub fn appending(inner: S, header_name: HeaderName, make: M) -> Self {
Self::new(inner, header_name, make, InsertHeaderMode::Append)
}
#[inline(always)]
pub fn if_not_present(inner: S, header_name: HeaderName, make: M) -> Self {
Self::new(inner, header_name, make, InsertHeaderMode::IfNotPresent)
}
fn new(inner: S, header_name: HeaderName, make: M, mode: InsertHeaderMode) -> Self {
Self {
inner,
header_name,
make,
mode,
}
}
define_inner_service_accessors!();
}
impl<S, F, A> SetResponseHeader<S, BoxMakeHeaderValueFactoryFn<F, A>> {
#[inline(always)]
pub fn overriding_fn(inner: S, header_name: HeaderName, make_fn: F) -> Self {
Self::new(
inner,
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::Override,
)
}
#[inline(always)]
pub fn appending_fn(inner: S, header_name: HeaderName, make_fn: F) -> Self {
Self::new(
inner,
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::Append,
)
}
#[inline(always)]
pub fn if_not_present_fn(inner: S, header_name: HeaderName, make_fn: F) -> Self {
Self::new(
inner,
header_name,
BoxMakeHeaderValueFactoryFn::new(make_fn),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<S, M> SetResponseHeader<S, M> {
#[inline(always)]
pub fn overriding_default(
inner: S,
header_name: HeaderName,
) -> SetResponseHeader<S, MakeHeaderValueDefault<M>> {
SetResponseHeader::new(
inner,
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::Override,
)
}
#[inline(always)]
pub fn appending_default(
inner: S,
header_name: HeaderName,
) -> SetResponseHeader<S, MakeHeaderValueDefault<M>> {
SetResponseHeader::new(
inner,
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::Append,
)
}
#[inline(always)]
pub fn if_not_present_default(
inner: S,
header_name: HeaderName,
) -> SetResponseHeader<S, MakeHeaderValueDefault<M>> {
SetResponseHeader::new(
inner,
header_name,
MakeHeaderValueDefault::new(),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<S, M: TypedHeader> SetResponseHeader<S, M> {
#[inline(always)]
pub fn overriding_default_typed(
inner: S,
) -> SetResponseHeader<S, MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeader::new(
inner,
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::Override,
)
}
#[inline(always)]
pub fn appending_default_typed(
inner: S,
) -> SetResponseHeader<S, MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeader::new(
inner,
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::Append,
)
}
#[inline(always)]
pub fn if_not_present_default_typed(
inner: S,
) -> SetResponseHeader<S, MakeHeaderValueDefault<TypedHeaderAsMaker<M>>> {
SetResponseHeader::new(
inner,
M::name().clone(),
MakeHeaderValueDefault::new(),
InsertHeaderMode::IfNotPresent,
)
}
}
impl<S, M> fmt::Debug for SetResponseHeader<S, M>
where
S: fmt::Debug,
{
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("SetResponseHeader")
.field("inner", &self.inner)
.field("header_name", &self.header_name)
.field("mode", &self.mode)
.field("make", &std::any::type_name::<M>())
.finish()
}
}
impl<ReqBody, ResBody, S, M> Service<Request<ReqBody>> for SetResponseHeader<S, M>
where
ReqBody: Send + 'static,
ResBody: Send + 'static,
S: Service<Request<ReqBody>, Output = Response<ResBody>>,
M: MakeHeaderValueFactory<ReqBody, ResBody>,
{
type Output = S::Output;
type Error = S::Error;
async fn serve(&self, req: Request<ReqBody>) -> Result<Self::Output, Self::Error> {
let (req, header_maker) = self.make.make_header_value_maker(req).await;
let res = self.inner.serve(req).await?;
let res = self.mode.apply(&self.header_name, res, header_maker).await;
Ok(res)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Body, HeaderValue, Request, Response, header};
use rama_core::service::service_fn;
use std::convert::Infallible;
#[tokio::test]
async fn test_override_mode() {
let svc = SetResponseHeader::overriding(
service_fn(async || {
let res = Response::builder()
.header(header::CONTENT_TYPE, "good-content")
.body(Body::empty())
.unwrap();
Ok::<_, Infallible>(res)
}),
header::CONTENT_TYPE,
HeaderValue::from_static("text/html"),
);
let res = svc.serve(Request::new(Body::empty())).await.unwrap();
let mut values = res.headers().get_all(header::CONTENT_TYPE).iter();
assert_eq!(values.next().unwrap(), "text/html");
assert_eq!(values.next(), None);
}
#[tokio::test]
async fn test_append_mode() {
let svc = SetResponseHeader::appending(
service_fn(async || {
let res = Response::builder()
.header(header::CONTENT_TYPE, "good-content")
.body(Body::empty())
.unwrap();
Ok::<_, Infallible>(res)
}),
header::CONTENT_TYPE,
HeaderValue::from_static("text/html"),
);
let res = svc.serve(Request::new(Body::empty())).await.unwrap();
let mut values = res.headers().get_all(header::CONTENT_TYPE).iter();
assert_eq!(values.next().unwrap(), "good-content");
assert_eq!(values.next().unwrap(), "text/html");
assert_eq!(values.next(), None);
}
#[tokio::test]
async fn test_skip_if_present_mode() {
let svc = SetResponseHeader::if_not_present(
service_fn(async || {
let res = Response::builder()
.header(header::CONTENT_TYPE, "good-content")
.body(Body::empty())
.unwrap();
Ok::<_, Infallible>(res)
}),
header::CONTENT_TYPE,
HeaderValue::from_static("text/html"),
);
let res = svc.serve(Request::new(Body::empty())).await.unwrap();
let mut values = res.headers().get_all(header::CONTENT_TYPE).iter();
assert_eq!(values.next().unwrap(), "good-content");
assert_eq!(values.next(), None);
}
#[tokio::test]
async fn test_skip_if_present_mode_when_not_present() {
let svc = SetResponseHeader::if_not_present(
service_fn(async || {
let res = Response::builder().body(Body::empty()).unwrap();
Ok::<_, Infallible>(res)
}),
header::CONTENT_TYPE,
HeaderValue::from_static("text/html"),
);
let res = svc.serve(Request::new(Body::empty())).await.unwrap();
let mut values = res.headers().get_all(header::CONTENT_TYPE).iter();
assert_eq!(values.next().unwrap(), "text/html");
assert_eq!(values.next(), None);
}
}