blob: 5896576b46e83767b55835833f17d06b4c09d179 [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "web/ServiceWorkerGlobalScopeProxy.h"
#include <memory>
#include <utility>
#include "bindings/core/v8/SourceLocation.h"
#include "bindings/core/v8/WorkerOrWorkletScriptController.h"
#include "core/dom/Document.h"
#include "core/dom/ExecutionContext.h"
#include "core/dom/MessagePort.h"
#include "core/inspector/ConsoleMessage.h"
#include "core/origin_trials/OriginTrials.h"
#include "core/workers/ParentFrameTaskRunners.h"
#include "core/workers/WorkerGlobalScope.h"
#include "core/workers/WorkerThread.h"
#include "modules/background_sync/SyncEvent.h"
#include "modules/fetch/Headers.h"
#include "modules/notifications/Notification.h"
#include "modules/notifications/NotificationEvent.h"
#include "modules/notifications/NotificationEventInit.h"
#include "modules/payments/PaymentAppRequest.h"
#include "modules/payments/PaymentAppRequestConversion.h"
#include "modules/payments/PaymentRequestEvent.h"
#include "modules/payments/PaymentRequestRespondWithObserver.h"
#include "modules/push_messaging/PushEvent.h"
#include "modules/push_messaging/PushMessageData.h"
#include "modules/serviceworkers/ExtendableEvent.h"
#include "modules/serviceworkers/ExtendableMessageEvent.h"
#include "modules/serviceworkers/FetchEvent.h"
#include "modules/serviceworkers/ForeignFetchEvent.h"
#include "modules/serviceworkers/InstallEvent.h"
#include "modules/serviceworkers/ServiceWorkerClient.h"
#include "modules/serviceworkers/ServiceWorkerGlobalScope.h"
#include "modules/serviceworkers/ServiceWorkerWindowClient.h"
#include "modules/serviceworkers/WaitUntilObserver.h"
#include "platform/CrossThreadFunctional.h"
#include "platform/RuntimeEnabledFeatures.h"
#include "public/platform/modules/notifications/WebNotificationData.h"
#include "public/platform/modules/serviceworker/WebServiceWorkerEventResult.h"
#include "public/platform/modules/serviceworker/WebServiceWorkerRequest.h"
#include "public/web/WebSerializedScriptValue.h"
#include "public/web/modules/serviceworker/WebServiceWorkerContextClient.h"
#include "web/WebEmbeddedWorkerImpl.h"
#include "wtf/Assertions.h"
#include "wtf/Functional.h"
#include "wtf/PtrUtil.h"
namespace blink {
ServiceWorkerGlobalScopeProxy* ServiceWorkerGlobalScopeProxy::create(
WebEmbeddedWorkerImpl& embeddedWorker,
Document& document,
WebServiceWorkerContextClient& client) {
return new ServiceWorkerGlobalScopeProxy(embeddedWorker, document, client);
}
ServiceWorkerGlobalScopeProxy::~ServiceWorkerGlobalScopeProxy() {
// Verify that the proxy has been detached.
DCHECK(!m_embeddedWorker);
}
DEFINE_TRACE(ServiceWorkerGlobalScopeProxy) {
visitor->trace(m_document);
visitor->trace(m_parentFrameTaskRunners);
visitor->trace(m_pendingPreloadFetchEvents);
}
void ServiceWorkerGlobalScopeProxy::setRegistration(
std::unique_ptr<WebServiceWorkerRegistration::Handle> handle) {
workerGlobalScope()->setRegistration(std::move(handle));
}
void ServiceWorkerGlobalScopeProxy::dispatchActivateEvent(int eventID) {
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Activate, eventID);
Event* event = ExtendableEvent::create(EventTypeNames::activate,
ExtendableEventInit(), observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchExtendableMessageEvent(
int eventID,
const WebString& message,
const WebSecurityOrigin& sourceOrigin,
WebMessagePortChannelArray webChannels,
const WebServiceWorkerClientInfo& client) {
WebSerializedScriptValue value =
WebSerializedScriptValue::fromString(message);
MessagePortArray* ports = MessagePort::toMessagePortArray(
m_workerGlobalScope, std::move(webChannels));
String origin;
if (!sourceOrigin.isUnique())
origin = sourceOrigin.toString();
ServiceWorkerClient* source = nullptr;
if (client.clientType == WebServiceWorkerClientTypeWindow)
source = ServiceWorkerWindowClient::create(client);
else
source = ServiceWorkerClient::create(client);
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Message, eventID);
Event* event =
ExtendableMessageEvent::create(value, origin, ports, source, observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchExtendableMessageEvent(
int eventID,
const WebString& message,
const WebSecurityOrigin& sourceOrigin,
WebMessagePortChannelArray webChannels,
std::unique_ptr<WebServiceWorker::Handle> handle) {
WebSerializedScriptValue value =
WebSerializedScriptValue::fromString(message);
MessagePortArray* ports = MessagePort::toMessagePortArray(
m_workerGlobalScope, std::move(webChannels));
String origin;
if (!sourceOrigin.isUnique())
origin = sourceOrigin.toString();
ServiceWorker* source =
ServiceWorker::from(m_workerGlobalScope->getExecutionContext(),
WTF::wrapUnique(handle.release()));
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Message, eventID);
Event* event =
ExtendableMessageEvent::create(value, origin, ports, source, observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchFetchEvent(
int fetchEventID,
const WebServiceWorkerRequest& webRequest,
bool navigationPreloadSent) {
ScriptState::Scope scope(
workerGlobalScope()->scriptController()->getScriptState());
WaitUntilObserver* waitUntilObserver = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Fetch, fetchEventID);
FetchRespondWithObserver* respondWithObserver =
FetchRespondWithObserver::create(
workerGlobalScope(), fetchEventID, webRequest.url(),
webRequest.mode(), webRequest.redirectMode(), webRequest.frameType(),
webRequest.requestContext(), waitUntilObserver);
Request* request = Request::create(
workerGlobalScope()->scriptController()->getScriptState(), webRequest);
request->getHeaders()->setGuard(Headers::ImmutableGuard);
FetchEventInit eventInit;
eventInit.setCancelable(true);
eventInit.setRequest(request);
eventInit.setClientId(
webRequest.isMainResourceLoad() ? WebString() : webRequest.clientId());
eventInit.setIsReload(webRequest.isReload());
ScriptState* scriptState =
workerGlobalScope()->scriptController()->getScriptState();
FetchEvent* fetchEvent = FetchEvent::create(
scriptState, EventTypeNames::fetch, eventInit, respondWithObserver,
waitUntilObserver, navigationPreloadSent);
if (navigationPreloadSent) {
// Keep |fetchEvent| until onNavigationPreloadResponse() or
// onNavigationPreloadError() will be called.
m_pendingPreloadFetchEvents.insert(fetchEventID, fetchEvent);
}
waitUntilObserver->willDispatchEvent();
respondWithObserver->willDispatchEvent();
DispatchEventResult dispatchResult =
workerGlobalScope()->dispatchEvent(fetchEvent);
respondWithObserver->didDispatchEvent(dispatchResult);
// false is okay because waitUntil for fetch event doesn't care about the
// promise rejection or an uncaught runtime script error.
waitUntilObserver->didDispatchEvent(false /* errorOccurred */);
}
void ServiceWorkerGlobalScopeProxy::onNavigationPreloadResponse(
int fetchEventID,
std::unique_ptr<WebURLResponse> response,
std::unique_ptr<WebDataConsumerHandle> dataConsumeHandle) {
FetchEvent* fetchEvent = m_pendingPreloadFetchEvents.take(fetchEventID);
DCHECK(fetchEvent);
fetchEvent->onNavigationPreloadResponse(
workerGlobalScope()->scriptController()->getScriptState(),
std::move(response), std::move(dataConsumeHandle));
}
void ServiceWorkerGlobalScopeProxy::onNavigationPreloadError(
int fetchEventID,
std::unique_ptr<WebServiceWorkerError> error) {
FetchEvent* fetchEvent = m_pendingPreloadFetchEvents.take(fetchEventID);
// This method may be called after onNavigationPreloadResponse() was called.
if (!fetchEvent)
return;
fetchEvent->onNavigationPreloadError(
workerGlobalScope()->scriptController()->getScriptState(),
std::move(error));
}
void ServiceWorkerGlobalScopeProxy::dispatchForeignFetchEvent(
int fetchEventID,
const WebServiceWorkerRequest& webRequest) {
if (!OriginTrials::foreignFetchEnabled(workerGlobalScope())) {
// If origin trial tokens have expired, or are otherwise no longer valid
// no events should be dispatched.
// TODO(mek): Ideally the browser wouldn't even start the service worker
// if its tokens have expired.
ServiceWorkerGlobalScopeClient::from(workerGlobalScope())
->respondToFetchEvent(fetchEventID, WTF::currentTime());
ServiceWorkerGlobalScopeClient::from(workerGlobalScope())
->didHandleFetchEvent(fetchEventID,
WebServiceWorkerEventResultCompleted,
WTF::currentTime());
return;
}
ScriptState::Scope scope(
workerGlobalScope()->scriptController()->getScriptState());
RefPtr<SecurityOrigin> origin =
SecurityOrigin::create(webRequest.referrerUrl());
WaitUntilObserver* waitUntilObserver = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Fetch, fetchEventID);
ForeignFetchRespondWithObserver* respondWithObserver =
ForeignFetchRespondWithObserver::create(
workerGlobalScope(), fetchEventID, webRequest.url(),
webRequest.mode(), webRequest.redirectMode(), webRequest.frameType(),
webRequest.requestContext(), origin, waitUntilObserver);
Request* request = Request::create(
workerGlobalScope()->scriptController()->getScriptState(), webRequest);
request->getHeaders()->setGuard(Headers::ImmutableGuard);
ForeignFetchEventInit eventInit;
eventInit.setCancelable(true);
eventInit.setRequest(request);
eventInit.setOrigin(origin->toString());
ForeignFetchEvent* fetchEvent = ForeignFetchEvent::create(
workerGlobalScope()->scriptController()->getScriptState(),
EventTypeNames::foreignfetch, eventInit, respondWithObserver,
waitUntilObserver);
waitUntilObserver->willDispatchEvent();
respondWithObserver->willDispatchEvent();
DispatchEventResult dispatchResult =
workerGlobalScope()->dispatchEvent(fetchEvent);
respondWithObserver->didDispatchEvent(dispatchResult);
// false is okay because waitUntil for foreign fetch event doesn't care
// about the promise rejection or an uncaught runtime script error.
waitUntilObserver->didDispatchEvent(false /* errorOccurred */);
}
void ServiceWorkerGlobalScopeProxy::dispatchInstallEvent(int eventID) {
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Install, eventID);
Event* event = InstallEvent::create(EventTypeNames::install,
ExtendableEventInit(), observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchNotificationClickEvent(
int eventID,
const WebString& notificationID,
const WebNotificationData& data,
int actionIndex,
const WebString& reply) {
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::NotificationClick, eventID);
NotificationEventInit eventInit;
eventInit.setNotification(Notification::create(
workerGlobalScope(), notificationID, data, true /* showing */));
if (0 <= actionIndex && actionIndex < static_cast<int>(data.actions.size()))
eventInit.setAction(data.actions[actionIndex].action);
eventInit.setReply(reply);
Event* event = NotificationEvent::create(EventTypeNames::notificationclick,
eventInit, observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchNotificationCloseEvent(
int eventID,
const WebString& notificationID,
const WebNotificationData& data) {
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::NotificationClose, eventID);
NotificationEventInit eventInit;
eventInit.setAction(WTF::String()); // initialize as null.
eventInit.setNotification(Notification::create(
workerGlobalScope(), notificationID, data, false /* showing */));
Event* event = NotificationEvent::create(EventTypeNames::notificationclose,
eventInit, observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchPushEvent(int eventID,
const WebString& data) {
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Push, eventID);
Event* event = PushEvent::create(EventTypeNames::push,
PushMessageData::create(data), observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchSyncEvent(
int eventID,
const WebString& tag,
LastChanceOption lastChance) {
if (!RuntimeEnabledFeatures::backgroundSyncEnabled()) {
ServiceWorkerGlobalScopeClient::from(workerGlobalScope())
->didHandleSyncEvent(eventID, WebServiceWorkerEventResultCompleted,
WTF::currentTime());
return;
}
WaitUntilObserver* observer = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::Sync, eventID);
Event* event = SyncEvent::create(EventTypeNames::sync, tag,
lastChance == IsLastChance, observer);
workerGlobalScope()->dispatchExtendableEvent(event, observer);
}
void ServiceWorkerGlobalScopeProxy::dispatchPaymentRequestEvent(
int eventID,
const WebPaymentAppRequest& webAppRequest) {
WaitUntilObserver* waitUntilObserver = WaitUntilObserver::create(
workerGlobalScope(), WaitUntilObserver::PaymentRequest, eventID);
RespondWithObserver* respondWithObserver =
PaymentRequestRespondWithObserver::create(workerGlobalScope(), eventID,
waitUntilObserver);
Event* event = PaymentRequestEvent::create(
EventTypeNames::paymentrequest,
PaymentAppRequestConversion::toPaymentAppRequest(
workerGlobalScope()->scriptController()->getScriptState(),
webAppRequest),
respondWithObserver, waitUntilObserver);
waitUntilObserver->willDispatchEvent();
respondWithObserver->willDispatchEvent();
DispatchEventResult dispatchResult =
workerGlobalScope()->dispatchEvent(event);
respondWithObserver->didDispatchEvent(dispatchResult);
// false is okay because waitUntil for payment request event doesn't care
// about the promise rejection or an uncaught runtime script error.
waitUntilObserver->didDispatchEvent(false /* errorOccurred */);
}
bool ServiceWorkerGlobalScopeProxy::hasFetchEventHandler() {
DCHECK(m_workerGlobalScope);
return m_workerGlobalScope->hasEventListeners(EventTypeNames::fetch);
}
void ServiceWorkerGlobalScopeProxy::countFeature(UseCounter::Feature feature) {
client().countFeature(static_cast<uint32_t>(feature));
}
void ServiceWorkerGlobalScopeProxy::countDeprecation(
UseCounter::Feature feature) {
// Go through the same code path with countFeature() because a deprecation
// message is already shown on the worker console and a remaining work is just
// to record an API use.
countFeature(feature);
}
void ServiceWorkerGlobalScopeProxy::reportException(
const String& errorMessage,
std::unique_ptr<SourceLocation> location,
int exceptionId) {
client().reportException(errorMessage, location->lineNumber(),
location->columnNumber(), location->url());
}
void ServiceWorkerGlobalScopeProxy::reportConsoleMessage(
MessageSource source,
MessageLevel level,
const String& message,
SourceLocation* location) {
client().reportConsoleMessage(source, level, message, location->lineNumber(),
location->url());
}
void ServiceWorkerGlobalScopeProxy::postMessageToPageInspector(
const String& message) {
DCHECK(m_embeddedWorker);
// The TaskType of Inspector tasks need to be Unthrottled because they need to
// run even on a suspended page.
m_parentFrameTaskRunners->get(TaskType::Unthrottled)
->postTask(
BLINK_FROM_HERE,
crossThreadBind(&WebEmbeddedWorkerImpl::postMessageToPageInspector,
crossThreadUnretained(m_embeddedWorker), message));
}
void ServiceWorkerGlobalScopeProxy::didCreateWorkerGlobalScope(
WorkerOrWorkletGlobalScope* workerGlobalScope) {
DCHECK(!m_workerGlobalScope);
m_workerGlobalScope =
static_cast<ServiceWorkerGlobalScope*>(workerGlobalScope);
client().workerContextStarted(this);
}
void ServiceWorkerGlobalScopeProxy::didInitializeWorkerContext() {
ScriptState::Scope scope(
workerGlobalScope()->scriptController()->getScriptState());
client().didInitializeWorkerContext(
workerGlobalScope()->scriptController()->context());
}
void ServiceWorkerGlobalScopeProxy::willEvaluateWorkerScript(
size_t scriptSize,
size_t cachedMetadataSize) {
DCHECK(m_workerGlobalScope);
m_workerGlobalScope->countScript(scriptSize, cachedMetadataSize);
}
void ServiceWorkerGlobalScopeProxy::willEvaluateImportedScript(
size_t scriptSize,
size_t cachedMetadataSize) {
DCHECK(m_workerGlobalScope);
m_workerGlobalScope->countScript(scriptSize, cachedMetadataSize);
}
void ServiceWorkerGlobalScopeProxy::didEvaluateWorkerScript(bool success) {
DCHECK(m_workerGlobalScope);
m_workerGlobalScope->didEvaluateWorkerScript();
client().didEvaluateWorkerScript(success);
}
void ServiceWorkerGlobalScopeProxy::didCloseWorkerGlobalScope() {
// This should never be called because close() is not defined in
// ServiceWorkerGlobalScope.
NOTREACHED();
}
void ServiceWorkerGlobalScopeProxy::willDestroyWorkerGlobalScope() {
v8::HandleScope handleScope(workerGlobalScope()->thread()->isolate());
client().willDestroyWorkerContext(
workerGlobalScope()->scriptController()->context());
m_workerGlobalScope = nullptr;
}
void ServiceWorkerGlobalScopeProxy::didTerminateWorkerThread() {
client().workerContextDestroyed();
}
ServiceWorkerGlobalScopeProxy::ServiceWorkerGlobalScopeProxy(
WebEmbeddedWorkerImpl& embeddedWorker,
Document& document,
WebServiceWorkerContextClient& client)
: m_embeddedWorker(&embeddedWorker),
m_document(&document),
m_client(&client),
m_workerGlobalScope(nullptr) {
// ServiceWorker can sometimes run tasks that are initiated by/associated with
// a document's frame but these documents can be from a different process. So
// we intentionally populate the task runners with null document in order to
// use the thread's default task runner. Note that |m_document| should not be
// used as it's a dummy document for loading that doesn't represent the frame
// of any associated document.
m_parentFrameTaskRunners = ParentFrameTaskRunners::create(nullptr);
}
void ServiceWorkerGlobalScopeProxy::detach() {
m_embeddedWorker = nullptr;
m_document = nullptr;
m_client = nullptr;
m_workerGlobalScope = nullptr;
}
WebServiceWorkerContextClient& ServiceWorkerGlobalScopeProxy::client() const {
DCHECK(m_client);
return *m_client;
}
Document& ServiceWorkerGlobalScopeProxy::document() const {
DCHECK(m_document);
return *m_document;
}
ServiceWorkerGlobalScope* ServiceWorkerGlobalScopeProxy::workerGlobalScope()
const {
DCHECK(m_workerGlobalScope);
return m_workerGlobalScope;
}
} // namespace blink