|  | /* | 
|  | * 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 "config.h" | 
|  | #include "ServiceWorkerGlobalScope.h" | 
|  |  | 
|  | #include "bindings/core/v8/ScriptPromise.h" | 
|  | #include "bindings/core/v8/ScriptPromiseResolver.h" | 
|  | #include "bindings/core/v8/ScriptState.h" | 
|  | #include "bindings/core/v8/V8ThrowException.h" | 
|  | #include "core/dom/ExceptionCode.h" | 
|  | #include "core/events/Event.h" | 
|  | #include "core/fetch/MemoryCache.h" | 
|  | #include "core/fetch/ResourceLoaderOptions.h" | 
|  | #include "core/inspector/ConsoleMessage.h" | 
|  | #include "core/inspector/ScriptCallStack.h" | 
|  | #include "core/inspector/WorkerInspectorController.h" | 
|  | #include "core/loader/ThreadableLoader.h" | 
|  | #include "core/workers/WorkerClients.h" | 
|  | #include "core/workers/WorkerThreadStartupData.h" | 
|  | #include "modules/EventTargetModules.h" | 
|  | #include "modules/cachestorage/CacheStorage.h" | 
|  | #include "modules/cachestorage/InspectorCacheStorageAgent.h" | 
|  | #include "modules/fetch/GlobalFetch.h" | 
|  | #include "modules/serviceworkers/ServiceWorkerClients.h" | 
|  | #include "modules/serviceworkers/ServiceWorkerGlobalScopeClient.h" | 
|  | #include "modules/serviceworkers/ServiceWorkerRegistration.h" | 
|  | #include "modules/serviceworkers/ServiceWorkerScriptCachedMetadataHandler.h" | 
|  | #include "modules/serviceworkers/ServiceWorkerThread.h" | 
|  | #include "modules/serviceworkers/WaitUntilObserver.h" | 
|  | #include "platform/network/ResourceRequest.h" | 
|  | #include "platform/weborigin/DatabaseIdentifier.h" | 
|  | #include "platform/weborigin/KURL.h" | 
|  | #include "public/platform/Platform.h" | 
|  | #include "public/platform/WebURL.h" | 
|  | #include "public/platform/modules/serviceworker/WebServiceWorkerSkipWaitingCallbacks.h" | 
|  | #include "wtf/CurrentTime.h" | 
|  |  | 
|  | namespace blink { | 
|  |  | 
|  | class ServiceWorkerGlobalScope::SkipWaitingCallback final : public WebServiceWorkerSkipWaitingCallbacks { | 
|  | WTF_MAKE_NONCOPYABLE(SkipWaitingCallback); | 
|  | public: | 
|  | explicit SkipWaitingCallback(ScriptPromiseResolver* resolver) | 
|  | : m_resolver(resolver) { } | 
|  | ~SkipWaitingCallback() { } | 
|  |  | 
|  | void onSuccess() override | 
|  | { | 
|  | m_resolver->resolve(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | Persistent<ScriptPromiseResolver> m_resolver; | 
|  | }; | 
|  |  | 
|  | PassRefPtrWillBeRawPtr<ServiceWorkerGlobalScope> ServiceWorkerGlobalScope::create(ServiceWorkerThread* thread, PassOwnPtr<WorkerThreadStartupData> startupData) | 
|  | { | 
|  | // Note: startupData is finalized on return. After the relevant parts has been | 
|  | // passed along to the created 'context'. | 
|  | RefPtrWillBeRawPtr<ServiceWorkerGlobalScope> context = adoptRefWillBeNoop(new ServiceWorkerGlobalScope(startupData->m_scriptURL, startupData->m_userAgent, thread, monotonicallyIncreasingTime(), startupData->m_starterOriginPrivilegeData.release(), startupData->m_workerClients.release())); | 
|  |  | 
|  | context->setV8CacheOptions(startupData->m_v8CacheOptions); | 
|  | context->applyContentSecurityPolicyFromVector(*startupData->m_contentSecurityPolicyHeaders); | 
|  |  | 
|  | return context.release(); | 
|  | } | 
|  |  | 
|  | ServiceWorkerGlobalScope::ServiceWorkerGlobalScope(const KURL& url, const String& userAgent, ServiceWorkerThread* thread, double timeOrigin, PassOwnPtr<SecurityOrigin::PrivilegeData> starterOriginPrivilegeData, PassOwnPtrWillBeRawPtr<WorkerClients> workerClients) | 
|  | : WorkerGlobalScope(url, userAgent, thread, timeOrigin, starterOriginPrivilegeData, workerClients) | 
|  | , m_didEvaluateScript(false) | 
|  | , m_hadErrorInTopLevelEventHandler(false) | 
|  | , m_eventNestingLevel(0) | 
|  | , m_scriptCount(0) | 
|  | , m_scriptTotalSize(0) | 
|  | , m_scriptCachedMetadataTotalSize(0) | 
|  | { | 
|  | } | 
|  |  | 
|  | ServiceWorkerGlobalScope::~ServiceWorkerGlobalScope() | 
|  | { | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::didEvaluateWorkerScript() | 
|  | { | 
|  | if (Platform* platform = Platform::current()) { | 
|  | platform->histogramCustomCounts("ServiceWorker.ScriptCount", m_scriptCount, 1, 1000, 50); | 
|  | platform->histogramCustomCounts("ServiceWorker.ScriptTotalSize", m_scriptTotalSize, 1000, 5000000, 50); | 
|  | if (m_scriptCachedMetadataTotalSize) | 
|  | platform->histogramCustomCounts("ServiceWorker.ScriptCachedMetadataTotalSize", m_scriptCachedMetadataTotalSize, 1000, 50000000, 50); | 
|  | } | 
|  | m_didEvaluateScript = true; | 
|  | } | 
|  |  | 
|  | ScriptPromise ServiceWorkerGlobalScope::fetch(ScriptState* scriptState, const RequestInfo& input, const Dictionary& init, ExceptionState& exceptionState) | 
|  | { | 
|  | return GlobalFetch::fetch(scriptState, *this, input, init, exceptionState); | 
|  | } | 
|  |  | 
|  | ServiceWorkerClients* ServiceWorkerGlobalScope::clients() | 
|  | { | 
|  | if (!m_clients) | 
|  | m_clients = ServiceWorkerClients::create(); | 
|  | return m_clients; | 
|  | } | 
|  |  | 
|  | ServiceWorkerRegistration* ServiceWorkerGlobalScope::registration() | 
|  | { | 
|  | return m_registration; | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::close(ExceptionState& exceptionState) | 
|  | { | 
|  | exceptionState.throwDOMException(InvalidAccessError, "Not supported."); | 
|  | } | 
|  |  | 
|  | ScriptPromise ServiceWorkerGlobalScope::skipWaiting(ScriptState* scriptState) | 
|  | { | 
|  | ExecutionContext* executionContext = scriptState->executionContext(); | 
|  | // FIXME: short-term fix, see details at: https://codereview.chromium.org/535193002/. | 
|  | if (!executionContext) | 
|  | return ScriptPromise(); | 
|  |  | 
|  | ScriptPromiseResolver* resolver = ScriptPromiseResolver::create(scriptState); | 
|  | ScriptPromise promise = resolver->promise(); | 
|  |  | 
|  | ServiceWorkerGlobalScopeClient::from(executionContext)->skipWaiting(new SkipWaitingCallback(resolver)); | 
|  | return promise; | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::setRegistration(WebPassOwnPtr<WebServiceWorkerRegistration::Handle> handle) | 
|  | { | 
|  | if (!executionContext()) | 
|  | return; | 
|  | m_registration = ServiceWorkerRegistration::create(executionContext(), handle.release()); | 
|  | } | 
|  |  | 
|  | bool ServiceWorkerGlobalScope::addEventListener(const AtomicString& eventType, PassRefPtrWillBeRawPtr<EventListener> listener, bool useCapture) | 
|  | { | 
|  | if (m_didEvaluateScript) { | 
|  | if (eventType == EventTypeNames::install) { | 
|  | RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, WarningMessageLevel, "Event handler of 'install' event must be added on the initial evaluation of worker script."); | 
|  | addMessageToWorkerConsole(consoleMessage.release()); | 
|  | } else if (eventType == EventTypeNames::activate) { | 
|  | RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, WarningMessageLevel, "Event handler of 'activate' event must be added on the initial evaluation of worker script."); | 
|  | addMessageToWorkerConsole(consoleMessage.release()); | 
|  | } | 
|  | } | 
|  | return WorkerGlobalScope::addEventListener(eventType, listener, useCapture); | 
|  | } | 
|  |  | 
|  | const AtomicString& ServiceWorkerGlobalScope::interfaceName() const | 
|  | { | 
|  | return EventTargetNames::ServiceWorkerGlobalScope; | 
|  | } | 
|  |  | 
|  | bool ServiceWorkerGlobalScope::dispatchEventInternal(PassRefPtrWillBeRawPtr<Event> event) | 
|  | { | 
|  | m_eventNestingLevel++; | 
|  | bool result = WorkerGlobalScope::dispatchEventInternal(event.get()); | 
|  | if (event->interfaceName() == EventNames::ErrorEvent && m_eventNestingLevel == 2 && !event->defaultPrevented()) | 
|  | m_hadErrorInTopLevelEventHandler = true; | 
|  | m_eventNestingLevel--; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::dispatchExtendableEvent(PassRefPtrWillBeRawPtr<Event> event, WaitUntilObserver* observer) | 
|  | { | 
|  | ASSERT(m_eventNestingLevel == 0); | 
|  | m_hadErrorInTopLevelEventHandler = false; | 
|  |  | 
|  | observer->willDispatchEvent(); | 
|  | dispatchEvent(event); | 
|  | if (thread()->terminated()) | 
|  | m_hadErrorInTopLevelEventHandler = true; | 
|  | observer->didDispatchEvent(m_hadErrorInTopLevelEventHandler); | 
|  | } | 
|  |  | 
|  | DEFINE_TRACE(ServiceWorkerGlobalScope) | 
|  | { | 
|  | visitor->trace(m_clients); | 
|  | visitor->trace(m_registration); | 
|  | WorkerGlobalScope::trace(visitor); | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::importScripts(const Vector<String>& urls, ExceptionState& exceptionState) | 
|  | { | 
|  | // Bust the MemoryCache to ensure script requests reach the browser-side | 
|  | // and get added to and retrieved from the ServiceWorker's script cache. | 
|  | // FIXME: Revisit in light of the solution to crbug/388375. | 
|  | for (Vector<String>::const_iterator it = urls.begin(); it != urls.end(); ++it) | 
|  | executionContext()->removeURLFromMemoryCache(completeURL(*it)); | 
|  | WorkerGlobalScope::importScripts(urls, exceptionState); | 
|  | } | 
|  |  | 
|  | PassOwnPtrWillBeRawPtr<CachedMetadataHandler> ServiceWorkerGlobalScope::createWorkerScriptCachedMetadataHandler(const KURL& scriptURL, const Vector<char>* metaData) | 
|  | { | 
|  | return ServiceWorkerScriptCachedMetadataHandler::create(this, scriptURL, metaData); | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::logExceptionToConsole(const String& errorMessage, int scriptId, const String& sourceURL, int lineNumber, int columnNumber, PassRefPtrWillBeRawPtr<ScriptCallStack> callStack) | 
|  | { | 
|  | WorkerGlobalScope::logExceptionToConsole(errorMessage, scriptId, sourceURL, lineNumber, columnNumber, callStack); | 
|  | RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, ErrorMessageLevel, errorMessage, sourceURL, lineNumber); | 
|  | consoleMessage->setScriptId(scriptId); | 
|  | consoleMessage->setCallStack(callStack); | 
|  | addMessageToWorkerConsole(consoleMessage.release()); | 
|  | } | 
|  |  | 
|  | void ServiceWorkerGlobalScope::scriptLoaded(size_t scriptSize, size_t cachedMetadataSize) | 
|  | { | 
|  | ++m_scriptCount; | 
|  | m_scriptTotalSize += scriptSize; | 
|  | m_scriptCachedMetadataTotalSize += cachedMetadataSize; | 
|  | } | 
|  |  | 
|  | } // namespace blink |