blob: ea4072db2a5958fa42fe35d9d82dda0152767717 [file] [log] [blame]
// Copyright 2013 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/renderer/service_worker/embedded_worker_context_client.h"
#include "base/lazy_instance.h"
#include "base/message_loop/message_loop_proxy.h"
#include "base/pickle.h"
#include "base/strings/string16.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_local.h"
#include "content/child/request_extra_data.h"
#include "content/child/service_worker/service_worker_network_provider.h"
#include "content/child/thread_safe_sender.h"
#include "content/child/worker_task_runner.h"
#include "content/child/worker_thread_task_runner.h"
#include "content/common/devtools_messages.h"
#include "content/common/service_worker/embedded_worker_messages.h"
#include "content/common/service_worker/service_worker_types.h"
#include "content/public/renderer/document_state.h"
#include "content/renderer/render_thread_impl.h"
#include "content/renderer/service_worker/embedded_worker_dispatcher.h"
#include "content/renderer/service_worker/service_worker_script_context.h"
#include "ipc/ipc_message_macros.h"
#include "third_party/WebKit/public/platform/WebServiceWorkerResponse.h"
#include "third_party/WebKit/public/platform/WebString.h"
#include "third_party/WebKit/public/web/WebDataSource.h"
#include "third_party/WebKit/public/web/WebServiceWorkerNetworkProvider.h"
namespace content {
namespace {
// For now client must be a per-thread instance.
// TODO(kinuko): This needs to be refactored when we start using thread pool
// or having multiple clients per one thread.
base::LazyInstance<base::ThreadLocalPointer<EmbeddedWorkerContextClient> >::
Leaky g_worker_client_tls = LAZY_INSTANCE_INITIALIZER;
void CallWorkerContextDestroyedOnMainThread(int embedded_worker_id) {
if (!RenderThreadImpl::current() ||
!RenderThreadImpl::current()->embedded_worker_dispatcher())
return;
RenderThreadImpl::current()->embedded_worker_dispatcher()->
WorkerContextDestroyed(embedded_worker_id);
}
// We store an instance of this class in the "extra data" of the WebDataSource
// and attach a ServiceWorkerNetworkProvider to it as base::UserData.
// (see createServiceWorkerNetworkProvider).
class DataSourceExtraData
: public blink::WebDataSource::ExtraData,
public base::SupportsUserData {
public:
DataSourceExtraData() {}
virtual ~DataSourceExtraData() {}
};
// Called on the main thread only and blink owns it.
class WebServiceWorkerNetworkProviderImpl
: public blink::WebServiceWorkerNetworkProvider {
public:
// Blink calls this method for each request starting with the main script,
// we tag them with the provider id.
virtual void willSendRequest(
blink::WebDataSource* data_source,
blink::WebURLRequest& request) {
ServiceWorkerNetworkProvider* provider =
ServiceWorkerNetworkProvider::FromDocumentState(
static_cast<DataSourceExtraData*>(data_source->extraData()));
scoped_ptr<RequestExtraData> extra_data(new RequestExtraData);
extra_data->set_service_worker_provider_id(provider->provider_id());
request.setExtraData(extra_data.release());
}
};
} // namespace
EmbeddedWorkerContextClient*
EmbeddedWorkerContextClient::ThreadSpecificInstance() {
return g_worker_client_tls.Pointer()->Get();
}
EmbeddedWorkerContextClient::EmbeddedWorkerContextClient(
int embedded_worker_id,
int64 service_worker_version_id,
const GURL& service_worker_scope,
const GURL& script_url,
int worker_devtools_agent_route_id)
: embedded_worker_id_(embedded_worker_id),
service_worker_version_id_(service_worker_version_id),
service_worker_scope_(service_worker_scope),
script_url_(script_url),
worker_devtools_agent_route_id_(worker_devtools_agent_route_id),
sender_(ChildThread::current()->thread_safe_sender()),
main_thread_proxy_(base::MessageLoopProxy::current()),
weak_factory_(this) {
}
EmbeddedWorkerContextClient::~EmbeddedWorkerContextClient() {
}
bool EmbeddedWorkerContextClient::OnMessageReceived(
const IPC::Message& msg) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(EmbeddedWorkerContextClient, msg)
IPC_MESSAGE_HANDLER(EmbeddedWorkerContextMsg_MessageToWorker,
OnMessageToWorker)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void EmbeddedWorkerContextClient::Send(IPC::Message* message) {
sender_->Send(message);
}
blink::WebURL EmbeddedWorkerContextClient::scope() const {
return service_worker_scope_;
}
void EmbeddedWorkerContextClient::getClients(
blink::WebServiceWorkerClientsCallbacks* callbacks) {
DCHECK(script_context_);
script_context_->GetClientDocuments(callbacks);
}
void EmbeddedWorkerContextClient::workerContextFailedToStart() {
DCHECK(main_thread_proxy_->RunsTasksOnCurrentThread());
DCHECK(!script_context_);
Send(new EmbeddedWorkerHostMsg_WorkerScriptLoadFailed(embedded_worker_id_));
RenderThreadImpl::current()->embedded_worker_dispatcher()->
WorkerContextDestroyed(embedded_worker_id_);
}
void EmbeddedWorkerContextClient::workerContextStarted(
blink::WebServiceWorkerContextProxy* proxy) {
DCHECK(!worker_task_runner_);
worker_task_runner_ = new WorkerThreadTaskRunner(
WorkerTaskRunner::Instance()->CurrentWorkerId());
DCHECK_NE(0, WorkerTaskRunner::Instance()->CurrentWorkerId());
// g_worker_client_tls.Pointer()->Get() could return NULL if this context
// gets deleted before workerContextStarted() is called.
DCHECK(g_worker_client_tls.Pointer()->Get() == NULL);
DCHECK(!script_context_);
g_worker_client_tls.Pointer()->Set(this);
script_context_.reset(new ServiceWorkerScriptContext(this, proxy));
Send(new EmbeddedWorkerHostMsg_WorkerScriptLoaded(embedded_worker_id_));
// Schedule a task to send back WorkerStarted asynchronously,
// so that at the time we send it we can be sure that the worker
// script has been evaluated and worker run loop has been started.
worker_task_runner_->PostTask(
FROM_HERE,
base::Bind(&EmbeddedWorkerContextClient::SendWorkerStarted,
weak_factory_.GetWeakPtr()));
}
void EmbeddedWorkerContextClient::willDestroyWorkerContext() {
// At this point OnWorkerRunLoopStopped is already called, so
// worker_task_runner_->RunsTasksOnCurrentThread() returns false
// (while we're still on the worker thread).
script_context_.reset();
// This also lets the message filter stop dispatching messages to
// this client.
g_worker_client_tls.Pointer()->Set(NULL);
}
void EmbeddedWorkerContextClient::workerContextDestroyed() {
DCHECK(g_worker_client_tls.Pointer()->Get() == NULL);
// Now we should be able to free the WebEmbeddedWorker container on the
// main thread.
main_thread_proxy_->PostTask(
FROM_HERE,
base::Bind(&CallWorkerContextDestroyedOnMainThread,
embedded_worker_id_));
}
void EmbeddedWorkerContextClient::reportException(
const blink::WebString& error_message,
int line_number,
int column_number,
const blink::WebString& source_url) {
Send(new EmbeddedWorkerHostMsg_ReportException(
embedded_worker_id_, error_message, line_number,
column_number, GURL(source_url)));
}
void EmbeddedWorkerContextClient::reportConsoleMessage(
int source,
int level,
const blink::WebString& message,
int line_number,
const blink::WebString& source_url) {
EmbeddedWorkerHostMsg_ReportConsoleMessage_Params params;
params.source_identifier = source;
params.message_level = level;
params.message = message;
params.line_number = line_number;
params.source_url = GURL(source_url);
Send(new EmbeddedWorkerHostMsg_ReportConsoleMessage(
embedded_worker_id_, params));
}
void EmbeddedWorkerContextClient::dispatchDevToolsMessage(
const blink::WebString& message) {
sender_->Send(new DevToolsClientMsg_DispatchOnInspectorFrontend(
worker_devtools_agent_route_id_, message.utf8()));
}
void EmbeddedWorkerContextClient::saveDevToolsAgentState(
const blink::WebString& state) {
sender_->Send(new DevToolsHostMsg_SaveAgentRuntimeState(
worker_devtools_agent_route_id_, state.utf8()));
}
void EmbeddedWorkerContextClient::didHandleActivateEvent(
int request_id,
blink::WebServiceWorkerEventResult result) {
DCHECK(script_context_);
script_context_->DidHandleActivateEvent(request_id, result);
}
void EmbeddedWorkerContextClient::didHandleInstallEvent(
int request_id,
blink::WebServiceWorkerEventResult result) {
DCHECK(script_context_);
script_context_->DidHandleInstallEvent(request_id, result);
}
void EmbeddedWorkerContextClient::didHandleFetchEvent(int request_id) {
DCHECK(script_context_);
script_context_->DidHandleFetchEvent(
request_id,
SERVICE_WORKER_FETCH_EVENT_RESULT_FALLBACK,
ServiceWorkerResponse());
}
void EmbeddedWorkerContextClient::didHandleFetchEvent(
int request_id,
const blink::WebServiceWorkerResponse& web_response) {
DCHECK(script_context_);
std::map<std::string, std::string> headers;
const blink::WebVector<blink::WebString>& header_keys =
web_response.getHeaderKeys();
for (size_t i = 0; i < header_keys.size(); ++i) {
const base::string16& key = header_keys[i];
headers[base::UTF16ToUTF8(key)] =
base::UTF16ToUTF8(web_response.getHeader(key));
}
ServiceWorkerResponse response(web_response.status(),
web_response.statusText().utf8(),
headers,
web_response.blobUUID().utf8());
script_context_->DidHandleFetchEvent(
request_id, SERVICE_WORKER_FETCH_EVENT_RESULT_RESPONSE, response);
}
void EmbeddedWorkerContextClient::didHandleSyncEvent(int request_id) {
DCHECK(script_context_);
script_context_->DidHandleSyncEvent(request_id);
}
blink::WebServiceWorkerNetworkProvider*
EmbeddedWorkerContextClient::createServiceWorkerNetworkProvider(
blink::WebDataSource* data_source) {
// Create a content::ServiceWorkerNetworkProvider for this data source so
// we can observe its requests.
scoped_ptr<ServiceWorkerNetworkProvider> provider(
new ServiceWorkerNetworkProvider());
// Tell the network provider about which version to load.
provider->SetServiceWorkerVersionId(service_worker_version_id_);
// The provider is kept around for the lifetime of the DataSource
// and ownership is transferred to the DataSource.
DataSourceExtraData* extra_data = new DataSourceExtraData();
data_source->setExtraData(extra_data);
ServiceWorkerNetworkProvider::AttachToDocumentState(
extra_data, provider.Pass());
// Blink is responsible for deleting the returned object.
return new WebServiceWorkerNetworkProviderImpl();
}
void EmbeddedWorkerContextClient::postMessageToClient(
int client_id,
const blink::WebString& message,
blink::WebMessagePortChannelArray* channels) {
DCHECK(script_context_);
script_context_->PostMessageToDocument(client_id, message,
make_scoped_ptr(channels));
}
void EmbeddedWorkerContextClient::OnMessageToWorker(
int thread_id,
int embedded_worker_id,
const IPC::Message& message) {
if (!script_context_)
return;
DCHECK_EQ(embedded_worker_id_, embedded_worker_id);
script_context_->OnMessageReceived(message);
}
void EmbeddedWorkerContextClient::SendWorkerStarted() {
DCHECK(worker_task_runner_->RunsTasksOnCurrentThread());
Send(new EmbeddedWorkerHostMsg_WorkerStarted(
WorkerTaskRunner::Instance()->CurrentWorkerId(),
embedded_worker_id_));
}
} // namespace content