mirror of
https://github.com/servo/servo.git
synced 2025-08-05 13:40:08 +01:00
Remove incorrect uses of asynchronous tasks from ServiceWorker job queue.
This commit is contained in:
parent
b5d2bd757b
commit
60d1717eca
2 changed files with 161 additions and 145 deletions
|
@ -35,7 +35,6 @@ use dom::bindings::inheritance::Castable;
|
||||||
use dom::bindings::js::{JS, MutNullableJS, Root, RootCollection};
|
use dom::bindings::js::{JS, MutNullableJS, Root, RootCollection};
|
||||||
use dom::bindings::js::{RootCollectionPtr, RootedReference};
|
use dom::bindings::js::{RootCollectionPtr, RootedReference};
|
||||||
use dom::bindings::num::Finite;
|
use dom::bindings::num::Finite;
|
||||||
use dom::bindings::refcounted::Trusted;
|
|
||||||
use dom::bindings::reflector::DomObject;
|
use dom::bindings::reflector::DomObject;
|
||||||
use dom::bindings::str::DOMString;
|
use dom::bindings::str::DOMString;
|
||||||
use dom::bindings::trace::JSTraceable;
|
use dom::bindings::trace::JSTraceable;
|
||||||
|
@ -94,7 +93,7 @@ use script_traits::CompositorEvent::{KeyEvent, MouseButtonEvent, MouseMoveEvent,
|
||||||
use script_traits::CompositorEvent::{TouchEvent, TouchpadPressureEvent};
|
use script_traits::CompositorEvent::{TouchEvent, TouchpadPressureEvent};
|
||||||
use script_traits::WebVREventMsg;
|
use script_traits::WebVREventMsg;
|
||||||
use script_traits::webdriver_msg::WebDriverScriptCommand;
|
use script_traits::webdriver_msg::WebDriverScriptCommand;
|
||||||
use serviceworkerjob::{Job, JobQueue, AsyncJobHandler, FinishJobHandler, InvokeType, SettleType};
|
use serviceworkerjob::{Job, JobQueue, AsyncJobHandler};
|
||||||
use servo_config::opts;
|
use servo_config::opts;
|
||||||
use servo_url::ServoUrl;
|
use servo_url::ServoUrl;
|
||||||
use std::cell::Cell;
|
use std::cell::Cell;
|
||||||
|
@ -112,7 +111,6 @@ use std::thread;
|
||||||
use style::context::ReflowGoal;
|
use style::context::ReflowGoal;
|
||||||
use style::dom::{TNode, UnsafeNode};
|
use style::dom::{TNode, UnsafeNode};
|
||||||
use style::thread_state;
|
use style::thread_state;
|
||||||
use task_source::TaskSource;
|
|
||||||
use task_source::dom_manipulation::{DOMManipulationTask, DOMManipulationTaskSource};
|
use task_source::dom_manipulation::{DOMManipulationTask, DOMManipulationTaskSource};
|
||||||
use task_source::file_reading::FileReadingTaskSource;
|
use task_source::file_reading::FileReadingTaskSource;
|
||||||
use task_source::history_traversal::HistoryTraversalTaskSource;
|
use task_source::history_traversal::HistoryTraversalTaskSource;
|
||||||
|
@ -1469,49 +1467,11 @@ impl ScriptThread {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn dispatch_job_queue(&self, job_handler: Box<AsyncJobHandler>) {
|
pub fn dispatch_job_queue(&self, job_handler: Box<AsyncJobHandler>) {
|
||||||
let scope_url = job_handler.scope_url.clone();
|
self.job_queue_map.run_job(job_handler, self);
|
||||||
let queue_ref = self.job_queue_map.0.borrow();
|
|
||||||
let front_job = {
|
|
||||||
let job_vec = queue_ref.get(&scope_url);
|
|
||||||
job_vec.unwrap().first().unwrap()
|
|
||||||
};
|
|
||||||
match job_handler.invoke_type {
|
|
||||||
InvokeType::Run => (&*self.job_queue_map).run_job(job_handler, self),
|
|
||||||
InvokeType::Register => self.job_queue_map.run_register(front_job, job_handler, self),
|
|
||||||
InvokeType::Update => self.job_queue_map.update(front_job, &*front_job.client.global(), self),
|
|
||||||
InvokeType::Settle(settle_type) => {
|
|
||||||
let promise = &front_job.promise;
|
|
||||||
let global = &*front_job.client.global();
|
|
||||||
let trusted_global = Trusted::new(global);
|
|
||||||
let _ac = JSAutoCompartment::new(global.get_cx(), promise.reflector().get_jsobject().get());
|
|
||||||
match settle_type {
|
|
||||||
SettleType::Resolve(reg) => promise.resolve_native(global.get_cx(), &*reg.root()),
|
|
||||||
SettleType::Reject(err) => promise.reject_error(global.get_cx(), err)
|
|
||||||
}
|
|
||||||
let finish_job_handler = box FinishJobHandler::new(scope_url, trusted_global);
|
|
||||||
self.queue_finish_job(finish_job_handler, global);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn queue_serviceworker_job(&self, async_job_handler: Box<AsyncJobHandler>, global: &GlobalScope) {
|
pub fn dom_manipulation_task_source(&self) -> &DOMManipulationTaskSource {
|
||||||
let _ = self.dom_manipulation_task_source.queue(async_job_handler, &*global);
|
&self.dom_manipulation_task_source
|
||||||
}
|
|
||||||
|
|
||||||
pub fn queue_finish_job(&self, finish_job_handler: Box<FinishJobHandler>, global: &GlobalScope) {
|
|
||||||
let _ = self.dom_manipulation_task_source.queue(finish_job_handler, global);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn invoke_finish_job(&self, finish_job_handler: Box<FinishJobHandler>) {
|
|
||||||
let job_queue = &*self.job_queue_map;
|
|
||||||
let global = &*finish_job_handler.global.root();
|
|
||||||
let scope_url = (*finish_job_handler).scope_url;
|
|
||||||
job_queue.finish_job(scope_url, global, self);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn invoke_job_update(&self, job: &Job, global: &GlobalScope) {
|
|
||||||
let job_queue = &*self.job_queue_map;
|
|
||||||
job_queue.update(job, global, self);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Handles a request for the window title.
|
/// Handles a request for the window title.
|
||||||
|
|
|
@ -10,20 +10,23 @@
|
||||||
use dom::bindings::cell::DOMRefCell;
|
use dom::bindings::cell::DOMRefCell;
|
||||||
use dom::bindings::error::Error;
|
use dom::bindings::error::Error;
|
||||||
use dom::bindings::js::JS;
|
use dom::bindings::js::JS;
|
||||||
use dom::bindings::refcounted::Trusted;
|
use dom::bindings::refcounted::{Trusted, TrustedPromise};
|
||||||
use dom::bindings::reflector::DomObject;
|
use dom::bindings::reflector::DomObject;
|
||||||
use dom::client::Client;
|
use dom::client::Client;
|
||||||
use dom::globalscope::GlobalScope;
|
use dom::globalscope::GlobalScope;
|
||||||
use dom::promise::Promise;
|
use dom::promise::Promise;
|
||||||
use dom::serviceworkerregistration::ServiceWorkerRegistration;
|
use dom::serviceworkerregistration::ServiceWorkerRegistration;
|
||||||
use dom::urlhelper::UrlHelper;
|
use dom::urlhelper::UrlHelper;
|
||||||
|
use js::jsapi::JSAutoCompartment;
|
||||||
use script_thread::{ScriptThread, Runnable};
|
use script_thread::{ScriptThread, Runnable};
|
||||||
use servo_url::ServoUrl;
|
use servo_url::ServoUrl;
|
||||||
use std::cmp::PartialEq;
|
use std::cmp::PartialEq;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
|
use task_source::TaskSource;
|
||||||
|
use task_source::dom_manipulation::DOMManipulationTaskSource;
|
||||||
|
|
||||||
#[derive(PartialEq, Clone, Debug, JSTraceable)]
|
#[derive(PartialEq, Copy, Clone, Debug, JSTraceable)]
|
||||||
pub enum JobType {
|
pub enum JobType {
|
||||||
Register,
|
Register,
|
||||||
Unregister,
|
Unregister,
|
||||||
|
@ -36,14 +39,6 @@ pub enum SettleType {
|
||||||
Reject(Error)
|
Reject(Error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// This encapsulates what operation to invoke of JobQueue from script thread
|
|
||||||
pub enum InvokeType {
|
|
||||||
Settle(SettleType),
|
|
||||||
Run,
|
|
||||||
Register,
|
|
||||||
Update
|
|
||||||
}
|
|
||||||
|
|
||||||
#[must_root]
|
#[must_root]
|
||||||
#[derive(JSTraceable)]
|
#[derive(JSTraceable)]
|
||||||
pub struct Job {
|
pub struct Job {
|
||||||
|
@ -98,36 +93,14 @@ impl PartialEq for Job {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct FinishJobHandler {
|
|
||||||
pub scope_url: ServoUrl,
|
|
||||||
pub global: Trusted<GlobalScope>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl FinishJobHandler {
|
|
||||||
pub fn new(scope_url: ServoUrl, global: Trusted<GlobalScope>) -> FinishJobHandler {
|
|
||||||
FinishJobHandler {
|
|
||||||
scope_url: scope_url,
|
|
||||||
global: global
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Runnable for FinishJobHandler {
|
|
||||||
fn main_thread_handler(self: Box<FinishJobHandler>, script_thread: &ScriptThread) {
|
|
||||||
script_thread.invoke_finish_job(self);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct AsyncJobHandler {
|
pub struct AsyncJobHandler {
|
||||||
pub scope_url: ServoUrl,
|
pub scope_url: ServoUrl,
|
||||||
pub invoke_type: InvokeType
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsyncJobHandler {
|
impl AsyncJobHandler {
|
||||||
fn new(scope_url: ServoUrl, invoke_type: InvokeType) -> AsyncJobHandler {
|
fn new(scope_url: ServoUrl) -> AsyncJobHandler {
|
||||||
AsyncJobHandler {
|
AsyncJobHandler {
|
||||||
scope_url: scope_url,
|
scope_url: scope_url,
|
||||||
invoke_type: invoke_type
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -153,25 +126,29 @@ impl JobQueue {
|
||||||
job: Job,
|
job: Job,
|
||||||
global: &GlobalScope,
|
global: &GlobalScope,
|
||||||
script_thread: &ScriptThread) {
|
script_thread: &ScriptThread) {
|
||||||
|
debug!("scheduling {:?} job", job.job_type);
|
||||||
let mut queue_ref = self.0.borrow_mut();
|
let mut queue_ref = self.0.borrow_mut();
|
||||||
let job_queue = queue_ref.entry(job.scope_url.clone()).or_insert(vec![]);
|
let job_queue = queue_ref.entry(job.scope_url.clone()).or_insert(vec![]);
|
||||||
// Step 1
|
// Step 1
|
||||||
if job_queue.is_empty() {
|
if job_queue.is_empty() {
|
||||||
let scope_url = job.scope_url.clone();
|
let scope_url = job.scope_url.clone();
|
||||||
job_queue.push(job);
|
job_queue.push(job);
|
||||||
let run_job_handler = AsyncJobHandler::new(scope_url, InvokeType::Run);
|
let run_job_handler = box AsyncJobHandler::new(scope_url);
|
||||||
script_thread.queue_serviceworker_job(box run_job_handler, global);
|
let _ = script_thread.dom_manipulation_task_source().queue(run_job_handler, global);
|
||||||
|
debug!("queued task to run newly-queued job");
|
||||||
} else {
|
} else {
|
||||||
// Step 2
|
// Step 2
|
||||||
let mut last_job = job_queue.pop().unwrap();
|
let mut last_job = job_queue.pop().unwrap();
|
||||||
if job == last_job && !last_job.promise.is_settled() {
|
if job == last_job && !last_job.promise.is_settled() {
|
||||||
last_job.append_equivalent_job(job);
|
last_job.append_equivalent_job(job);
|
||||||
job_queue.push(last_job);
|
job_queue.push(last_job);
|
||||||
|
debug!("appended equivalent job");
|
||||||
} else {
|
} else {
|
||||||
// restore the popped last_job
|
// restore the popped last_job
|
||||||
job_queue.push(last_job);
|
job_queue.push(last_job);
|
||||||
// and push this new job to job queue
|
// and push this new job to job queue
|
||||||
job_queue.push(job);
|
job_queue.push(job);
|
||||||
|
debug!("pushed onto job queue job");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -179,41 +156,46 @@ impl JobQueue {
|
||||||
#[allow(unrooted_must_root)]
|
#[allow(unrooted_must_root)]
|
||||||
// https://w3c.github.io/ServiceWorker/#run-job-algorithm
|
// https://w3c.github.io/ServiceWorker/#run-job-algorithm
|
||||||
pub fn run_job(&self, run_job_handler: Box<AsyncJobHandler>, script_thread: &ScriptThread) {
|
pub fn run_job(&self, run_job_handler: Box<AsyncJobHandler>, script_thread: &ScriptThread) {
|
||||||
let queue_ref = &*self.0.borrow();
|
debug!("running a job");
|
||||||
let front_job = {
|
let url = {
|
||||||
let job_vec = queue_ref.get(&run_job_handler.scope_url);
|
let queue_ref = self.0.borrow();
|
||||||
job_vec.unwrap().first().unwrap()
|
let front_job = {
|
||||||
|
let job_vec = queue_ref.get(&run_job_handler.scope_url);
|
||||||
|
job_vec.unwrap().first().unwrap()
|
||||||
|
};
|
||||||
|
let scope_url = front_job.scope_url.clone();
|
||||||
|
match front_job.job_type {
|
||||||
|
JobType::Register => self.run_register(front_job, run_job_handler, script_thread),
|
||||||
|
JobType::Update => self.update(front_job, script_thread),
|
||||||
|
JobType::Unregister => unreachable!(),
|
||||||
|
};
|
||||||
|
scope_url
|
||||||
};
|
};
|
||||||
let global = &*front_job.client.global();
|
self.finish_job(url, script_thread);
|
||||||
let handler = *run_job_handler;
|
|
||||||
match front_job.job_type {
|
|
||||||
JobType::Register => {
|
|
||||||
let register_job_handler = AsyncJobHandler::new(handler.scope_url, InvokeType::Register);
|
|
||||||
script_thread.queue_serviceworker_job(box register_job_handler, global);
|
|
||||||
},
|
|
||||||
JobType::Update => {
|
|
||||||
let update_job_handler = AsyncJobHandler::new(handler.scope_url, InvokeType::Update);
|
|
||||||
script_thread.queue_serviceworker_job(box update_job_handler, global);
|
|
||||||
}
|
|
||||||
_ => { /* TODO implement Unregister */ }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unrooted_must_root)]
|
#[allow(unrooted_must_root)]
|
||||||
// https://w3c.github.io/ServiceWorker/#register-algorithm
|
// https://w3c.github.io/ServiceWorker/#register-algorithm
|
||||||
pub fn run_register(&self, job: &Job, register_job_handler: Box<AsyncJobHandler>, script_thread: &ScriptThread) {
|
fn run_register(&self, job: &Job, register_job_handler: Box<AsyncJobHandler>, script_thread: &ScriptThread) {
|
||||||
let global = &*job.client.global();
|
debug!("running register job");
|
||||||
let AsyncJobHandler { scope_url, .. } = *register_job_handler;
|
let AsyncJobHandler { scope_url, .. } = *register_job_handler;
|
||||||
// Step 1-3
|
// Step 1-3
|
||||||
if !UrlHelper::is_origin_trustworthy(&job.script_url) {
|
if !UrlHelper::is_origin_trustworthy(&job.script_url) {
|
||||||
let settle_type = SettleType::Reject(Error::Type("Invalid script ServoURL".to_owned()));
|
// Step 1.1
|
||||||
let async_job_handler = AsyncJobHandler::new(scope_url, InvokeType::Settle(settle_type));
|
reject_job_promise(job,
|
||||||
return script_thread.queue_serviceworker_job(box async_job_handler, global);
|
Error::Type("Invalid script ServoURL".to_owned()),
|
||||||
|
script_thread.dom_manipulation_task_source());
|
||||||
|
// Step 1.2 (see run_job)
|
||||||
|
return;
|
||||||
} else if job.script_url.origin() != job.referrer.origin() || job.scope_url.origin() != job.referrer.origin() {
|
} else if job.script_url.origin() != job.referrer.origin() || job.scope_url.origin() != job.referrer.origin() {
|
||||||
let settle_type = SettleType::Reject(Error::Security);
|
// Step 2.1/3.1
|
||||||
let async_job_handler = AsyncJobHandler::new(scope_url, InvokeType::Settle(settle_type));
|
reject_job_promise(job,
|
||||||
return script_thread.queue_serviceworker_job(box async_job_handler, global);
|
Error::Security,
|
||||||
|
script_thread.dom_manipulation_task_source());
|
||||||
|
// Step 2.2/3.2 (see run_job)
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Step 4-5
|
// Step 4-5
|
||||||
if let Some(reg) = script_thread.handle_get_registration(&job.scope_url) {
|
if let Some(reg) = script_thread.handle_get_registration(&job.scope_url) {
|
||||||
// Step 5.1
|
// Step 5.1
|
||||||
|
@ -223,70 +205,144 @@ impl JobQueue {
|
||||||
// Step 5.3
|
// Step 5.3
|
||||||
if let Some(ref newest_worker) = reg.get_newest_worker() {
|
if let Some(ref newest_worker) = reg.get_newest_worker() {
|
||||||
if (&*newest_worker).get_script_url() == job.script_url {
|
if (&*newest_worker).get_script_url() == job.script_url {
|
||||||
let settle_type = SettleType::Resolve(Trusted::new(&*reg));
|
// Step 5.3.1
|
||||||
let async_job_handler = AsyncJobHandler::new(scope_url, InvokeType::Settle(settle_type));
|
resolve_job_promise(job, &*reg, script_thread.dom_manipulation_task_source());
|
||||||
script_thread.queue_serviceworker_job(box async_job_handler, global);
|
// Step 5.3.2 (see run_job)
|
||||||
let finish_job_handler = box FinishJobHandler::new(job.scope_url.clone(), Trusted::new(&*global));
|
return;
|
||||||
script_thread.queue_finish_job(finish_job_handler, &*global);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Step 6.1
|
// Step 6.1
|
||||||
|
let global = &*job.client.global();
|
||||||
let pipeline = global.pipeline_id();
|
let pipeline = global.pipeline_id();
|
||||||
let new_reg = ServiceWorkerRegistration::new(&*global, &job.script_url, scope_url);
|
let new_reg = ServiceWorkerRegistration::new(&*global, &job.script_url, scope_url);
|
||||||
script_thread.handle_serviceworker_registration(&job.scope_url, &*new_reg, pipeline);
|
script_thread.handle_serviceworker_registration(&job.scope_url, &*new_reg, pipeline);
|
||||||
}
|
}
|
||||||
// Step 7
|
// Step 7
|
||||||
script_thread.invoke_job_update(job, &*global);
|
self.update(job, script_thread)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unrooted_must_root)]
|
#[allow(unrooted_must_root)]
|
||||||
// https://w3c.github.io/ServiceWorker/#finish-job-algorithm
|
// https://w3c.github.io/ServiceWorker/#finish-job-algorithm
|
||||||
pub fn finish_job(&self, scope_url: ServoUrl, global: &GlobalScope, script_thread: &ScriptThread) {
|
pub fn finish_job(&self, scope_url: ServoUrl, script_thread: &ScriptThread) {
|
||||||
if let Some(job_vec) = (*self.0.borrow_mut()).get_mut(&scope_url) {
|
debug!("finishing previous job");
|
||||||
if job_vec.first().map_or(false, |job| job.scope_url == scope_url) {
|
let run_job = if let Some(job_vec) = (*self.0.borrow_mut()).get_mut(&scope_url) {
|
||||||
let _ = job_vec.remove(0);
|
assert_eq!(job_vec.first().as_ref().unwrap().scope_url, scope_url);
|
||||||
}
|
let _ = job_vec.remove(0);
|
||||||
if !job_vec.is_empty() {
|
!job_vec.is_empty()
|
||||||
let run_job_handler = AsyncJobHandler::new(scope_url, InvokeType::Run);
|
|
||||||
script_thread.queue_serviceworker_job(box run_job_handler, global);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
warn!("non-existent job vector for Servourl: {:?}", scope_url);
|
warn!("non-existent job vector for Servourl: {:?}", scope_url);
|
||||||
|
false
|
||||||
|
};
|
||||||
|
|
||||||
|
if run_job {
|
||||||
|
debug!("further jobs in queue after finishing");
|
||||||
|
let handler = box AsyncJobHandler::new(scope_url);
|
||||||
|
self.run_job(handler, script_thread);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://w3c.github.io/ServiceWorker/#update-algorithm
|
// https://w3c.github.io/ServiceWorker/#update-algorithm
|
||||||
pub fn update(&self, job: &Job, global: &GlobalScope, script_thread: &ScriptThread) {
|
fn update(&self, job: &Job, script_thread: &ScriptThread) {
|
||||||
|
debug!("running update job");
|
||||||
|
// Step 1
|
||||||
let reg = match script_thread.handle_get_registration(&job.scope_url) {
|
let reg = match script_thread.handle_get_registration(&job.scope_url) {
|
||||||
Some(reg) => reg,
|
Some(reg) => reg,
|
||||||
None => return
|
None => {
|
||||||
};
|
let err_type = Error::Type("No registration to update".to_owned());
|
||||||
// Step 1
|
// Step 2.1
|
||||||
if reg.get_uninstalling() {
|
reject_job_promise(job, err_type, script_thread.dom_manipulation_task_source());
|
||||||
let err_type = Error::Type("Update called on an uninstalling registration".to_owned());
|
// Step 2.2 (see run_job)
|
||||||
let settle_type = SettleType::Reject(err_type);
|
return;
|
||||||
let async_job_handler = AsyncJobHandler::new(job.scope_url.clone(), InvokeType::Settle(settle_type));
|
}
|
||||||
return script_thread.queue_serviceworker_job(box async_job_handler, global);
|
|
||||||
}
|
|
||||||
let newest_worker = match reg.get_newest_worker() {
|
|
||||||
Some(worker) => worker,
|
|
||||||
None => return
|
|
||||||
};
|
};
|
||||||
// Step 2
|
// Step 2
|
||||||
if (&*newest_worker).get_script_url() == job.script_url && job.job_type == JobType::Update {
|
if reg.get_uninstalling() {
|
||||||
// Step 4
|
let err_type = Error::Type("Update called on an uninstalling registration".to_owned());
|
||||||
let err_type = Error::Type("Invalid script ServoURL".to_owned());
|
// Step 2.1
|
||||||
let settle_type = SettleType::Reject(err_type);
|
reject_job_promise(job, err_type, script_thread.dom_manipulation_task_source());
|
||||||
let async_job_handler = AsyncJobHandler::new(job.scope_url.clone(), InvokeType::Settle(settle_type));
|
// Step 2.2 (see run_job)
|
||||||
script_thread.queue_serviceworker_job(box async_job_handler, global);
|
return;
|
||||||
} else {
|
|
||||||
job.client.set_controller(&*newest_worker);
|
|
||||||
let settle_type = SettleType::Resolve(Trusted::new(&*reg));
|
|
||||||
let async_job_handler = AsyncJobHandler::new(job.scope_url.clone(), InvokeType::Settle(settle_type));
|
|
||||||
script_thread.queue_serviceworker_job(box async_job_handler, global);
|
|
||||||
}
|
}
|
||||||
let finish_job_handler = box FinishJobHandler::new(job.scope_url.clone(), Trusted::new(global));
|
// Step 3
|
||||||
script_thread.queue_finish_job(finish_job_handler, global);
|
let newest_worker = reg.get_newest_worker();
|
||||||
|
let newest_worker_url = newest_worker.as_ref().map(|w| w.get_script_url());
|
||||||
|
// Step 4
|
||||||
|
if newest_worker_url.as_ref() == Some(&job.script_url) && job.job_type == JobType::Update {
|
||||||
|
let err_type = Error::Type("Invalid script ServoURL".to_owned());
|
||||||
|
// Step 4.1
|
||||||
|
reject_job_promise(job, err_type, script_thread.dom_manipulation_task_source());
|
||||||
|
// Step 4.2 (see run_job)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Step 8
|
||||||
|
if let Some(newest_worker) = newest_worker {
|
||||||
|
job.client.set_controller(&*newest_worker);
|
||||||
|
// Step 8.1
|
||||||
|
resolve_job_promise(job, &*reg, script_thread.dom_manipulation_task_source());
|
||||||
|
// Step 8.2 present in run_job
|
||||||
|
}
|
||||||
|
// TODO Step 9 (create new service worker)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct AsyncPromiseSettle {
|
||||||
|
global: Trusted<GlobalScope>,
|
||||||
|
promise: TrustedPromise,
|
||||||
|
settle_type: SettleType,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Runnable for AsyncPromiseSettle {
|
||||||
|
#[allow(unrooted_must_root)]
|
||||||
|
fn handler(self: Box<AsyncPromiseSettle>) {
|
||||||
|
let global = self.global.root();
|
||||||
|
let settle_type = self.settle_type.clone();
|
||||||
|
let promise = self.promise.root();
|
||||||
|
settle_job_promise(&*global, &*promise, settle_type)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsyncPromiseSettle {
|
||||||
|
#[allow(unrooted_must_root)]
|
||||||
|
fn new(promise: Rc<Promise>, settle_type: SettleType) -> AsyncPromiseSettle {
|
||||||
|
AsyncPromiseSettle {
|
||||||
|
global: Trusted::new(&*promise.global()),
|
||||||
|
promise: TrustedPromise::new(promise),
|
||||||
|
settle_type: settle_type,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn settle_job_promise(global: &GlobalScope, promise: &Promise, settle: SettleType) {
|
||||||
|
let _ac = JSAutoCompartment::new(global.get_cx(), promise.reflector().get_jsobject().get());
|
||||||
|
match settle {
|
||||||
|
SettleType::Resolve(reg) => promise.resolve_native(global.get_cx(), &*reg.root()),
|
||||||
|
SettleType::Reject(err) => promise.reject_error(global.get_cx(), err),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn queue_settle_promise_for_job(job: &Job, settle: SettleType, task_source: &DOMManipulationTaskSource) {
|
||||||
|
let task = box AsyncPromiseSettle::new(job.promise.clone(), settle);
|
||||||
|
let global = job.client.global();
|
||||||
|
let _ = task_source.queue(task, &*global);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://w3c.github.io/ServiceWorker/#reject-job-promise-algorithm
|
||||||
|
// https://w3c.github.io/ServiceWorker/#resolve-job-promise-algorithm
|
||||||
|
fn queue_settle_promise(job: &Job, settle: SettleType, task_source: &DOMManipulationTaskSource) {
|
||||||
|
// Step 1
|
||||||
|
queue_settle_promise_for_job(job, settle.clone(), task_source);
|
||||||
|
// Step 2
|
||||||
|
for job in &job.equivalent_jobs {
|
||||||
|
queue_settle_promise_for_job(job, settle.clone(), task_source);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn reject_job_promise(job: &Job, err: Error, task_source: &DOMManipulationTaskSource) {
|
||||||
|
queue_settle_promise(job, SettleType::Reject(err), task_source)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn resolve_job_promise(job: &Job, reg: &ServiceWorkerRegistration, task_source: &DOMManipulationTaskSource) {
|
||||||
|
queue_settle_promise(job, SettleType::Resolve(Trusted::new(reg)), task_source)
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue