mirror of
https://github.com/servo/servo.git
synced 2025-06-09 09:03:23 +00:00
322 lines
12 KiB
Rust
322 lines
12 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
//! An implementation of the [CORS preflight cache](https://fetch.spec.whatwg.org/#cors-preflight-cache)
|
|
//! For now this library is XHR-specific.
|
|
//! For stuff involving `<img>`, `<iframe>`, `<form>`, etc please check what
|
|
//! the request mode should be and compare with the fetch spec
|
|
//! This library will eventually become the core of the Fetch crate
|
|
//! with CORSRequest being expanded into FetchRequest (etc)
|
|
|
|
use hyper::method::Method;
|
|
use std::ascii::AsciiExt;
|
|
use std::sync::mpsc::{Sender, Receiver, channel};
|
|
use time;
|
|
use time::{now, Timespec};
|
|
use url::Url;
|
|
|
|
/// Union type for CORS cache entries
|
|
///
|
|
/// Each entry might pertain to a header or method
|
|
#[derive(Clone)]
|
|
pub enum HeaderOrMethod {
|
|
HeaderData(String),
|
|
MethodData(Method)
|
|
}
|
|
|
|
impl HeaderOrMethod {
|
|
fn match_header(&self, header_name: &str) -> bool {
|
|
match *self {
|
|
HeaderOrMethod::HeaderData(ref s) => s.eq_ignore_ascii_case(header_name),
|
|
_ => false
|
|
}
|
|
}
|
|
|
|
fn match_method(&self, method: &Method) -> bool {
|
|
match *self {
|
|
HeaderOrMethod::MethodData(ref m) => m == method,
|
|
_ => false
|
|
}
|
|
}
|
|
}
|
|
|
|
/// An entry in the CORS cache
|
|
#[derive(Clone)]
|
|
pub struct CORSCacheEntry {
|
|
pub origin: Url,
|
|
pub url: Url,
|
|
pub max_age: u32,
|
|
pub credentials: bool,
|
|
pub header_or_method: HeaderOrMethod,
|
|
created: Timespec
|
|
}
|
|
|
|
impl CORSCacheEntry {
|
|
fn new(origin: Url, url: Url, max_age: u32, credentials: bool,
|
|
header_or_method: HeaderOrMethod) -> CORSCacheEntry {
|
|
CORSCacheEntry {
|
|
origin: origin,
|
|
url: url,
|
|
max_age: max_age,
|
|
credentials: credentials,
|
|
header_or_method: header_or_method,
|
|
created: time::now().to_timespec()
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Properties of Request required to cache match.
|
|
pub struct CacheRequestDetails {
|
|
pub origin: Url,
|
|
pub destination: Url,
|
|
pub credentials: bool
|
|
}
|
|
|
|
/// Trait for a generic CORS Cache
|
|
pub trait CORSCache {
|
|
/// [Clear the cache](https://fetch.spec.whatwg.org/#concept-cache-clear)
|
|
fn clear (&mut self, request: CacheRequestDetails);
|
|
|
|
/// Remove old entries
|
|
fn cleanup(&mut self);
|
|
|
|
/// Returns true if an entry with a
|
|
/// [matching header](https://fetch.spec.whatwg.org/#concept-cache-match-header) is found
|
|
fn match_header(&mut self, request: CacheRequestDetails, header_name: &str) -> bool;
|
|
|
|
/// Updates max age if an entry for a
|
|
/// [matching header](https://fetch.spec.whatwg.org/#concept-cache-match-header) is found.
|
|
///
|
|
/// If not, it will insert an equivalent entry
|
|
fn match_header_and_update(&mut self, request: CacheRequestDetails, header_name: &str, new_max_age: u32) -> bool;
|
|
|
|
/// Returns true if an entry with a
|
|
/// [matching method](https://fetch.spec.whatwg.org/#concept-cache-match-method) is found
|
|
fn match_method(&mut self, request: CacheRequestDetails, method: Method) -> bool;
|
|
|
|
/// Updates max age if an entry for
|
|
/// [a matching method](https://fetch.spec.whatwg.org/#concept-cache-match-method) is found.
|
|
///
|
|
/// If not, it will insert an equivalent entry
|
|
fn match_method_and_update(&mut self, request: CacheRequestDetails, method: Method, new_max_age: u32) -> bool;
|
|
/// Insert an entry
|
|
fn insert(&mut self, entry: CORSCacheEntry);
|
|
}
|
|
|
|
/// A simple, vector-based CORS Cache
|
|
#[derive(Clone)]
|
|
pub struct BasicCORSCache(Vec<CORSCacheEntry>);
|
|
|
|
fn match_headers(cors_cache: &CORSCacheEntry, cors_req: &CacheRequestDetails) -> bool {
|
|
cors_cache.origin.scheme == cors_req.origin.scheme &&
|
|
cors_cache.origin.host() == cors_req.origin.host() &&
|
|
cors_cache.origin.port() == cors_req.origin.port() &&
|
|
cors_cache.url == cors_req.destination &&
|
|
cors_cache.credentials == cors_req.credentials
|
|
}
|
|
|
|
impl BasicCORSCache {
|
|
fn find_entry_by_header<'a>(&'a mut self, request: &CacheRequestDetails,
|
|
header_name: &str) -> Option<&'a mut CORSCacheEntry> {
|
|
self.cleanup();
|
|
let BasicCORSCache(ref mut buf) = *self;
|
|
buf.iter_mut().find(|e| match_headers(e, request) && e.header_or_method.match_header(header_name))
|
|
}
|
|
|
|
fn find_entry_by_method<'a>(&'a mut self, request: &CacheRequestDetails,
|
|
method: Method) -> Option<&'a mut CORSCacheEntry> {
|
|
// we can take the method from CORSRequest itself
|
|
self.cleanup();
|
|
let BasicCORSCache(ref mut buf) = *self;
|
|
buf.iter_mut().find(|e| match_headers(e, request) && e.header_or_method.match_method(&method))
|
|
}
|
|
}
|
|
|
|
impl CORSCache for BasicCORSCache {
|
|
/// https://fetch.spec.whatwg.org/#concept-cache-clear
|
|
#[allow(dead_code)]
|
|
fn clear (&mut self, request: CacheRequestDetails) {
|
|
let BasicCORSCache(buf) = self.clone();
|
|
let new_buf: Vec<CORSCacheEntry> =
|
|
buf.into_iter().filter(|e| e.origin == request.origin && request.destination == e.url).collect();
|
|
*self = BasicCORSCache(new_buf);
|
|
}
|
|
|
|
// Remove old entries
|
|
fn cleanup(&mut self) {
|
|
let BasicCORSCache(buf) = self.clone();
|
|
let now = time::now().to_timespec();
|
|
let new_buf: Vec<CORSCacheEntry> = buf.into_iter()
|
|
.filter(|e| now.sec > e.created.sec + e.max_age as i64)
|
|
.collect();
|
|
*self = BasicCORSCache(new_buf);
|
|
}
|
|
|
|
fn match_header(&mut self, request: CacheRequestDetails, header_name: &str) -> bool {
|
|
self.find_entry_by_header(&request, header_name).is_some()
|
|
}
|
|
|
|
fn match_header_and_update(&mut self, request: CacheRequestDetails, header_name: &str, new_max_age: u32) -> bool {
|
|
match self.find_entry_by_header(&request, header_name).map(|e| e.max_age = new_max_age) {
|
|
Some(_) => true,
|
|
None => {
|
|
self.insert(CORSCacheEntry::new(request.origin, request.destination, new_max_age,
|
|
request.credentials,
|
|
HeaderOrMethod::HeaderData(header_name.to_owned())));
|
|
false
|
|
}
|
|
}
|
|
}
|
|
|
|
fn match_method(&mut self, request: CacheRequestDetails, method: Method) -> bool {
|
|
self.find_entry_by_method(&request, method).is_some()
|
|
}
|
|
|
|
fn match_method_and_update(&mut self, request: CacheRequestDetails, method: Method, new_max_age: u32) -> bool {
|
|
match self.find_entry_by_method(&request, method.clone()).map(|e| e.max_age = new_max_age) {
|
|
Some(_) => true,
|
|
None => {
|
|
self.insert(CORSCacheEntry::new(request.origin, request.destination, new_max_age,
|
|
request.credentials, HeaderOrMethod::MethodData(method)));
|
|
false
|
|
}
|
|
}
|
|
}
|
|
|
|
fn insert(&mut self, entry: CORSCacheEntry) {
|
|
self.cleanup();
|
|
let BasicCORSCache(ref mut buf) = *self;
|
|
buf.push(entry);
|
|
}
|
|
}
|
|
|
|
/// Various messages that can be sent to a CORSCacheTask
|
|
pub enum CORSCacheTaskMsg {
|
|
Clear(CacheRequestDetails, Sender<()>),
|
|
Cleanup(Sender<()>),
|
|
MatchHeader(CacheRequestDetails, String, Sender<bool>),
|
|
MatchHeaderUpdate(CacheRequestDetails, String, u32, Sender<bool>),
|
|
MatchMethod(CacheRequestDetails, Method, Sender<bool>),
|
|
MatchMethodUpdate(CacheRequestDetails, Method, u32, Sender<bool>),
|
|
Insert(CORSCacheEntry, Sender<()>),
|
|
ExitMsg
|
|
}
|
|
|
|
/// A Sender to a CORSCacheTask
|
|
///
|
|
/// This can be used as a CORS Cache.
|
|
/// The methods on this type block until they can run, and it behaves similar to a mutex
|
|
pub type CORSCacheSender = Sender<CORSCacheTaskMsg>;
|
|
|
|
impl CORSCache for CORSCacheSender {
|
|
fn clear (&mut self, request: CacheRequestDetails) {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::Clear(request, tx));
|
|
let _ = rx.recv();
|
|
}
|
|
|
|
fn cleanup(&mut self) {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::Cleanup(tx));
|
|
let _ = rx.recv();
|
|
}
|
|
|
|
fn match_header(&mut self, request: CacheRequestDetails, header_name: &str) -> bool {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::MatchHeader(request, header_name.to_owned(), tx));
|
|
rx.recv().unwrap_or(false)
|
|
}
|
|
|
|
fn match_header_and_update(&mut self, request: CacheRequestDetails, header_name: &str, new_max_age: u32) -> bool {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::MatchHeaderUpdate(request, header_name.to_owned(), new_max_age, tx));
|
|
rx.recv().unwrap_or(false)
|
|
}
|
|
|
|
fn match_method(&mut self, request: CacheRequestDetails, method: Method) -> bool {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::MatchMethod(request, method, tx));
|
|
rx.recv().unwrap_or(false)
|
|
}
|
|
|
|
fn match_method_and_update(&mut self, request: CacheRequestDetails, method: Method, new_max_age: u32) -> bool {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::MatchMethodUpdate(request, method, new_max_age, tx));
|
|
rx.recv().unwrap_or(false)
|
|
}
|
|
|
|
fn insert(&mut self, entry: CORSCacheEntry) {
|
|
let (tx, rx) = channel();
|
|
self.send(CORSCacheTaskMsg::Insert(entry, tx));
|
|
let _ = rx.recv();
|
|
}
|
|
}
|
|
|
|
/// A simple task-based CORS Cache that can be sent messages
|
|
///
|
|
/// #Example
|
|
/// ```ignore
|
|
/// let task = CORSCacheTask::new();
|
|
/// let builder = TaskBuilder::new().named("XHRTask");
|
|
/// let mut sender = task.sender();
|
|
/// builder.spawn(move || { task.run() });
|
|
/// sender.insert(CORSCacheEntry::new(/* parameters here */));
|
|
/// ```
|
|
pub struct CORSCacheTask {
|
|
receiver: Receiver<CORSCacheTaskMsg>,
|
|
cache: BasicCORSCache,
|
|
sender: CORSCacheSender
|
|
}
|
|
|
|
impl CORSCacheTask {
|
|
pub fn new() -> CORSCacheTask {
|
|
let (tx, rx) = channel();
|
|
CORSCacheTask {
|
|
receiver: rx,
|
|
cache: BasicCORSCache(vec![]),
|
|
sender: tx
|
|
}
|
|
}
|
|
|
|
/// Provides a sender to the cache task
|
|
pub fn sender(&self) -> CORSCacheSender {
|
|
self.sender.clone()
|
|
}
|
|
|
|
/// Runs the cache task
|
|
/// This blocks the current task, so it is advised
|
|
/// to spawn a new task for this
|
|
/// Send ExitMsg to the associated Sender to exit
|
|
pub fn run(&mut self) {
|
|
loop {
|
|
match self.receiver.recv().unwrap() {
|
|
CORSCacheTaskMsg::Clear(request, tx) => {
|
|
self.cache.clear(request);
|
|
tx.send(());
|
|
},
|
|
CORSCacheTaskMsg::Cleanup(tx) => {
|
|
self.cache.cleanup();
|
|
tx.send(());
|
|
},
|
|
CORSCacheTaskMsg::MatchHeader(request, header, tx) => {
|
|
tx.send(self.cache.match_header(request, &header));
|
|
},
|
|
CORSCacheTaskMsg::MatchHeaderUpdate(request, header, new_max_age, tx) => {
|
|
tx.send(self.cache.match_header_and_update(request, &header, new_max_age));
|
|
},
|
|
CORSCacheTaskMsg::MatchMethod(request, method, tx) => {
|
|
tx.send(self.cache.match_method(request, method));
|
|
},
|
|
CORSCacheTaskMsg::MatchMethodUpdate(request, method, new_max_age, tx) => {
|
|
tx.send(self.cache.match_method_and_update(request, method, new_max_age));
|
|
},
|
|
CORSCacheTaskMsg::Insert(entry, tx) => {
|
|
self.cache.insert(entry);
|
|
tx.send(());
|
|
},
|
|
CORSCacheTaskMsg::ExitMsg => break
|
|
}
|
|
}
|
|
}
|
|
}
|