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

164 lines
4.8 KiB
Rust
Raw Normal View History

2019-03-01 22:51:32 -08:00
//! Middleware for setting default response headers
use std::rc::Rc;
2017-12-03 20:47:15 -08:00
2019-03-01 22:51:32 -08:00
use actix_http::http::header::{HeaderName, HeaderValue, CONTENT_TYPE};
use actix_http::http::{HeaderMap, HttpTryFrom};
use actix_service::{IntoNewTransform, Service, Transform};
use futures::{Async, Future, Poll};
use crate::middleware::MiddlewareFactory;
use crate::service::{ServiceRequest, ServiceResponse};
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.
///
2019-03-01 22:51:32 -08:00
/// ```rust,ignore
2017-12-06 11:00:39 -08:00
/// # extern crate actix_web;
2018-03-31 00:16:55 -07:00
/// use actix_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()
2018-06-01 09:37:14 -07:00
/// .middleware(middleware::DefaultHeaders::new().header("X-Version", "0.2"))
2017-12-03 20:47:15 -08:00
/// .resource("/test", |r| {
2018-06-01 09:37:14 -07:00
/// r.method(http::Method::GET).f(|_| HttpResponse::Ok());
/// r.method(http::Method::HEAD)
/// .f(|_| HttpResponse::MethodNotAllowed());
2019-03-01 22:51:32 -08:00
/// });
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
HeaderName: HttpTryFrom<K>,
HeaderValue: HttpTryFrom<V>,
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
}
2019-03-01 22:51:32 -08:00
impl<S, State, B> IntoNewTransform<MiddlewareFactory<DefaultHeaders, S>, S>
for DefaultHeaders
where
S: Service<Request = ServiceRequest<State>, Response = ServiceResponse<B>>,
S::Future: 'static,
{
fn into_new_transform(self) -> MiddlewareFactory<DefaultHeaders, S> {
MiddlewareFactory::new(self)
2017-12-03 20:47:15 -08:00
}
}
2019-03-01 22:51:32 -08:00
impl<S, State, B> Transform<S> for DefaultHeaders
where
S: Service<Request = ServiceRequest<State>, Response = ServiceResponse<B>>,
S::Future: 'static,
{
type Request = ServiceRequest<State>;
type Response = ServiceResponse<B>;
type Error = S::Error;
type Future = Box<Future<Item = Self::Response, Error = Self::Error>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, req: ServiceRequest<State>, srv: &mut S) -> Self::Future {
let inner = self.inner.clone();
Box::new(srv.call(req).map(move |mut res| {
// set response headers
for (key, value) in inner.headers.iter() {
if !res.headers().contains_key(key) {
res.headers_mut().insert(key, value.clone());
}
}
// default content-type
if inner.ct && !res.headers().contains_key(CONTENT_TYPE) {
res.headers_mut().insert(
CONTENT_TYPE,
HeaderValue::from_static("application/octet-stream"),
);
}
res
}))
2017-12-03 20:47:15 -08:00
}
}
2019-03-01 22:51:32 -08:00
// #[cfg(test)]
// mod tests {
// use super::*;
// use actix_http::http::header::CONTENT_TYPE;
// use actix_http::test::TestRequest;
// #[test]
// fn test_default_headers() {
// let mw = DefaultHeaders::new().header(CONTENT_TYPE, "0001");
// let req = TestRequest::default().finish();
// let resp = Response::Ok().finish();
// let resp = match mw.response(&req, resp) {
// Ok(Response::Done(resp)) => resp,
// _ => panic!(),
// };
// assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0001");
// let resp = Response::Ok().header(CONTENT_TYPE, "0002").finish();
// let resp = match mw.response(&req, resp) {
// Ok(Response::Done(resp)) => resp,
// _ => panic!(),
// };
// assert_eq!(resp.headers().get(CONTENT_TYPE).unwrap(), "0002");
// }
// }