blob: cb86fa56050c983d402d9277e3b9271bc6721576 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/child/service_worker/service_worker_provider_context.h"
#include "base/bind.h"
#include "base/message_loop/message_loop_proxy.h"
#include "base/stl_util.h"
#include "content/child/child_thread.h"
#include "content/child/service_worker/service_worker_dispatcher.h"
#include "content/child/service_worker/service_worker_handle_reference.h"
#include "content/child/service_worker/service_worker_registration_handle_reference.h"
#include "content/child/thread_safe_sender.h"
#include "content/child/worker_task_runner.h"
#include "content/common/service_worker/service_worker_messages.h"
namespace content {
ServiceWorkerProviderContext::ServiceWorkerProviderContext(int provider_id)
: provider_id_(provider_id),
main_thread_loop_proxy_(base::MessageLoopProxy::current()) {
if (!ChildThread::current())
return; // May be null in some tests.
thread_safe_sender_ = ChildThread::current()->thread_safe_sender();
ServiceWorkerDispatcher* dispatcher =
ServiceWorkerDispatcher::GetOrCreateThreadSpecificInstance(
thread_safe_sender_.get());
DCHECK(dispatcher);
dispatcher->AddProviderContext(this);
}
ServiceWorkerProviderContext::~ServiceWorkerProviderContext() {
if (ServiceWorkerDispatcher* dispatcher =
ServiceWorkerDispatcher::GetThreadSpecificInstance()) {
dispatcher->RemoveProviderContext(this);
}
}
ServiceWorkerHandleReference* ServiceWorkerProviderContext::installing() {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return installing_.get();
}
ServiceWorkerHandleReference* ServiceWorkerProviderContext::waiting() {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return waiting_.get();
}
ServiceWorkerHandleReference* ServiceWorkerProviderContext::active() {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return active_.get();
}
ServiceWorkerHandleReference* ServiceWorkerProviderContext::controller() {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return controller_.get();
}
ServiceWorkerRegistrationHandleReference*
ServiceWorkerProviderContext::registration() {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return registration_.get();
}
ServiceWorkerVersionAttributes
ServiceWorkerProviderContext::GetVersionAttributes() {
ServiceWorkerVersionAttributes attrs;
if (installing())
attrs.installing = installing()->info();
if (waiting())
attrs.waiting = waiting()->info();
if (active())
attrs.active = active()->info();
return attrs;
}
void ServiceWorkerProviderContext::OnAssociateRegistration(
const ServiceWorkerRegistrationObjectInfo& info,
const ServiceWorkerVersionAttributes& attrs) {
DCHECK(!registration_);
DCHECK_NE(kInvalidServiceWorkerRegistrationHandleId, info.handle_id);
registration_ = ServiceWorkerRegistrationHandleReference::Adopt(
info, thread_safe_sender_.get());
installing_ = ServiceWorkerHandleReference::Adopt(
attrs.installing, thread_safe_sender_.get());
waiting_ = ServiceWorkerHandleReference::Adopt(
attrs.waiting, thread_safe_sender_.get());
active_ = ServiceWorkerHandleReference::Adopt(
attrs.active, thread_safe_sender_.get());
}
void ServiceWorkerProviderContext::OnDisassociateRegistration() {
controller_.reset();
active_.reset();
waiting_.reset();
installing_.reset();
registration_.reset();
}
void ServiceWorkerProviderContext::OnServiceWorkerStateChanged(
int handle_id,
blink::WebServiceWorkerState state) {
ServiceWorkerHandleReference* which = NULL;
if (handle_id == controller_handle_id())
which = controller_.get();
else if (handle_id == active_handle_id())
which = active_.get();
else if (handle_id == waiting_handle_id())
which = waiting_.get();
else if (handle_id == installing_handle_id())
which = installing_.get();
// We should only get messages for ServiceWorkers associated with
// this provider.
DCHECK(which);
which->set_state(state);
// TODO(kinuko): We can forward the message to other threads here
// when we support navigator.serviceWorker in dedicated workers.
}
void ServiceWorkerProviderContext::OnSetInstallingServiceWorker(
int registration_handle_id,
const ServiceWorkerObjectInfo& info) {
DCHECK(IsAssociatedWithRegistration(registration_handle_id));
installing_ =
ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get());
}
void ServiceWorkerProviderContext::OnSetWaitingServiceWorker(
int registration_handle_id,
const ServiceWorkerObjectInfo& info) {
DCHECK(IsAssociatedWithRegistration(registration_handle_id));
waiting_ =
ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get());
}
void ServiceWorkerProviderContext::OnSetActiveServiceWorker(
int registration_handle_id,
const ServiceWorkerObjectInfo& info) {
DCHECK(IsAssociatedWithRegistration(registration_handle_id));
active_ =
ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get());
}
void ServiceWorkerProviderContext::OnSetControllerServiceWorker(
int registration_handle_id,
const ServiceWorkerObjectInfo& info) {
DCHECK(IsAssociatedWithRegistration(registration_handle_id));
// This context is is the primary owner of this handle, keeps the
// initial reference until it goes away.
controller_ =
ServiceWorkerHandleReference::Adopt(info, thread_safe_sender_.get());
// TODO(kinuko): We can forward the message to other threads here
// when we support navigator.serviceWorker in dedicated workers.
}
int ServiceWorkerProviderContext::installing_handle_id() const {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return installing_ ? installing_->info().handle_id
: kInvalidServiceWorkerHandleId;
}
int ServiceWorkerProviderContext::waiting_handle_id() const {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return waiting_ ? waiting_->info().handle_id
: kInvalidServiceWorkerHandleId;
}
int ServiceWorkerProviderContext::active_handle_id() const {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return active_ ? active_->info().handle_id
: kInvalidServiceWorkerHandleId;
}
int ServiceWorkerProviderContext::controller_handle_id() const {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return controller_ ? controller_->info().handle_id
: kInvalidServiceWorkerHandleId;
}
int ServiceWorkerProviderContext::registration_handle_id() const {
DCHECK(main_thread_loop_proxy_->RunsTasksOnCurrentThread());
return registration_ ? registration_->info().handle_id
: kInvalidServiceWorkerRegistrationHandleId;
}
bool ServiceWorkerProviderContext::IsAssociatedWithRegistration(
int registration_handle_id) const {
if (!registration_)
return false;
if (registration_handle_id == kInvalidServiceWorkerRegistrationHandleId)
return false;
return registration_->info().handle_id == registration_handle_id;
}
} // namespace content