| /* |
| * Copyright (C) 2017 Apple 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: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. 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. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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. |
| */ |
| |
| #pragma once |
| |
| #if ENABLE(SERVICE_WORKER) |
| |
| #include "ExceptionOr.h" |
| #include "PageIdentifier.h" |
| #include "PushSubscriptionData.h" |
| #include "ServiceWorkerClientData.h" |
| #include "ServiceWorkerClientQueryOptions.h" |
| #include "ServiceWorkerIdentifier.h" |
| #include "ServiceWorkerThreadProxy.h" |
| #include <wtf/CompletionHandler.h> |
| #include <wtf/HashMap.h> |
| #include <wtf/Lock.h> |
| #include <wtf/URLHash.h> |
| |
| namespace WebCore { |
| |
| class SerializedScriptValue; |
| class ServiceWorkerGlobalScope; |
| |
| class SWContextManager { |
| public: |
| WEBCORE_EXPORT static SWContextManager& singleton(); |
| |
| class Connection { |
| public: |
| virtual ~Connection() { } |
| |
| virtual void establishConnection(CompletionHandler<void()>&&) = 0; |
| virtual void postMessageToServiceWorkerClient(const ScriptExecutionContextIdentifier& destinationIdentifier, const MessageWithMessagePorts&, ServiceWorkerIdentifier source, const String& sourceOrigin) = 0; |
| virtual void serviceWorkerStarted(std::optional<ServiceWorkerJobDataIdentifier>, ServiceWorkerIdentifier, bool doesHandleFetch) = 0; |
| virtual void serviceWorkerFailedToStart(std::optional<ServiceWorkerJobDataIdentifier>, ServiceWorkerIdentifier, const String& message) = 0; |
| virtual void didFinishInstall(std::optional<ServiceWorkerJobDataIdentifier>, ServiceWorkerIdentifier, bool wasSuccessful) = 0; |
| virtual void didFinishActivation(ServiceWorkerIdentifier) = 0; |
| virtual void setServiceWorkerHasPendingEvents(ServiceWorkerIdentifier, bool) = 0; |
| virtual void workerTerminated(ServiceWorkerIdentifier) = 0; |
| virtual void skipWaiting(ServiceWorkerIdentifier, CompletionHandler<void()>&&) = 0; |
| virtual void setScriptResource(ServiceWorkerIdentifier, const URL&, const ServiceWorkerContextData::ImportedScript&) = 0; |
| |
| using FindClientByIdentifierCallback = CompletionHandler<void(std::optional<ServiceWorkerClientData>&&)>; |
| virtual void findClientByVisibleIdentifier(ServiceWorkerIdentifier, const String&, FindClientByIdentifierCallback&&) = 0; |
| virtual void matchAll(ServiceWorkerIdentifier, const ServiceWorkerClientQueryOptions&, ServiceWorkerClientsMatchAllCallback&&) = 0; |
| virtual void claim(ServiceWorkerIdentifier, CompletionHandler<void(ExceptionOr<void>&&)>&&) = 0; |
| virtual void focus(ScriptExecutionContextIdentifier, CompletionHandler<void(std::optional<ServiceWorkerClientData>&&)>&&) = 0; |
| |
| using OpenWindowCallback = CompletionHandler<void(ExceptionOr<std::optional<ServiceWorkerClientData>>&&)>; |
| virtual void openWindow(ServiceWorkerIdentifier, const URL&, OpenWindowCallback&&) = 0; |
| |
| using NavigateCallback = CompletionHandler<void(ExceptionOr<std::optional<WebCore::ServiceWorkerClientData>>&&)>; |
| virtual void navigate(ScriptExecutionContextIdentifier, ServiceWorkerIdentifier, const URL&, NavigateCallback&&) = 0; |
| |
| virtual void didFailHeartBeatCheck(ServiceWorkerIdentifier) = 0; |
| virtual void setAsInspected(ServiceWorkerIdentifier, bool) = 0; |
| |
| virtual bool isThrottleable() const = 0; |
| virtual PageIdentifier pageIdentifier() const = 0; |
| |
| virtual void ref() const = 0; |
| virtual void deref() const = 0; |
| virtual void stop() = 0; |
| |
| bool isClosed() const { return m_isClosed; } |
| |
| protected: |
| void setAsClosed() { m_isClosed = true; } |
| |
| private: |
| bool m_isClosed { false }; |
| }; |
| |
| WEBCORE_EXPORT void setConnection(Ref<Connection>&&); |
| WEBCORE_EXPORT Connection* connection() const; |
| |
| WEBCORE_EXPORT void registerServiceWorkerThreadForInstall(Ref<ServiceWorkerThreadProxy>&&); |
| WEBCORE_EXPORT ServiceWorkerThreadProxy* serviceWorkerThreadProxy(ServiceWorkerIdentifier) const; |
| WEBCORE_EXPORT RefPtr<ServiceWorkerThreadProxy> serviceWorkerThreadProxyFromBackgroundThread(ServiceWorkerIdentifier) const; |
| WEBCORE_EXPORT void fireInstallEvent(ServiceWorkerIdentifier); |
| WEBCORE_EXPORT void fireActivateEvent(ServiceWorkerIdentifier); |
| WEBCORE_EXPORT void firePushEvent(ServiceWorkerIdentifier, std::optional<Vector<uint8_t>>&&, CompletionHandler<void(bool)>&&); |
| WEBCORE_EXPORT void firePushSubscriptionChangeEvent(ServiceWorkerIdentifier, std::optional<PushSubscriptionData>&& newSubscriptionData, std::optional<PushSubscriptionData>&& oldSubscriptionData); |
| WEBCORE_EXPORT void fireNotificationEvent(ServiceWorkerIdentifier, NotificationData&&, NotificationEventType, CompletionHandler<void(bool)>&&); |
| |
| WEBCORE_EXPORT void terminateWorker(ServiceWorkerIdentifier, Seconds timeout, Function<void()>&&); |
| |
| void forEachServiceWorker(const Function<Function<void(ScriptExecutionContext&)>()>&); |
| |
| WEBCORE_EXPORT bool postTaskToServiceWorker(ServiceWorkerIdentifier, Function<void(ServiceWorkerGlobalScope&)>&&); |
| |
| using ServiceWorkerCreationCallback = void(uint64_t); |
| void setServiceWorkerCreationCallback(ServiceWorkerCreationCallback* callback) { m_serviceWorkerCreationCallback = callback; } |
| |
| WEBCORE_EXPORT void stopAllServiceWorkers(); |
| |
| static constexpr Seconds workerTerminationTimeout { 10_s }; |
| static constexpr Seconds syncWorkerTerminationTimeout { 100_ms }; // Only used by layout tests. |
| |
| WEBCORE_EXPORT void setAsInspected(ServiceWorkerIdentifier, bool); |
| |
| private: |
| SWContextManager() = default; |
| |
| void startedServiceWorker(std::optional<ServiceWorkerJobDataIdentifier>, ServiceWorkerIdentifier, const String& exceptionMessage, bool doesHandleFetch); |
| NO_RETURN_DUE_TO_CRASH void serviceWorkerFailedToTerminate(ServiceWorkerIdentifier); |
| |
| void stopWorker(ServiceWorkerThreadProxy&, Seconds, Function<void()>&&); |
| |
| HashMap<ServiceWorkerIdentifier, Ref<ServiceWorkerThreadProxy>> m_workerMap WTF_GUARDED_BY_LOCK(m_workerMapLock); |
| mutable Lock m_workerMapLock; |
| RefPtr<Connection> m_connection; |
| ServiceWorkerCreationCallback* m_serviceWorkerCreationCallback { nullptr }; |
| |
| class ServiceWorkerTerminationRequest { |
| WTF_MAKE_FAST_ALLOCATED; |
| public: |
| ServiceWorkerTerminationRequest(SWContextManager&, ServiceWorkerIdentifier, Seconds timeout); |
| |
| private: |
| Timer m_timeoutTimer; |
| }; |
| HashMap<ServiceWorkerIdentifier, std::unique_ptr<ServiceWorkerTerminationRequest>> m_pendingServiceWorkerTerminationRequests; |
| }; |
| |
| } // namespace WebCore |
| |
| #endif // ENABLE(SERVICE_WORKER) |