1
0
mirror of https://github.com/fafhrd91/actix-web synced 2025-02-23 04:23:02 +01:00
actix-web/src/middleware/defaultheaders.rs

237 lines
6.8 KiB
Rust
Raw Normal View History

2019-03-01 22:51:32 -08:00
//! Middleware for setting default response headers
2019-12-05 23:35:43 +06:00
use std::convert::TryFrom;
use std::future::Future;
use std::marker::PhantomData;
use std::pin::Pin;
2019-03-01 22:51:32 -08:00
use std::rc::Rc;
2019-11-20 23:33:22 +06:00
use std::task::{Context, Poll};
2017-12-03 20:47:15 -08:00
2019-03-04 21:37:57 -08:00
use actix_service::{Service, Transform};
use futures_util::future::{ready, Ready};
use futures_util::ready;
2019-03-01 22:51:32 -08:00
2019-03-06 19:19:27 -08:00
use crate::http::header::{HeaderName, HeaderValue, CONTENT_TYPE};
2019-12-05 23:35:43 +06:00
use crate::http::{Error as HttpError, HeaderMap};
2019-03-01 22:51:32 -08:00
use crate::service::{ServiceRequest, ServiceResponse};
2019-04-25 11:14:32 -07:00
use crate::Error;
2017-12-03 20:47:15 -08:00
/// `Middleware` for setting default response headers.
///
/// This middleware does not set header if response headers already contains it.
///
/// ```rust
/// use actix_web::{web, http, middleware, App, HttpResponse};
2017-12-03 20:47:15 -08:00
///
/// fn main() {
2018-03-31 00:16:55 -07:00
/// let app = App::new()
2019-03-25 13:02:10 -07:00
/// .wrap(middleware::DefaultHeaders::new().header("X-Version", "0.2"))
/// .service(
/// web::resource("/test")
/// .route(web::get().to(|| HttpResponse::Ok()))
/// .route(web::method(http::Method::HEAD).to(|| HttpResponse::MethodNotAllowed()))
/// );
2017-12-03 20:47:15 -08:00
/// }
/// ```
2019-03-01 22:51:32 -08:00
#[derive(Clone)]
2018-04-13 16:02:01 -07:00
pub struct DefaultHeaders {
2019-03-01 22:51:32 -08:00
inner: Rc<Inner>,
}
struct Inner {
ct: bool,
headers: HeaderMap,
}
2017-12-03 20:47:15 -08:00
2018-04-02 21:43:50 -07:00
impl Default for DefaultHeaders {
fn default() -> Self {
2018-04-13 16:02:01 -07:00
DefaultHeaders {
2019-03-01 22:51:32 -08:00
inner: Rc::new(Inner {
ct: false,
headers: HeaderMap::new(),
}),
2018-04-13 16:02:01 -07:00
}
2017-12-03 20:47:15 -08:00
}
}
2018-04-02 21:43:50 -07:00
impl DefaultHeaders {
/// Construct `DefaultHeaders` middleware.
pub fn new() -> DefaultHeaders {
DefaultHeaders::default()
2017-12-03 20:47:15 -08:00
}
/// Set a header.
#[inline]
2018-04-02 21:43:50 -07:00
pub fn header<K, V>(mut self, key: K, value: V) -> Self
2018-04-13 16:02:01 -07:00
where
2019-12-05 23:35:43 +06:00
HeaderName: TryFrom<K>,
<HeaderName as TryFrom<K>>::Error: Into<HttpError>,
HeaderValue: TryFrom<V>,
<HeaderValue as TryFrom<V>>::Error: Into<HttpError>,
2017-12-03 20:47:15 -08:00
{
2019-03-01 22:51:32 -08:00
#[allow(clippy::match_wild_err_arm)]
2018-04-02 21:43:50 -07:00
match HeaderName::try_from(key) {
2018-04-13 16:02:01 -07:00
Ok(key) => match HeaderValue::try_from(value) {
Ok(value) => {
2019-03-01 22:51:32 -08:00
Rc::get_mut(&mut self.inner)
.expect("Multiple copies exist")
.headers
.append(key, value);
2018-04-02 21:43:50 -07:00
}
2018-04-13 16:02:01 -07:00
Err(_) => panic!("Can not create header value"),
2018-04-02 21:43:50 -07:00
},
Err(_) => panic!("Can not create header name"),
2017-12-03 20:47:15 -08:00
}
self
}
/// Set *CONTENT-TYPE* header if response does not contain this header.
2018-04-02 21:43:50 -07:00
pub fn content_type(mut self) -> Self {
2019-03-01 22:51:32 -08:00
Rc::get_mut(&mut self.inner)
.expect("Multiple copies exist")
.ct = true;
self
}
2018-04-02 21:43:50 -07:00
}
impl<S, B> Transform<S> for DefaultHeaders
2019-03-01 22:51:32 -08:00
where
2019-04-25 11:14:32 -07:00
S: Service<Request = ServiceRequest, Response = ServiceResponse<B>, Error = Error>,
2019-03-01 22:51:32 -08:00
S::Future: 'static,
{
type Request = ServiceRequest;
2019-03-04 21:37:57 -08:00
type Response = ServiceResponse<B>;
2019-04-25 11:14:32 -07:00
type Error = Error;
2019-03-04 21:37:57 -08:00
type Transform = DefaultHeadersMiddleware<S>;
type InitError = ();
2019-11-20 23:33:22 +06:00
type Future = Ready<Result<Self::Transform, Self::InitError>>;
2019-03-04 21:37:57 -08:00
fn new_transform(&self, service: S) -> Self::Future {
ready(Ok(DefaultHeadersMiddleware {
2019-03-04 21:37:57 -08:00
service,
inner: self.inner.clone(),
}))
2017-12-03 20:47:15 -08:00
}
}
2019-03-04 21:37:57 -08:00
pub struct DefaultHeadersMiddleware<S> {
service: S,
inner: Rc<Inner>,
}
impl<S, B> Service for DefaultHeadersMiddleware<S>
2019-03-01 22:51:32 -08:00
where
2019-04-25 11:14:32 -07:00
S: Service<Request = ServiceRequest, Response = ServiceResponse<B>, Error = Error>,
2019-03-01 22:51:32 -08:00
S::Future: 'static,
{
type Request = ServiceRequest;
2019-03-01 22:51:32 -08:00
type Response = ServiceResponse<B>;
2019-04-25 11:14:32 -07:00
type Error = Error;
type Future = DefaultHeaderFuture<S, B>;
2019-03-01 22:51:32 -08:00
2019-12-08 00:46:51 +06:00
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
2019-11-20 23:33:22 +06:00
self.service.poll_ready(cx)
2019-03-01 22:51:32 -08:00
}
fn call(&mut self, req: ServiceRequest) -> Self::Future {
2019-03-01 22:51:32 -08:00
let inner = self.inner.clone();
2019-11-20 23:33:22 +06:00
let fut = self.service.call(req);
DefaultHeaderFuture {
fut,
inner,
_body: PhantomData,
}
}
}
2019-03-01 22:51:32 -08:00
#[pin_project::pin_project]
pub struct DefaultHeaderFuture<S: Service, B> {
#[pin]
fut: S::Future,
inner: Rc<Inner>,
_body: PhantomData<B>,
}
impl<S, B> Future for DefaultHeaderFuture<S, B>
where
S: Service<Response = ServiceResponse<B>, Error = Error>,
{
type Output = <S::Future as Future>::Output;
#[allow(clippy::borrow_interior_mutable_const)]
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let this = self.project();
let mut res = ready!(this.fut.poll(cx))?;
// set response headers
for (key, value) in this.inner.headers.iter() {
if !res.headers().contains_key(key) {
res.headers_mut().insert(key.clone(), value.clone());
2019-03-01 22:51:32 -08:00
}
2019-11-20 23:33:22 +06:00
}
// default content-type
if this.inner.ct && !res.headers().contains_key(&CONTENT_TYPE) {
res.headers_mut().insert(
CONTENT_TYPE,
HeaderValue::from_static("application/octet-stream"),
);
}
Poll::Ready(Ok(res))
2017-12-03 20:47:15 -08:00
}
}
2019-03-01 22:51:32 -08:00
#[cfg(test)]
mod tests {
2019-05-12 08:34:51 -07:00
use actix_service::IntoService;
2020-05-18 11:47:20 +09:00
use futures_util::future::ok;
use super::*;
use crate::dev::ServiceRequest;
2019-03-06 19:19:27 -08:00
use crate::http::header::CONTENT_TYPE;
2019-11-26 11:25:50 +06:00
use crate::test::{ok_service, TestRequest};
use crate::HttpResponse;
2019-11-26 11:25:50 +06:00
#[actix_rt::test]
async fn test_default_headers() {
let mut mw = DefaultHeaders::new()
.header(CONTENT_TYPE, "0001")
.new_transform(ok_service())
.await
.unwrap();
let req = TestRequest::default().to_srv_request();
let resp = mw.call(req).await.unwrap();
assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0001");
let req = TestRequest::default().to_srv_request();
let srv = |req: ServiceRequest| {
ok(req
.into_response(HttpResponse::Ok().header(CONTENT_TYPE, "0002").finish()))
};
let mut mw = DefaultHeaders::new()
.header(CONTENT_TYPE, "0001")
.new_transform(srv.into_service())
.await
.unwrap();
let resp = mw.call(req).await.unwrap();
assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0002");
}
2019-11-26 11:25:50 +06:00
#[actix_rt::test]
async fn test_content_type() {
let srv =
|req: ServiceRequest| ok(req.into_response(HttpResponse::Ok().finish()));
let mut mw = DefaultHeaders::new()
.content_type()
.new_transform(srv.into_service())
.await
.unwrap();
let req = TestRequest::default().to_srv_request();
let resp = mw.call(req).await.unwrap();
assert_eq!(
resp.headers().get(CONTENT_TYPE).unwrap(),
"application/octet-stream"
);
}
}