2018-01-09 22:33:51 -08:00
|
|
|
|
//! Cross-origin resource sharing (CORS) for Actix applications
|
2018-01-09 23:55:42 -08:00
|
|
|
|
//!
|
|
|
|
|
//! CORS middleware could be used with application and with resource.
|
|
|
|
|
//! First you need to construct CORS middleware instance.
|
|
|
|
|
//!
|
|
|
|
|
//! To construct a cors:
|
|
|
|
|
//!
|
|
|
|
|
//! 1. Call [`Cors::build`](struct.Cors.html#method.build) to start building.
|
|
|
|
|
//! 2. Use any of the builder methods to set fields in the backend.
|
|
|
|
|
//! 3. Call [finish](struct.Cors.html#method.finish) to retrieve the constructed backend.
|
|
|
|
|
//!
|
2018-03-31 00:16:55 -07:00
|
|
|
|
//! Cors middleware could be used as parameter for `App::middleware()` or
|
2018-04-09 21:39:32 -07:00
|
|
|
|
//! `ResourceHandler::middleware()` methods. But you have to use
|
|
|
|
|
//! `Cors::for_app()` method to support *preflight* OPTIONS request.
|
2018-01-11 11:14:18 -08:00
|
|
|
|
//!
|
2018-01-09 23:55:42 -08:00
|
|
|
|
//!
|
|
|
|
|
//! # Example
|
|
|
|
|
//!
|
|
|
|
|
//! ```rust
|
|
|
|
|
//! # extern crate actix_web;
|
2018-03-31 00:16:55 -07:00
|
|
|
|
//! use actix_web::{http, App, HttpRequest, HttpResponse};
|
2018-04-09 21:39:32 -07:00
|
|
|
|
//! use actix_web::middleware::cors::Cors;
|
2018-01-09 23:55:42 -08:00
|
|
|
|
//!
|
|
|
|
|
//! fn index(mut req: HttpRequest) -> &'static str {
|
|
|
|
|
//! "Hello world"
|
|
|
|
|
//! }
|
|
|
|
|
//!
|
|
|
|
|
//! fn main() {
|
2018-03-31 00:16:55 -07:00
|
|
|
|
//! let app = App::new()
|
2018-04-09 21:39:32 -07:00
|
|
|
|
//! .configure(|app| Cors::for_app(app) // <- Construct CORS middleware builder
|
|
|
|
|
//! .allowed_origin("https://www.rust-lang.org/")
|
|
|
|
|
//! .allowed_methods(vec!["GET", "POST"])
|
|
|
|
|
//! .allowed_headers(vec![http::header::AUTHORIZATION, http::header::ACCEPT])
|
|
|
|
|
//! .allowed_header(http::header::CONTENT_TYPE)
|
|
|
|
|
//! .max_age(3600)
|
|
|
|
|
//! .resource("/index.html", |r| {
|
|
|
|
|
//! r.method(http::Method::GET).f(|_| HttpResponse::Ok());
|
|
|
|
|
//! r.method(http::Method::HEAD).f(|_| HttpResponse::MethodNotAllowed());
|
|
|
|
|
//! })
|
|
|
|
|
//! .register());
|
2018-01-09 23:55:42 -08:00
|
|
|
|
//! }
|
|
|
|
|
//! ```
|
|
|
|
|
//! In this example custom *CORS* middleware get registered for "/index.html" endpoint.
|
|
|
|
|
//!
|
|
|
|
|
//! Cors middleware automatically handle *OPTIONS* preflight request.
|
2018-01-09 22:33:51 -08:00
|
|
|
|
use std::collections::HashSet;
|
2018-01-10 13:41:33 -08:00
|
|
|
|
use std::iter::FromIterator;
|
2018-04-09 14:20:12 -07:00
|
|
|
|
use std::rc::Rc;
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
2018-03-30 23:07:33 -07:00
|
|
|
|
use http::{self, Method, HttpTryFrom, Uri, StatusCode};
|
2018-01-10 13:41:33 -08:00
|
|
|
|
use http::header::{self, HeaderName, HeaderValue};
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
2018-04-09 21:11:15 -07:00
|
|
|
|
use application::App;
|
2018-01-09 22:48:35 -08:00
|
|
|
|
use error::{Result, ResponseError};
|
2018-04-01 17:37:22 -07:00
|
|
|
|
use resource::ResourceHandler;
|
2018-02-27 15:03:28 -08:00
|
|
|
|
use httpmessage::HttpMessage;
|
2018-01-09 22:33:51 -08:00
|
|
|
|
use httprequest::HttpRequest;
|
|
|
|
|
use httpresponse::HttpResponse;
|
2018-01-10 13:41:33 -08:00
|
|
|
|
use middleware::{Middleware, Response, Started};
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
|
|
|
|
/// A set of errors that can occur during processing CORS
|
|
|
|
|
#[derive(Debug, Fail)]
|
2018-01-10 14:20:00 -08:00
|
|
|
|
pub enum CorsError {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
/// The HTTP request header `Origin` is required but was not provided
|
|
|
|
|
#[fail(display="The HTTP request header `Origin` is required but was not provided")]
|
|
|
|
|
MissingOrigin,
|
|
|
|
|
/// The HTTP request header `Origin` could not be parsed correctly.
|
|
|
|
|
#[fail(display="The HTTP request header `Origin` could not be parsed correctly.")]
|
|
|
|
|
BadOrigin,
|
|
|
|
|
/// The request header `Access-Control-Request-Method` is required but is missing
|
|
|
|
|
#[fail(display="The request header `Access-Control-Request-Method` is required but is missing")]
|
|
|
|
|
MissingRequestMethod,
|
|
|
|
|
/// The request header `Access-Control-Request-Method` has an invalid value
|
|
|
|
|
#[fail(display="The request header `Access-Control-Request-Method` has an invalid value")]
|
|
|
|
|
BadRequestMethod,
|
2018-01-09 23:55:42 -08:00
|
|
|
|
/// The request header `Access-Control-Request-Headers` has an invalid value
|
|
|
|
|
#[fail(display="The request header `Access-Control-Request-Headers` has an invalid value")]
|
|
|
|
|
BadRequestHeaders,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
/// The request header `Access-Control-Request-Headers` is required but is missing.
|
|
|
|
|
#[fail(display="The request header `Access-Control-Request-Headers` is required but is
|
|
|
|
|
missing")]
|
|
|
|
|
MissingRequestHeaders,
|
|
|
|
|
/// Origin is not allowed to make this request
|
|
|
|
|
#[fail(display="Origin is not allowed to make this request")]
|
|
|
|
|
OriginNotAllowed,
|
|
|
|
|
/// Requested method is not allowed
|
|
|
|
|
#[fail(display="Requested method is not allowed")]
|
|
|
|
|
MethodNotAllowed,
|
|
|
|
|
/// One or more headers requested are not allowed
|
|
|
|
|
#[fail(display="One or more headers requested are not allowed")]
|
|
|
|
|
HeadersNotAllowed,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 14:20:00 -08:00
|
|
|
|
impl ResponseError for CorsError {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
|
|
|
|
fn error_response(&self) -> HttpResponse {
|
2018-03-30 23:07:33 -07:00
|
|
|
|
HttpResponse::with_body(StatusCode::BAD_REQUEST, format!("{}", self))
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// An enum signifying that some of type T is allowed, or `All` (everything is allowed).
|
|
|
|
|
///
|
|
|
|
|
/// `Default` is implemented for this enum and is `All`.
|
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
|
|
|
pub enum AllOrSome<T> {
|
|
|
|
|
/// Everything is allowed. Usually equivalent to the "*" value.
|
|
|
|
|
All,
|
|
|
|
|
/// Only some of `T` is allowed
|
|
|
|
|
Some(T),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<T> Default for AllOrSome<T> {
|
|
|
|
|
fn default() -> Self {
|
|
|
|
|
AllOrSome::All
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<T> AllOrSome<T> {
|
|
|
|
|
/// Returns whether this is an `All` variant
|
|
|
|
|
pub fn is_all(&self) -> bool {
|
|
|
|
|
match *self {
|
|
|
|
|
AllOrSome::All => true,
|
|
|
|
|
AllOrSome::Some(_) => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns whether this is a `Some` variant
|
|
|
|
|
pub fn is_some(&self) -> bool {
|
|
|
|
|
!self.is_all()
|
|
|
|
|
}
|
2018-01-09 23:55:42 -08:00
|
|
|
|
|
|
|
|
|
/// Returns &T
|
|
|
|
|
pub fn as_ref(&self) -> Option<&T> {
|
|
|
|
|
match *self {
|
|
|
|
|
AllOrSome::All => None,
|
|
|
|
|
AllOrSome::Some(ref t) => Some(t),
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// `Middleware` for Cross-origin resource sharing support
|
|
|
|
|
///
|
|
|
|
|
/// The Cors struct contains the settings for CORS requests to be validated and
|
|
|
|
|
/// for responses to be generated.
|
2018-04-09 14:20:12 -07:00
|
|
|
|
#[derive(Clone)]
|
2018-01-09 22:33:51 -08:00
|
|
|
|
pub struct Cors {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
inner: Rc<Inner>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct Inner {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
methods: HashSet<Method>,
|
2018-01-11 20:11:34 -08:00
|
|
|
|
origins: AllOrSome<HashSet<String>>,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
origins_str: Option<HeaderValue>,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
headers: AllOrSome<HashSet<HeaderName>>,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
expose_hdrs: Option<String>,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
max_age: Option<usize>,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
preflight: bool,
|
|
|
|
|
send_wildcard: bool,
|
|
|
|
|
supports_credentials: bool,
|
|
|
|
|
vary_header: bool,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Default for Cors {
|
|
|
|
|
fn default() -> Cors {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
let inner = Inner {
|
2018-01-10 14:20:00 -08:00
|
|
|
|
origins: AllOrSome::default(),
|
2018-01-10 13:41:33 -08:00
|
|
|
|
origins_str: None,
|
|
|
|
|
methods: HashSet::from_iter(
|
|
|
|
|
vec![Method::GET, Method::HEAD,
|
|
|
|
|
Method::POST, Method::OPTIONS, Method::PUT,
|
|
|
|
|
Method::PATCH, Method::DELETE].into_iter()),
|
|
|
|
|
headers: AllOrSome::All,
|
|
|
|
|
expose_hdrs: None,
|
|
|
|
|
max_age: None,
|
|
|
|
|
preflight: true,
|
|
|
|
|
send_wildcard: false,
|
|
|
|
|
supports_credentials: false,
|
|
|
|
|
vary_header: true,
|
2018-04-09 14:20:12 -07:00
|
|
|
|
};
|
|
|
|
|
Cors{inner: Rc::new(inner)}
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Cors {
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn build() -> CorsBuilder<()> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
CorsBuilder {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
cors: Some(Inner {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
origins: AllOrSome::All,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
origins_str: None,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
methods: HashSet::new(),
|
|
|
|
|
headers: AllOrSome::All,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
expose_hdrs: None,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
max_age: None,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
preflight: true,
|
|
|
|
|
send_wildcard: false,
|
|
|
|
|
supports_credentials: false,
|
|
|
|
|
vary_header: true,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}),
|
|
|
|
|
methods: false,
|
|
|
|
|
error: None,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
expose_hdrs: HashSet::new(),
|
2018-04-09 21:11:15 -07:00
|
|
|
|
resources: Vec::new(),
|
|
|
|
|
app: None,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Create CorsBuilder for a specified application.
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// # extern crate actix_web;
|
|
|
|
|
/// use actix_web::{http, App, HttpResponse};
|
|
|
|
|
/// use actix_web::middleware::cors::Cors;
|
|
|
|
|
///
|
|
|
|
|
/// fn main() {
|
|
|
|
|
/// let app = App::new()
|
|
|
|
|
/// .configure(|app| Cors::for_app(app) // <- Construct CORS builder
|
|
|
|
|
/// .allowed_origin("https://www.rust-lang.org/")
|
|
|
|
|
/// .resource("/resource", |r| { // register resource
|
|
|
|
|
/// r.method(http::Method::GET).f(|_| HttpResponse::Ok());
|
|
|
|
|
/// })
|
|
|
|
|
/// .register() // construct CORS and return application instance
|
|
|
|
|
/// );
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn for_app<S: 'static>(app: App<S>) -> CorsBuilder<S> {
|
|
|
|
|
CorsBuilder {
|
|
|
|
|
cors: Some(Inner {
|
|
|
|
|
origins: AllOrSome::All,
|
|
|
|
|
origins_str: None,
|
|
|
|
|
methods: HashSet::new(),
|
|
|
|
|
headers: AllOrSome::All,
|
|
|
|
|
expose_hdrs: None,
|
|
|
|
|
max_age: None,
|
|
|
|
|
preflight: true,
|
|
|
|
|
send_wildcard: false,
|
|
|
|
|
supports_credentials: false,
|
|
|
|
|
vary_header: true,
|
|
|
|
|
}),
|
|
|
|
|
methods: false,
|
|
|
|
|
error: None,
|
|
|
|
|
expose_hdrs: HashSet::new(),
|
|
|
|
|
resources: Vec::new(),
|
|
|
|
|
app: Some(app),
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 11:14:18 -08:00
|
|
|
|
/// This method register cors middleware with resource and
|
|
|
|
|
/// adds route for *OPTIONS* preflight requests.
|
|
|
|
|
///
|
2018-04-01 17:37:22 -07:00
|
|
|
|
/// It is possible to register *Cors* middleware with `ResourceHandler::middleware()`
|
2018-01-11 11:14:18 -08:00
|
|
|
|
/// method, but in that case *Cors* middleware wont be able to handle *OPTIONS*
|
|
|
|
|
/// requests.
|
2018-04-01 17:37:22 -07:00
|
|
|
|
pub fn register<S: 'static>(self, resource: &mut ResourceHandler<S>) {
|
2018-03-30 23:07:33 -07:00
|
|
|
|
resource.method(Method::OPTIONS).h(|_| HttpResponse::Ok());
|
2018-01-11 11:14:18 -08:00
|
|
|
|
resource.middleware(self);
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 14:20:00 -08:00
|
|
|
|
fn validate_origin<S>(&self, req: &mut HttpRequest<S>) -> Result<(), CorsError> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if let Some(hdr) = req.headers().get(header::ORIGIN) {
|
|
|
|
|
if let Ok(origin) = hdr.to_str() {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
return match self.inner.origins {
|
2018-01-11 20:11:34 -08:00
|
|
|
|
AllOrSome::All => Ok(()),
|
|
|
|
|
AllOrSome::Some(ref allowed_origins) => {
|
|
|
|
|
allowed_origins
|
|
|
|
|
.get(origin)
|
|
|
|
|
.and_then(|_| Some(()))
|
|
|
|
|
.ok_or_else(|| CorsError::OriginNotAllowed)
|
|
|
|
|
}
|
|
|
|
|
};
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
2018-01-10 14:20:00 -08:00
|
|
|
|
Err(CorsError::BadOrigin)
|
2018-01-09 22:33:51 -08:00
|
|
|
|
} else {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
return match self.inner.origins {
|
2018-01-10 14:56:45 -08:00
|
|
|
|
AllOrSome::All => Ok(()),
|
|
|
|
|
_ => Err(CorsError::MissingOrigin)
|
|
|
|
|
}
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 14:20:00 -08:00
|
|
|
|
fn validate_allowed_method<S>(&self, req: &mut HttpRequest<S>) -> Result<(), CorsError> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if let Some(hdr) = req.headers().get(header::ACCESS_CONTROL_REQUEST_METHOD) {
|
|
|
|
|
if let Ok(meth) = hdr.to_str() {
|
|
|
|
|
if let Ok(method) = Method::try_from(meth) {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
return self.inner.methods.get(&method)
|
2018-01-09 22:33:51 -08:00
|
|
|
|
.and_then(|_| Some(()))
|
2018-01-10 14:20:00 -08:00
|
|
|
|
.ok_or_else(|| CorsError::MethodNotAllowed);
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-10 14:20:00 -08:00
|
|
|
|
Err(CorsError::BadRequestMethod)
|
2018-01-09 22:33:51 -08:00
|
|
|
|
} else {
|
2018-01-10 14:20:00 -08:00
|
|
|
|
Err(CorsError::MissingRequestMethod)
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-09 23:55:42 -08:00
|
|
|
|
|
2018-01-10 14:20:00 -08:00
|
|
|
|
fn validate_allowed_headers<S>(&self, req: &mut HttpRequest<S>) -> Result<(), CorsError> {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
match self.inner.headers {
|
2018-01-11 20:11:34 -08:00
|
|
|
|
AllOrSome::All => Ok(()),
|
|
|
|
|
AllOrSome::Some(ref allowed_headers) => {
|
|
|
|
|
if let Some(hdr) = req.headers().get(header::ACCESS_CONTROL_REQUEST_HEADERS) {
|
|
|
|
|
if let Ok(headers) = hdr.to_str() {
|
2018-01-09 23:55:42 -08:00
|
|
|
|
let mut hdrs = HashSet::new();
|
|
|
|
|
for hdr in headers.split(',') {
|
|
|
|
|
match HeaderName::try_from(hdr.trim()) {
|
|
|
|
|
Ok(hdr) => hdrs.insert(hdr),
|
2018-01-10 14:20:00 -08:00
|
|
|
|
Err(_) => return Err(CorsError::BadRequestHeaders)
|
2018-01-09 23:55:42 -08:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !hdrs.is_empty() && !hdrs.is_subset(allowed_headers) {
|
2018-01-10 14:20:00 -08:00
|
|
|
|
return Err(CorsError::HeadersNotAllowed)
|
2018-01-09 23:55:42 -08:00
|
|
|
|
}
|
|
|
|
|
return Ok(())
|
|
|
|
|
}
|
2018-01-11 20:11:34 -08:00
|
|
|
|
Err(CorsError::BadRequestHeaders)
|
|
|
|
|
} else {
|
|
|
|
|
Err(CorsError::MissingRequestHeaders)
|
2018-01-09 23:55:42 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<S> Middleware<S> for Cors {
|
|
|
|
|
|
2018-01-09 22:48:35 -08:00
|
|
|
|
fn start(&self, req: &mut HttpRequest<S>) -> Result<Started> {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
if self.inner.preflight && Method::OPTIONS == *req.method() {
|
2018-01-09 22:48:35 -08:00
|
|
|
|
self.validate_origin(req)?;
|
|
|
|
|
self.validate_allowed_method(req)?;
|
2018-01-09 23:55:42 -08:00
|
|
|
|
self.validate_allowed_headers(req)?;
|
|
|
|
|
|
2018-01-20 16:36:57 -08:00
|
|
|
|
// allowed headers
|
2018-04-09 14:20:12 -07:00
|
|
|
|
let headers = if let Some(headers) = self.inner.headers.as_ref() {
|
2018-01-20 16:36:57 -08:00
|
|
|
|
Some(HeaderValue::try_from(&headers.iter().fold(
|
|
|
|
|
String::new(), |s, v| s + "," + v.as_str()).as_str()[1..]).unwrap())
|
|
|
|
|
} else if let Some(hdr) = req.headers().get(header::ACCESS_CONTROL_REQUEST_HEADERS) {
|
|
|
|
|
Some(hdr.clone())
|
|
|
|
|
} else {
|
|
|
|
|
None
|
|
|
|
|
};
|
|
|
|
|
|
2018-01-09 23:55:42 -08:00
|
|
|
|
Ok(Started::Response(
|
2018-03-30 23:07:33 -07:00
|
|
|
|
HttpResponse::Ok()
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.if_some(self.inner.max_age.as_ref(), |max_age, resp| {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
let _ = resp.header(
|
2018-01-09 23:55:42 -08:00
|
|
|
|
header::ACCESS_CONTROL_MAX_AGE, format!("{}", max_age).as_str());})
|
2018-01-20 16:36:57 -08:00
|
|
|
|
.if_some(headers, |headers, resp| {
|
|
|
|
|
let _ = resp.header(header::ACCESS_CONTROL_ALLOW_HEADERS, headers); })
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.if_true(self.inner.origins.is_all(), |resp| {
|
|
|
|
|
if self.inner.send_wildcard {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.header(header::ACCESS_CONTROL_ALLOW_ORIGIN, "*");
|
|
|
|
|
} else {
|
|
|
|
|
let origin = req.headers().get(header::ORIGIN).unwrap();
|
|
|
|
|
resp.header(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_ORIGIN, origin.clone());
|
|
|
|
|
}
|
|
|
|
|
})
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.if_true(self.inner.origins.is_some(), |resp| {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.header(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_ORIGIN,
|
2018-04-09 14:20:12 -07:00
|
|
|
|
self.inner.origins_str.as_ref().unwrap().clone());
|
2018-01-10 13:41:33 -08:00
|
|
|
|
})
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.if_true(self.inner.supports_credentials, |resp| {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.header(header::ACCESS_CONTROL_ALLOW_CREDENTIALS, "true");
|
|
|
|
|
})
|
2018-01-09 23:55:42 -08:00
|
|
|
|
.header(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_METHODS,
|
2018-04-09 14:20:12 -07:00
|
|
|
|
&self.inner.methods.iter().fold(
|
2018-01-10 13:41:33 -08:00
|
|
|
|
String::new(), |s, v| s + "," + v.as_str()).as_str()[1..])
|
2018-03-30 23:07:33 -07:00
|
|
|
|
.finish()))
|
2018-01-09 23:55:42 -08:00
|
|
|
|
} else {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
self.validate_origin(req)?;
|
|
|
|
|
|
2018-01-09 23:55:42 -08:00
|
|
|
|
Ok(Started::Done)
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 13:41:33 -08:00
|
|
|
|
fn response(&self, req: &mut HttpRequest<S>, mut resp: HttpResponse) -> Result<Response> {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
match self.inner.origins {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
AllOrSome::All => {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
if self.inner.send_wildcard {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.headers_mut().insert(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_ORIGIN, HeaderValue::from_static("*"));
|
2018-03-10 08:31:20 -08:00
|
|
|
|
} else if let Some(origin) = req.headers().get(header::ORIGIN) {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.headers_mut().insert(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_ORIGIN, origin.clone());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
AllOrSome::Some(_) => {
|
|
|
|
|
resp.headers_mut().insert(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_ORIGIN,
|
2018-04-09 14:20:12 -07:00
|
|
|
|
self.inner.origins_str.as_ref().unwrap().clone());
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-09 14:20:12 -07:00
|
|
|
|
if let Some(ref expose) = self.inner.expose_hdrs {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.headers_mut().insert(
|
|
|
|
|
header::ACCESS_CONTROL_EXPOSE_HEADERS,
|
|
|
|
|
HeaderValue::try_from(expose.as_str()).unwrap());
|
|
|
|
|
}
|
2018-04-09 14:20:12 -07:00
|
|
|
|
if self.inner.supports_credentials {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
resp.headers_mut().insert(
|
|
|
|
|
header::ACCESS_CONTROL_ALLOW_CREDENTIALS, HeaderValue::from_static("true"));
|
|
|
|
|
}
|
2018-04-09 14:20:12 -07:00
|
|
|
|
if self.inner.vary_header {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
let value = if let Some(hdr) = resp.headers_mut().get(header::VARY) {
|
|
|
|
|
let mut val: Vec<u8> = Vec::with_capacity(hdr.as_bytes().len() + 8);
|
|
|
|
|
val.extend(hdr.as_bytes());
|
|
|
|
|
val.extend(b", Origin");
|
|
|
|
|
HeaderValue::try_from(&val[..]).unwrap()
|
|
|
|
|
} else {
|
|
|
|
|
HeaderValue::from_static("Origin")
|
|
|
|
|
};
|
|
|
|
|
resp.headers_mut().insert(header::VARY, value);
|
|
|
|
|
}
|
2018-01-09 22:48:35 -08:00
|
|
|
|
Ok(Response::Done(resp))
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Structure that follows the builder pattern for building `Cors` middleware structs.
|
|
|
|
|
///
|
|
|
|
|
/// To construct a cors:
|
|
|
|
|
///
|
|
|
|
|
/// 1. Call [`Cors::build`](struct.Cors.html#method.build) to start building.
|
|
|
|
|
/// 2. Use any of the builder methods to set fields in the backend.
|
2018-01-09 23:55:42 -08:00
|
|
|
|
/// 3. Call [finish](struct.Cors.html#method.finish) to retrieve the constructed backend.
|
2018-01-09 22:33:51 -08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// # extern crate http;
|
|
|
|
|
/// # extern crate actix_web;
|
|
|
|
|
/// use http::header;
|
|
|
|
|
/// use actix_web::middleware::cors;
|
|
|
|
|
///
|
|
|
|
|
/// # fn main() {
|
|
|
|
|
/// let cors = cors::Cors::build()
|
|
|
|
|
/// .allowed_origin("https://www.rust-lang.org/")
|
|
|
|
|
/// .allowed_methods(vec!["GET", "POST"])
|
|
|
|
|
/// .allowed_headers(vec![header::AUTHORIZATION, header::ACCEPT])
|
|
|
|
|
/// .allowed_header(header::CONTENT_TYPE)
|
|
|
|
|
/// .max_age(3600)
|
2018-04-09 14:20:12 -07:00
|
|
|
|
/// .finish();
|
2018-01-09 22:33:51 -08:00
|
|
|
|
/// # }
|
|
|
|
|
/// ```
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub struct CorsBuilder<S=()> {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
cors: Option<Inner>,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
methods: bool,
|
|
|
|
|
error: Option<http::Error>,
|
2018-01-10 13:41:33 -08:00
|
|
|
|
expose_hdrs: HashSet<HeaderName>,
|
2018-04-09 21:11:15 -07:00
|
|
|
|
resources: Vec<(String, ResourceHandler<S>)>,
|
|
|
|
|
app: Option<App<S>>,
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
|
2018-04-09 14:20:12 -07:00
|
|
|
|
fn cors<'a>(parts: &'a mut Option<Inner>, err: &Option<http::Error>)
|
|
|
|
|
-> Option<&'a mut Inner>
|
|
|
|
|
{
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if err.is_some() {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
parts.as_mut()
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-09 21:11:15 -07:00
|
|
|
|
impl<S: 'static> CorsBuilder<S> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
|
|
|
|
/// Add an origin that are allowed to make requests.
|
|
|
|
|
/// Will be verified against the `Origin` request header.
|
|
|
|
|
///
|
|
|
|
|
/// When `All` is set, and `send_wildcard` is set, "*" will be sent in
|
|
|
|
|
/// the `Access-Control-Allow-Origin` response header. Otherwise, the client's `Origin` request
|
|
|
|
|
/// header will be echoed back in the `Access-Control-Allow-Origin` response header.
|
|
|
|
|
///
|
|
|
|
|
/// When `Some` is set, the client's `Origin` request header will be checked in a
|
|
|
|
|
/// case-sensitive manner.
|
|
|
|
|
///
|
|
|
|
|
/// This is the `list of origins` in the
|
|
|
|
|
/// [Resource Processing Model](https://www.w3.org/TR/cors/#resource-processing-model).
|
|
|
|
|
///
|
|
|
|
|
/// Defaults to `All`.
|
2018-04-09 14:20:12 -07:00
|
|
|
|
///
|
|
|
|
|
/// Builder panics if supplied origin is not valid uri.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn allowed_origin(&mut self, origin: &str) -> &mut CorsBuilder<S> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
match Uri::try_from(origin) {
|
2018-01-11 20:11:34 -08:00
|
|
|
|
Ok(_) => {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if cors.origins.is_all() {
|
|
|
|
|
cors.origins = AllOrSome::Some(HashSet::new());
|
|
|
|
|
}
|
|
|
|
|
if let AllOrSome::Some(ref mut origins) = cors.origins {
|
2018-01-11 20:11:34 -08:00
|
|
|
|
origins.insert(origin.to_owned());
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
self.error = Some(e.into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Set a list of methods which the allowed origins are allowed to access for
|
|
|
|
|
/// requests.
|
|
|
|
|
///
|
|
|
|
|
/// This is the `list of methods` in the
|
|
|
|
|
/// [Resource Processing Model](https://www.w3.org/TR/cors/#resource-processing-model).
|
|
|
|
|
///
|
|
|
|
|
/// Defaults to `[GET, HEAD, POST, OPTIONS, PUT, PATCH, DELETE]`
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn allowed_methods<U, M>(&mut self, methods: U) -> &mut CorsBuilder<S>
|
2018-01-09 22:33:51 -08:00
|
|
|
|
where U: IntoIterator<Item=M>, Method: HttpTryFrom<M>
|
|
|
|
|
{
|
|
|
|
|
self.methods = true;
|
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
2018-01-10 10:12:34 -08:00
|
|
|
|
for m in methods {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
match Method::try_from(m) {
|
|
|
|
|
Ok(method) => {
|
|
|
|
|
cors.methods.insert(method);
|
|
|
|
|
},
|
|
|
|
|
Err(e) => {
|
|
|
|
|
self.error = Some(e.into());
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Set an allowed header
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn allowed_header<H>(&mut self, header: H) -> &mut CorsBuilder<S>
|
2018-01-09 22:33:51 -08:00
|
|
|
|
where HeaderName: HttpTryFrom<H>
|
|
|
|
|
{
|
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
match HeaderName::try_from(header) {
|
|
|
|
|
Ok(method) => {
|
|
|
|
|
if cors.headers.is_all() {
|
|
|
|
|
cors.headers = AllOrSome::Some(HashSet::new());
|
|
|
|
|
}
|
|
|
|
|
if let AllOrSome::Some(ref mut headers) = cors.headers {
|
|
|
|
|
headers.insert(method);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(e) => self.error = Some(e.into()),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Set a list of header field names which can be used when
|
|
|
|
|
/// this resource is accessed by allowed origins.
|
|
|
|
|
///
|
|
|
|
|
/// If `All` is set, whatever is requested by the client in `Access-Control-Request-Headers`
|
|
|
|
|
/// will be echoed back in the `Access-Control-Allow-Headers` header.
|
|
|
|
|
///
|
|
|
|
|
/// This is the `list of headers` in the
|
|
|
|
|
/// [Resource Processing Model](https://www.w3.org/TR/cors/#resource-processing-model).
|
|
|
|
|
///
|
|
|
|
|
/// Defaults to `All`.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn allowed_headers<U, H>(&mut self, headers: U) -> &mut CorsBuilder<S>
|
2018-01-09 22:33:51 -08:00
|
|
|
|
where U: IntoIterator<Item=H>, HeaderName: HttpTryFrom<H>
|
|
|
|
|
{
|
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
2018-01-10 10:12:34 -08:00
|
|
|
|
for h in headers {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
match HeaderName::try_from(h) {
|
|
|
|
|
Ok(method) => {
|
|
|
|
|
if cors.headers.is_all() {
|
|
|
|
|
cors.headers = AllOrSome::Some(HashSet::new());
|
|
|
|
|
}
|
|
|
|
|
if let AllOrSome::Some(ref mut headers) = cors.headers {
|
|
|
|
|
headers.insert(method);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
self.error = Some(e.into());
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 13:41:33 -08:00
|
|
|
|
/// Set a list of headers which are safe to expose to the API of a CORS API specification.
|
2018-01-13 11:17:48 -08:00
|
|
|
|
/// This corresponds to the `Access-Control-Expose-Headers` response header.
|
2018-01-10 13:41:33 -08:00
|
|
|
|
///
|
|
|
|
|
/// This is the `list of exposed headers` in the
|
|
|
|
|
/// [Resource Processing Model](https://www.w3.org/TR/cors/#resource-processing-model).
|
|
|
|
|
///
|
|
|
|
|
/// This defaults to an empty set.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn expose_headers<U, H>(&mut self, headers: U) -> &mut CorsBuilder<S>
|
2018-01-10 13:41:33 -08:00
|
|
|
|
where U: IntoIterator<Item=H>, HeaderName: HttpTryFrom<H>
|
|
|
|
|
{
|
|
|
|
|
for h in headers {
|
|
|
|
|
match HeaderName::try_from(h) {
|
|
|
|
|
Ok(method) => {
|
|
|
|
|
self.expose_hdrs.insert(method);
|
|
|
|
|
},
|
|
|
|
|
Err(e) => {
|
|
|
|
|
self.error = Some(e.into());
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-09 22:33:51 -08:00
|
|
|
|
/// Set a maximum time for which this CORS request maybe cached.
|
|
|
|
|
/// This value is set as the `Access-Control-Max-Age` header.
|
|
|
|
|
///
|
|
|
|
|
/// This defaults to `None` (unset).
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn max_age(&mut self, max_age: usize) -> &mut CorsBuilder<S> {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
cors.max_age = Some(max_age)
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-09 23:55:42 -08:00
|
|
|
|
/// Set a wildcard origins
|
|
|
|
|
///
|
2018-01-31 20:28:53 +03:00
|
|
|
|
/// If send wildcard is set and the `allowed_origins` parameter is `All`, a wildcard
|
2018-01-09 23:55:42 -08:00
|
|
|
|
/// `Access-Control-Allow-Origin` response header is sent, rather than the request’s
|
|
|
|
|
/// `Origin` header.
|
|
|
|
|
///
|
|
|
|
|
/// This is the `supports credentials flag` in the
|
|
|
|
|
/// [Resource Processing Model](https://www.w3.org/TR/cors/#resource-processing-model).
|
|
|
|
|
///
|
|
|
|
|
/// This **CANNOT** be used in conjunction with `allowed_origins` set to `All` and
|
|
|
|
|
/// `allow_credentials` set to `true`. Depending on the mode of usage, this will either result
|
|
|
|
|
/// in an `Error::CredentialsWithWildcardOrigin` error during actix launch or runtime.
|
|
|
|
|
///
|
|
|
|
|
/// Defaults to `false`.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn send_wildcard(&mut self) -> &mut CorsBuilder<S> {
|
2018-01-09 23:55:42 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
cors.send_wildcard = true
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Allows users to make authenticated requests
|
|
|
|
|
///
|
|
|
|
|
/// If true, injects the `Access-Control-Allow-Credentials` header in responses.
|
|
|
|
|
/// This allows cookies and credentials to be submitted across domains.
|
|
|
|
|
///
|
2018-01-31 20:28:53 +03:00
|
|
|
|
/// This option cannot be used in conjunction with an `allowed_origin` set to `All`
|
2018-01-10 13:41:33 -08:00
|
|
|
|
/// and `send_wildcards` set to `true`.
|
|
|
|
|
///
|
|
|
|
|
/// Defaults to `false`.
|
2018-04-09 14:20:12 -07:00
|
|
|
|
///
|
|
|
|
|
/// Builder panics if credentials are allowed, but the Origin is set to "*".
|
|
|
|
|
/// This is not allowed by W3C
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn supports_credentials(&mut self) -> &mut CorsBuilder<S> {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
cors.supports_credentials = true
|
2018-01-09 23:55:42 -08:00
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
2018-01-10 13:41:33 -08:00
|
|
|
|
|
|
|
|
|
/// Disable `Vary` header support.
|
|
|
|
|
///
|
|
|
|
|
/// When enabled the header `Vary: Origin` will be returned as per the W3
|
|
|
|
|
/// implementation guidelines.
|
|
|
|
|
///
|
|
|
|
|
/// Setting this header when the `Access-Control-Allow-Origin` is
|
|
|
|
|
/// dynamically generated (e.g. when there is more than one allowed
|
|
|
|
|
/// origin, and an Origin than '*' is returned) informs CDNs and other
|
|
|
|
|
/// caches that the CORS headers are dynamic, and cannot be cached.
|
|
|
|
|
///
|
|
|
|
|
/// By default `vary` header support is enabled.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn disable_vary_header(&mut self) -> &mut CorsBuilder<S> {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
cors.vary_header = false
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Disable *preflight* request support.
|
|
|
|
|
///
|
|
|
|
|
/// When enabled cors middleware automatically handles *OPTIONS* request.
|
|
|
|
|
/// This is useful application level middleware.
|
|
|
|
|
///
|
|
|
|
|
/// By default *preflight* support is enabled.
|
2018-04-09 21:11:15 -07:00
|
|
|
|
pub fn disable_preflight(&mut self) -> &mut CorsBuilder<S> {
|
2018-01-10 13:41:33 -08:00
|
|
|
|
if let Some(cors) = cors(&mut self.cors, &self.error) {
|
|
|
|
|
cors.preflight = false
|
|
|
|
|
}
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-09 21:11:15 -07:00
|
|
|
|
/// Configure resource for a specific path.
|
2018-04-09 14:20:12 -07:00
|
|
|
|
///
|
2018-04-09 21:11:15 -07:00
|
|
|
|
/// This is similar to a `App::resource()` method. Except, cors middleware
|
|
|
|
|
/// get registered for the resource.
|
|
|
|
|
///
|
|
|
|
|
/// ```rust
|
|
|
|
|
/// # extern crate actix_web;
|
|
|
|
|
/// use actix_web::{http, App, HttpResponse};
|
|
|
|
|
/// use actix_web::middleware::cors::Cors;
|
|
|
|
|
///
|
|
|
|
|
/// fn main() {
|
|
|
|
|
/// let app = App::new()
|
|
|
|
|
/// .configure(|app| Cors::for_app(app) // <- Construct CORS builder
|
|
|
|
|
/// .allowed_origin("https://www.rust-lang.org/")
|
|
|
|
|
/// .allowed_methods(vec!["GET", "POST"])
|
|
|
|
|
/// .allowed_header(http::header::CONTENT_TYPE)
|
|
|
|
|
/// .max_age(3600)
|
|
|
|
|
/// .resource("/resource1", |r| { // register resource
|
|
|
|
|
/// r.method(http::Method::GET).f(|_| HttpResponse::Ok());
|
|
|
|
|
/// })
|
|
|
|
|
/// .resource("/resource2", |r| { // register another resource
|
|
|
|
|
/// r.method(http::Method::HEAD)
|
|
|
|
|
/// .f(|_| HttpResponse::MethodNotAllowed());
|
|
|
|
|
/// })
|
|
|
|
|
/// .register() // construct CORS and return application instance
|
|
|
|
|
/// );
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
pub fn resource<F, R>(&mut self, path: &str, f: F) -> &mut CorsBuilder<S>
|
|
|
|
|
where F: FnOnce(&mut ResourceHandler<S>) -> R + 'static
|
|
|
|
|
{
|
|
|
|
|
// add resource handler
|
|
|
|
|
let mut handler = ResourceHandler::default();
|
|
|
|
|
f(&mut handler);
|
|
|
|
|
|
|
|
|
|
self.resources.push((path.to_owned(), handler));
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn construct(&mut self) -> Cors {
|
2018-01-09 22:33:51 -08:00
|
|
|
|
if !self.methods {
|
|
|
|
|
self.allowed_methods(vec![Method::GET, Method::HEAD,
|
2018-04-09 21:11:15 -07:00
|
|
|
|
Method::POST, Method::OPTIONS, Method::PUT,
|
|
|
|
|
Method::PATCH, Method::DELETE]);
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if let Some(e) = self.error.take() {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
panic!("{}", e);
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let mut cors = self.cors.take().expect("cannot reuse CorsBuilder");
|
|
|
|
|
|
|
|
|
|
if cors.supports_credentials && cors.send_wildcard && cors.origins.is_all() {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
panic!("Credentials are allowed, but the Origin is set to \"*\"");
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if let AllOrSome::Some(ref origins) = cors.origins {
|
|
|
|
|
let s = origins.iter().fold(String::new(), |s, v| s + &format!("{}", v));
|
|
|
|
|
cors.origins_str = Some(HeaderValue::try_from(s.as_str()).unwrap());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !self.expose_hdrs.is_empty() {
|
|
|
|
|
cors.expose_hdrs = Some(
|
|
|
|
|
self.expose_hdrs.iter().fold(
|
|
|
|
|
String::new(), |s, v| s + v.as_str())[1..].to_owned());
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
2018-04-09 14:20:12 -07:00
|
|
|
|
Cors{inner: Rc::new(cors)}
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
2018-04-09 21:11:15 -07:00
|
|
|
|
|
|
|
|
|
/// Finishes building and returns the built `Cors` instance.
|
|
|
|
|
///
|
|
|
|
|
/// This method panics in case of any configuration error.
|
|
|
|
|
pub fn finish(&mut self) -> Cors {
|
|
|
|
|
if !self.resources.is_empty() {
|
|
|
|
|
panic!("CorsBuilder::resource() was used,
|
|
|
|
|
to construct CORS `.register(app)` method should be used");
|
|
|
|
|
}
|
|
|
|
|
self.construct()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Finishes building Cors middleware and register middleware for application
|
|
|
|
|
///
|
|
|
|
|
/// This method panics in case of any configuration error or if non of
|
|
|
|
|
/// resources are registered.
|
|
|
|
|
pub fn register(&mut self) -> App<S> {
|
|
|
|
|
if self.resources.is_empty() {
|
|
|
|
|
panic!("No resources are registered.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let cors = self.construct();
|
|
|
|
|
let mut app = self.app.take().expect(
|
|
|
|
|
"CorsBuilder has to be constructed with Cors::for_app(app)");
|
|
|
|
|
|
|
|
|
|
// register resources
|
|
|
|
|
for (path, mut resource) in self.resources.drain(..) {
|
|
|
|
|
cors.clone().register(&mut resource);
|
|
|
|
|
app.register_resource(&path, resource);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app
|
|
|
|
|
}
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
|
|
|
|
use super::*;
|
2018-04-09 21:29:57 -07:00
|
|
|
|
use test::{self, TestRequest};
|
2018-01-10 13:41:33 -08:00
|
|
|
|
|
|
|
|
|
impl Started {
|
|
|
|
|
fn is_done(&self) -> bool {
|
|
|
|
|
match *self {
|
|
|
|
|
Started::Done => true,
|
|
|
|
|
_ => false,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
fn response(self) -> HttpResponse {
|
|
|
|
|
match self {
|
|
|
|
|
Started::Response(resp) => resp,
|
|
|
|
|
_ => panic!(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-10 14:56:45 -08:00
|
|
|
|
impl Response {
|
|
|
|
|
fn response(self) -> HttpResponse {
|
|
|
|
|
match self {
|
|
|
|
|
Response::Done(resp) => resp,
|
|
|
|
|
_ => panic!(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-10 13:41:33 -08:00
|
|
|
|
|
|
|
|
|
#[test]
|
2018-04-09 14:20:12 -07:00
|
|
|
|
#[should_panic(expected = "Credentials are allowed, but the Origin is set to")]
|
2018-01-10 13:41:33 -08:00
|
|
|
|
fn cors_validates_illegal_allow_credentials() {
|
|
|
|
|
Cors::build()
|
|
|
|
|
.supports_credentials()
|
|
|
|
|
.send_wildcard()
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.finish();
|
2018-01-10 13:41:33 -08:00
|
|
|
|
}
|
|
|
|
|
|
2018-04-09 21:11:15 -07:00
|
|
|
|
#[test]
|
|
|
|
|
#[should_panic(expected = "No resources are registered")]
|
|
|
|
|
fn no_resource() {
|
|
|
|
|
Cors::build()
|
|
|
|
|
.supports_credentials()
|
|
|
|
|
.send_wildcard()
|
|
|
|
|
.register();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[should_panic(expected = "Cors::for_app(app)")]
|
|
|
|
|
fn no_resource2() {
|
|
|
|
|
Cors::build()
|
|
|
|
|
.resource("/test", |r| r.f(|_| HttpResponse::Ok()))
|
|
|
|
|
.register();
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 13:41:33 -08:00
|
|
|
|
#[test]
|
|
|
|
|
fn validate_origin_allows_all_origins() {
|
|
|
|
|
let cors = Cors::default();
|
|
|
|
|
let mut req = TestRequest::with_header(
|
|
|
|
|
"Origin", "https://www.example.com").finish();
|
|
|
|
|
|
|
|
|
|
assert!(cors.start(&mut req).ok().unwrap().is_done())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_preflight() {
|
|
|
|
|
let mut cors = Cors::build()
|
|
|
|
|
.send_wildcard()
|
|
|
|
|
.max_age(3600)
|
|
|
|
|
.allowed_methods(vec![Method::GET, Method::OPTIONS, Method::POST])
|
|
|
|
|
.allowed_headers(vec![header::AUTHORIZATION, header::ACCEPT])
|
|
|
|
|
.allowed_header(header::CONTENT_TYPE)
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.finish();
|
2018-01-10 13:41:33 -08:00
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header(
|
|
|
|
|
"Origin", "https://www.example.com")
|
|
|
|
|
.method(Method::OPTIONS)
|
|
|
|
|
.finish();
|
|
|
|
|
|
|
|
|
|
assert!(cors.start(&mut req).is_err());
|
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header("Origin", "https://www.example.com")
|
|
|
|
|
.header(header::ACCESS_CONTROL_REQUEST_METHOD, "put")
|
|
|
|
|
.method(Method::OPTIONS)
|
|
|
|
|
.finish();
|
|
|
|
|
|
|
|
|
|
assert!(cors.start(&mut req).is_err());
|
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header("Origin", "https://www.example.com")
|
|
|
|
|
.header(header::ACCESS_CONTROL_REQUEST_METHOD, "POST")
|
|
|
|
|
.header(header::ACCESS_CONTROL_REQUEST_HEADERS, "AUTHORIZATION,ACCEPT")
|
|
|
|
|
.method(Method::OPTIONS)
|
|
|
|
|
.finish();
|
|
|
|
|
|
|
|
|
|
let resp = cors.start(&mut req).unwrap().response();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"*"[..],
|
|
|
|
|
resp.headers().get(header::ACCESS_CONTROL_ALLOW_ORIGIN).unwrap().as_bytes());
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"3600"[..],
|
|
|
|
|
resp.headers().get(header::ACCESS_CONTROL_MAX_AGE).unwrap().as_bytes());
|
|
|
|
|
//assert_eq!(
|
|
|
|
|
// &b"authorization,accept,content-type"[..],
|
|
|
|
|
// resp.headers().get(header::ACCESS_CONTROL_ALLOW_HEADERS).unwrap().as_bytes());
|
|
|
|
|
//assert_eq!(
|
|
|
|
|
// &b"POST,GET,OPTIONS"[..],
|
|
|
|
|
// resp.headers().get(header::ACCESS_CONTROL_ALLOW_METHODS).unwrap().as_bytes());
|
2018-01-09 22:33:51 -08:00
|
|
|
|
|
2018-04-09 14:20:12 -07:00
|
|
|
|
Rc::get_mut(&mut cors.inner).unwrap().preflight = false;
|
2018-01-10 13:41:33 -08:00
|
|
|
|
assert!(cors.start(&mut req).unwrap().is_done());
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|
2018-01-10 14:20:00 -08:00
|
|
|
|
|
|
|
|
|
#[test]
|
2018-01-10 14:56:45 -08:00
|
|
|
|
#[should_panic(expected = "MissingOrigin")]
|
|
|
|
|
fn test_validate_missing_origin() {
|
|
|
|
|
let cors = Cors::build()
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.allowed_origin("https://www.example.com").finish();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
|
|
|
|
|
let mut req = HttpRequest::default();
|
|
|
|
|
cors.start(&mut req).unwrap();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[should_panic(expected = "OriginNotAllowed")]
|
|
|
|
|
fn test_validate_not_allowed_origin() {
|
2018-01-10 14:20:00 -08:00
|
|
|
|
let cors = Cors::build()
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.allowed_origin("https://www.example.com").finish();
|
2018-01-10 14:20:00 -08:00
|
|
|
|
|
2018-01-10 14:21:48 -08:00
|
|
|
|
let mut req = TestRequest::with_header("Origin", "https://www.unknown.com")
|
2018-01-10 14:20:00 -08:00
|
|
|
|
.method(Method::GET)
|
|
|
|
|
.finish();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
cors.start(&mut req).unwrap();
|
|
|
|
|
}
|
2018-01-10 14:20:00 -08:00
|
|
|
|
|
2018-01-10 14:56:45 -08:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_validate_origin() {
|
|
|
|
|
let cors = Cors::build()
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.allowed_origin("https://www.example.com").finish();
|
2018-01-10 14:21:48 -08:00
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header("Origin", "https://www.example.com")
|
|
|
|
|
.method(Method::GET)
|
|
|
|
|
.finish();
|
|
|
|
|
|
|
|
|
|
assert!(cors.start(&mut req).unwrap().is_done());
|
2018-01-10 14:20:00 -08:00
|
|
|
|
}
|
2018-01-10 14:56:45 -08:00
|
|
|
|
|
2018-03-10 08:31:20 -08:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_no_origin_response() {
|
2018-04-09 14:20:12 -07:00
|
|
|
|
let cors = Cors::build().finish();
|
2018-03-10 08:31:20 -08:00
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::default().method(Method::GET).finish();
|
2018-03-30 23:07:33 -07:00
|
|
|
|
let resp: HttpResponse = HttpResponse::Ok().into();
|
2018-03-10 08:31:20 -08:00
|
|
|
|
let resp = cors.response(&mut req, resp).unwrap().response();
|
|
|
|
|
assert!(resp.headers().get(header::ACCESS_CONTROL_ALLOW_ORIGIN).is_none());
|
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header(
|
|
|
|
|
"Origin", "https://www.example.com")
|
|
|
|
|
.method(Method::OPTIONS)
|
|
|
|
|
.finish();
|
|
|
|
|
let resp = cors.response(&mut req, resp).unwrap().response();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"https://www.example.com"[..],
|
|
|
|
|
resp.headers().get(header::ACCESS_CONTROL_ALLOW_ORIGIN).unwrap().as_bytes());
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-10 14:56:45 -08:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_response() {
|
|
|
|
|
let cors = Cors::build()
|
|
|
|
|
.send_wildcard()
|
2018-01-10 15:28:33 -08:00
|
|
|
|
.disable_preflight()
|
2018-01-10 14:56:45 -08:00
|
|
|
|
.max_age(3600)
|
|
|
|
|
.allowed_methods(vec![Method::GET, Method::OPTIONS, Method::POST])
|
|
|
|
|
.allowed_headers(vec![header::AUTHORIZATION, header::ACCEPT])
|
|
|
|
|
.allowed_header(header::CONTENT_TYPE)
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.finish();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
|
|
|
|
|
let mut req = TestRequest::with_header(
|
|
|
|
|
"Origin", "https://www.example.com")
|
|
|
|
|
.method(Method::OPTIONS)
|
|
|
|
|
.finish();
|
|
|
|
|
|
2018-03-30 23:07:33 -07:00
|
|
|
|
let resp: HttpResponse = HttpResponse::Ok().into();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
let resp = cors.response(&mut req, resp).unwrap().response();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"*"[..],
|
|
|
|
|
resp.headers().get(header::ACCESS_CONTROL_ALLOW_ORIGIN).unwrap().as_bytes());
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"Origin"[..],
|
|
|
|
|
resp.headers().get(header::VARY).unwrap().as_bytes());
|
|
|
|
|
|
2018-03-30 23:07:33 -07:00
|
|
|
|
let resp: HttpResponse = HttpResponse::Ok()
|
2018-01-10 14:56:45 -08:00
|
|
|
|
.header(header::VARY, "Accept")
|
2018-03-30 23:07:33 -07:00
|
|
|
|
.finish();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
let resp = cors.response(&mut req, resp).unwrap().response();
|
|
|
|
|
assert_eq!(
|
|
|
|
|
&b"Accept, Origin"[..],
|
|
|
|
|
resp.headers().get(header::VARY).unwrap().as_bytes());
|
|
|
|
|
|
|
|
|
|
let cors = Cors::build()
|
2018-01-10 15:28:33 -08:00
|
|
|
|
.disable_vary_header()
|
2018-01-10 14:56:45 -08:00
|
|
|
|
.allowed_origin("https://www.example.com")
|
2018-04-09 14:20:12 -07:00
|
|
|
|
.finish();
|
2018-03-30 23:07:33 -07:00
|
|
|
|
let resp: HttpResponse = HttpResponse::Ok().into();
|
2018-01-10 14:56:45 -08:00
|
|
|
|
let resp = cors.response(&mut req, resp).unwrap().response();
|
|
|
|
|
assert_eq!(
|
2018-01-11 20:20:50 -08:00
|
|
|
|
&b"https://www.example.com"[..],
|
2018-01-10 14:56:45 -08:00
|
|
|
|
resp.headers().get(header::ACCESS_CONTROL_ALLOW_ORIGIN).unwrap().as_bytes());
|
|
|
|
|
}
|
2018-04-09 21:11:15 -07:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn cors_resource() {
|
2018-04-09 21:29:57 -07:00
|
|
|
|
let mut srv = test::TestServer::with_factory(
|
|
|
|
|
|| App::new()
|
|
|
|
|
.configure(
|
|
|
|
|
|app| Cors::for_app(app)
|
|
|
|
|
.allowed_origin("https://www.example.com")
|
|
|
|
|
.resource("/test", |r| r.f(|_| HttpResponse::Ok()))
|
|
|
|
|
.register()));
|
|
|
|
|
|
|
|
|
|
let request = srv.get().uri(srv.url("/test")).finish().unwrap();
|
|
|
|
|
let response = srv.execute(request.send()).unwrap();
|
|
|
|
|
assert_eq!(response.status(), StatusCode::BAD_REQUEST);
|
|
|
|
|
|
|
|
|
|
let request = srv.get().uri(srv.url("/test"))
|
|
|
|
|
.header("ORIGIN", "https://www.example.com").finish().unwrap();
|
|
|
|
|
let response = srv.execute(request.send()).unwrap();
|
|
|
|
|
assert_eq!(response.status(), StatusCode::OK);
|
2018-04-09 21:11:15 -07:00
|
|
|
|
}
|
2018-01-09 22:33:51 -08:00
|
|
|
|
}
|