diff options
author | Manish Goregaokar <manishsmail@gmail.com> | 2014-08-23 14:48:10 +0530 |
---|---|---|
committer | Manish Goregaokar <manishsmail@gmail.com> | 2014-08-23 21:30:02 +0530 |
commit | 5d7438a7dbeda64a3aa853d6ed27a76c2a26949a (patch) | |
tree | 99a3089fe7b854649315a2d26cd7ed6a826e6ec1 /src | |
parent | a3b5395d5069fabe24854778dd976d7b1bf03a58 (diff) | |
download | servo-5d7438a7dbeda64a3aa853d6ed27a76c2a26949a.tar.gz servo-5d7438a7dbeda64a3aa853d6ed27a76c2a26949a.zip |
Add a task-based CORS Cache
Diffstat (limited to 'src')
-rw-r--r-- | src/components/net/fetch/cors_cache.rs | 249 | ||||
-rw-r--r-- | src/components/net/fetch/request.rs | 7 |
2 files changed, 208 insertions, 48 deletions
diff --git a/src/components/net/fetch/cors_cache.rs b/src/components/net/fetch/cors_cache.rs index 482dc093e47..529edc683f7 100644 --- a/src/components/net/fetch/cors_cache.rs +++ b/src/components/net/fetch/cors_cache.rs @@ -4,14 +4,13 @@ use http::method::Method; use std::ascii::StrAsciiExt; +use std::comm::{Sender, Receiver, channel}; use time; use time::{now, Timespec}; use url::Url; -#[deriving(Clone)] -pub struct BasicCORSCache(Vec<CORSCacheEntry>); - /// Union type for CORS cache entries +/// /// Each entry might pertain to a header or method #[deriving(Clone)] pub enum HeaderOrMethod { @@ -61,65 +60,44 @@ impl CORSCacheEntry { /// Properties of Request required to cache match. pub struct CacheRequestDetails { - origin: Url, - destination: Url, + pub origin: Url, + pub destination: Url, + pub credentials: bool } -trait CORSCache { +/// Trait for a generic CORS Cache +pub trait CORSCache { /// [Clear the cache](http://fetch.spec.whatwg.org/#concept-cache-clear) - fn clear (&mut self, request: &CacheRequestDetails); + fn clear (&mut self, request: CacheRequestDetails); /// Remove old entries fn cleanup(&mut self); - /// [Finds an entry with a matching header](http://fetch.spec.whatwg.org/#concept-cache-match-header) - fn find_entry_by_header<'a>(&'a mut self, request: &CacheRequestDetails, header_name: &str) -> Option<&'a mut CORSCacheEntry>; + /// Returns true if an entry with a [matching header](http://fetch.spec.whatwg.org/#concept-cache-match-header) is found + fn match_header(&mut self, request: CacheRequestDetails, header_name: &str) -> bool; - /// Returns true if an entry with a matching header is found - fn match_header(&mut self, request: &CacheRequestDetails, header_name: &str) -> bool { - self.find_entry_by_header(request, header_name).is_some() - } + /// Updates max age if an entry for a [matching header](http://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: uint) -> bool; - /// Updates max age if an entry for the same header is found. - fn match_header_and_update(&mut self, request: &CacheRequestDetails, header_name: &str, new_max_age: uint) -> bool { - self.find_entry_by_header(request, header_name).map(|e| e.max_age = new_max_age).is_some() - } - - /// [Finds an entry with a matching method](http://fetch.spec.whatwg.org/#concept-cache-match-method) - fn find_entry_by_method<'a>(&'a mut self, request: &CacheRequestDetails, method: &Method) -> Option<&'a mut CORSCacheEntry>; - - /// Returns true if an entry with a matching method is found - fn match_method(&mut self, request: &CacheRequestDetails, method: &Method) -> bool { - self.find_entry_by_method(request, method).is_some() - } - - /// Updates max age if an entry for the same method is found. - fn match_method_and_update(&mut self, request: &CacheRequestDetails, method: &Method, new_max_age: uint) -> bool { - self.find_entry_by_method(request, method).map(|e| e.max_age = new_max_age).is_some() - } + /// Returns true if an entry with a [matching method](http://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](http://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: uint) -> bool; /// Insert an entry fn insert(&mut self, entry: CORSCacheEntry); } -impl CORSCache for BasicCORSCache { - /// http://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.move_iter().filter(|e| e.origin == request.origin && request.destination == e.url).collect(); - *self = BasicCORSCache(new_buf); - } +/// A simple, vector-based CORS Cache +#[deriving(Clone)] +pub struct BasicCORSCache(Vec<CORSCacheEntry>); - // Remove old entries - fn cleanup(&mut self) { - let BasicCORSCache(buf) = self.clone(); - let now = time::now().to_timespec(); - let new_buf: Vec<CORSCacheEntry> = buf.move_iter().filter(|e| now.sec > e.created.sec + e.max_age as i64).collect(); - *self = BasicCORSCache(new_buf); - } - /// http://fetch.spec.whatwg.org/#concept-cache-match-header +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; @@ -128,6 +106,7 @@ impl CORSCache for BasicCORSCache { e.origin.host() == request.origin.host() && e.origin.port() == request.origin.port() && e.url == request.destination && + e.credentials == request.credentials && e.header_or_method.match_header(header_name)); entry } @@ -141,9 +120,58 @@ impl CORSCache for BasicCORSCache { e.origin.host() == request.origin.host() && e.origin.port() == request.origin.port() && e.url == request.destination && + e.credentials == request.credentials && e.header_or_method.match_method(method)); entry } +} + +impl CORSCache for BasicCORSCache { + /// http://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.move_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.move_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: uint) -> 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, HeaderData(header_name.to_string()))); + 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: uint) -> bool { + match self.find_entry_by_method(&request, method).map(|e| e.max_age = new_max_age) { + Some(_) => true, + None => { + self.insert(CORSCacheEntry::new(request.origin, request.destination, new_max_age, + request.credentials, MethodData(method.clone()))); + false + } + } + } fn insert(&mut self, entry: CORSCacheEntry) { self.cleanup(); @@ -151,3 +179,132 @@ impl CORSCache for BasicCORSCache { 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, uint, Sender<bool>), + MatchMethod(CacheRequestDetails, Method, Sender<bool>), + MatchMethodUpdate(CacheRequestDetails, Method, uint, Sender<bool>), + Insert(CORSCacheEntry, Sender<()>) +} + +/// 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(Clear(request, tx)); + rx.recv_opt().ok(); + } + + fn cleanup(&mut self) { + let (tx, rx) = channel(); + self.send(Cleanup(tx)); + rx.recv_opt().ok(); + } + + fn match_header(&mut self, request: CacheRequestDetails, header_name: &str) -> bool { + let (tx, rx) = channel(); + self.send(MatchHeader(request, header_name.to_string(), tx)); + rx.recv_opt().ok().unwrap_or(false) + } + + fn match_header_and_update(&mut self, request: CacheRequestDetails, header_name: &str, new_max_age: uint) -> bool { + let (tx, rx) = channel(); + self.send(MatchHeaderUpdate(request, header_name.to_string(), new_max_age, tx)); + rx.recv_opt().ok().unwrap_or(false) + } + + fn match_method(&mut self, request: CacheRequestDetails, method: &Method) -> bool { + let (tx, rx) = channel(); + self.send(MatchMethod(request, method.clone(), tx)); + rx.recv_opt().ok().unwrap_or(false) + } + + fn match_method_and_update(&mut self, request: CacheRequestDetails, method: &Method, new_max_age: uint) -> bool { + let (tx, rx) = channel(); + self.send(MatchMethodUpdate(request, method.clone(), new_max_age, tx)); + rx.recv_opt().ok().unwrap_or(false) + } + + fn insert(&mut self, entry: CORSCacheEntry) { + let (tx, rx) = channel(); + self.send(Insert(entry, tx)); + rx.recv_opt().ok() ; + } +} + +/// A simple task-based CORS Cache that can be sent messages +/// +/// #Example +/// ``` +/// let task = CORSCacheTask::new(); +/// let builder = TaskBuilder::new().named("XHRTask"); +/// let mut sender = task.get_sender(); +/// builder.spawn(proc() { task.run() }); +/// sender.insert(CORSCacheEntry::new(/* parameters here */)); +/// ``` +pub struct CORSCacheTask { + receiver: Receiver<CORSCacheTaskMsg>, + pub 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 get_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 + pub fn run(&mut self) { + loop { + // The recv() here should never fail, we always + // carry a copy of the Sender with us. + match self.receiver.recv() { + Clear(request, tx) => { + self.cache.clear(request); + tx.send(()); + }, + Cleanup(tx) => { + self.cache.cleanup(); + tx.send(()); + }, + MatchHeader(request, header, tx) => { + tx.send(self.cache.match_header(request, header.as_slice())); + }, + MatchHeaderUpdate(request, header, new_max_age, tx) => { + tx.send(self.cache.match_header_and_update(request, header.as_slice(), new_max_age)); + }, + MatchMethod(request, method, tx) => { + tx.send(self.cache.match_method(request, &method)); + }, + MatchMethodUpdate(request, method, new_max_age, tx) => { + tx.send(self.cache.match_method_and_update(request, &method, new_max_age)); + }, + Insert(entry, tx) => { + self.cache.insert(entry); + tx.send(()); + } + } + } + } +}
\ No newline at end of file diff --git a/src/components/net/fetch/request.rs b/src/components/net/fetch/request.rs index 2dfd5572a97..bd5d87b7950 100644 --- a/src/components/net/fetch/request.rs +++ b/src/components/net/fetch/request.rs @@ -5,6 +5,7 @@ use url::Url; use http::method::{Get, Method}; use http::headers::request::HeaderCollection; +use fetch::cors_cache::CORSCache; use fetch::response::Response; /// A [request context](http://fetch.spec.whatwg.org/#concept-request-context) @@ -77,7 +78,8 @@ pub struct Request { pub use_url_credentials: bool, pub manual_redirect: bool, pub redirect_count: uint, - pub response_tainting: ResponseTainting + pub response_tainting: ResponseTainting, + pub cache: Option<Box<CORSCache>> } impl Request { @@ -103,7 +105,8 @@ impl Request { use_url_credentials: false, manual_redirect: false, redirect_count: 0, - response_tainting: Basic + response_tainting: Basic, + cache: None } } |