forked from mirrors/gecko-dev
		
	 be2b512750
			
		
	
	
		be2b512750
		
	
	
	
	
		
			
			The rustup is needed for https://github.com/rust-lang/rust/pull/40039. Source-Repo: https://github.com/servo/servo Source-Revision: a204c4176dcccdad8ec99d74055c66794c3f64ba --HG-- rename : servo/components/domobject_derive/Cargo.toml => servo/components/dom_struct/Cargo.toml extra : subtree_source : https%3A//hg.mozilla.org/projects/converted-servo-linear extra : subtree_revision : 2cad140c3d6b99837f0bd15a6f3ccba0f353e049
		
			
				
	
	
		
			120 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
	
		
			4.5 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/. */
 | |
| 
 | |
| use dom::bindings::codegen::Bindings::ServiceWorkerContainerBinding::{ServiceWorkerContainerMethods, Wrap};
 | |
| use dom::bindings::codegen::Bindings::ServiceWorkerContainerBinding::RegistrationOptions;
 | |
| use dom::bindings::error::Error;
 | |
| use dom::bindings::js::{JS, MutNullableJS, Root};
 | |
| use dom::bindings::reflector::{DomObject, reflect_dom_object};
 | |
| use dom::bindings::str::USVString;
 | |
| use dom::client::Client;
 | |
| use dom::eventtarget::EventTarget;
 | |
| use dom::globalscope::GlobalScope;
 | |
| use dom::promise::Promise;
 | |
| use dom::serviceworker::ServiceWorker;
 | |
| use dom_struct::dom_struct;
 | |
| use script_thread::ScriptThread;
 | |
| use serviceworkerjob::{Job, JobType};
 | |
| use std::ascii::AsciiExt;
 | |
| use std::default::Default;
 | |
| use std::rc::Rc;
 | |
| 
 | |
| #[dom_struct]
 | |
| pub struct ServiceWorkerContainer {
 | |
|     eventtarget: EventTarget,
 | |
|     controller: MutNullableJS<ServiceWorker>,
 | |
|     client: JS<Client>
 | |
| }
 | |
| 
 | |
| impl ServiceWorkerContainer {
 | |
|     fn new_inherited(client: &Client) -> ServiceWorkerContainer {
 | |
|         ServiceWorkerContainer {
 | |
|             eventtarget: EventTarget::new_inherited(),
 | |
|             controller: Default::default(),
 | |
|             client: JS::from_ref(client),
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[allow(unrooted_must_root)]
 | |
|     pub fn new(global: &GlobalScope) -> Root<ServiceWorkerContainer> {
 | |
|         let client = Client::new(&global.as_window());
 | |
|         let container = ServiceWorkerContainer::new_inherited(&*client);
 | |
|         reflect_dom_object(box container, global, Wrap)
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl ServiceWorkerContainerMethods for ServiceWorkerContainer {
 | |
|     // https://w3c.github.io/ServiceWorker/#service-worker-container-controller-attribute
 | |
|     fn GetController(&self) -> Option<Root<ServiceWorker>> {
 | |
|         self.client.get_controller()
 | |
|     }
 | |
| 
 | |
|     #[allow(unrooted_must_root)]
 | |
|     // https://w3c.github.io/ServiceWorker/#service-worker-container-register-method and - A
 | |
|     // https://w3c.github.io/ServiceWorker/#start-register-algorithm - B
 | |
|     fn Register(&self,
 | |
|                 script_url: USVString,
 | |
|                 options: &RegistrationOptions) -> Rc<Promise> {
 | |
|         // A: Step 1
 | |
|         let promise = Promise::new(&*self.global());
 | |
|         let ctx = (&*self.global()).get_cx();
 | |
|         let USVString(ref script_url) = script_url;
 | |
|         let api_base_url = self.global().api_base_url();
 | |
|         // A: Step 3-5
 | |
|         let script_url = match api_base_url.join(script_url) {
 | |
|             Ok(url) => url,
 | |
|             Err(_) => {
 | |
|                 promise.reject_error(ctx, Error::Type("Invalid script URL".to_owned()));
 | |
|                 return promise;
 | |
|             }
 | |
|         };
 | |
|         // B: Step 2
 | |
|         match script_url.scheme() {
 | |
|             "https" | "http" => {},
 | |
|             _ => {
 | |
|                 promise.reject_error(ctx, Error::Type("Only secure origins are allowed".to_owned()));
 | |
|                 return promise;
 | |
|             }
 | |
|         }
 | |
|         // B: Step 3
 | |
|         if script_url.path().to_ascii_lowercase().contains("%2f") ||
 | |
|         script_url.path().to_ascii_lowercase().contains("%5c") {
 | |
|             promise.reject_error(ctx, Error::Type("Script URL contains forbidden characters".to_owned()));
 | |
|             return promise;
 | |
|         }
 | |
|         // B: Step 4-5
 | |
|         let scope = match options.scope {
 | |
|             Some(ref scope) => {
 | |
|                 let &USVString(ref inner_scope) = scope;
 | |
|                 match api_base_url.join(inner_scope) {
 | |
|                     Ok(url) => url,
 | |
|                     Err(_) => {
 | |
|                         promise.reject_error(ctx, Error::Type("Invalid scope URL".to_owned()));
 | |
|                         return promise;
 | |
|                     }
 | |
|                 }
 | |
|             },
 | |
|             None => script_url.join("./").unwrap()
 | |
|         };
 | |
|         // B: Step 6
 | |
|         match scope.scheme() {
 | |
|             "https" | "http" => {},
 | |
|             _ => {
 | |
|                 promise.reject_error(ctx, Error::Type("Only secure origins are allowed".to_owned()));
 | |
|                 return promise;
 | |
|             }
 | |
|         }
 | |
|         // B: Step 7
 | |
|         if scope.path().to_ascii_lowercase().contains("%2f") ||
 | |
|         scope.path().to_ascii_lowercase().contains("%5c") {
 | |
|             promise.reject_error(ctx, Error::Type("Scope URL contains forbidden characters".to_owned()));
 | |
|             return promise;
 | |
|         }
 | |
| 
 | |
|         // B: Step 8
 | |
|         let job = Job::create_job(JobType::Register, scope, script_url, promise.clone(), &*self.client);
 | |
|         ScriptThread::schedule_job(job, &*self.global());
 | |
|         promise
 | |
|     }
 | |
| }
 |