blob: a6de7d2e3930547dc8d8da5a290081a09f022dfe [file] [log] [blame]
/*
* 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.
*/
#include "config.h"
#include "SWContextManager.h"
#if ENABLE(SERVICE_WORKER)
#include "Logging.h"
#include "MessageWithMessagePorts.h"
#include "ServiceWorkerClientIdentifier.h"
#include "ServiceWorkerGlobalScope.h"
namespace WebCore {
SWContextManager& SWContextManager::singleton()
{
static SWContextManager* sharedManager = new SWContextManager;
return *sharedManager;
}
void SWContextManager::setConnection(std::unique_ptr<Connection>&& connection)
{
ASSERT(!m_connection || m_connection->isClosed());
m_connection = WTFMove(connection);
}
auto SWContextManager::connection() const -> Connection*
{
return m_connection.get();
}
void SWContextManager::registerServiceWorkerThreadForInstall(Ref<ServiceWorkerThreadProxy>&& serviceWorkerThreadProxy)
{
auto serviceWorkerIdentifier = serviceWorkerThreadProxy->identifier();
auto jobDataIdentifier = serviceWorkerThreadProxy->thread().contextData().jobDataIdentifier;
auto* threadProxy = serviceWorkerThreadProxy.ptr();
auto result = m_workerMap.add(serviceWorkerIdentifier, WTFMove(serviceWorkerThreadProxy));
ASSERT_UNUSED(result, result.isNewEntry);
threadProxy->thread().start([jobDataIdentifier, serviceWorkerIdentifier](const String& exceptionMessage, bool doesHandleFetch) {
SWContextManager::singleton().startedServiceWorker(jobDataIdentifier, serviceWorkerIdentifier, exceptionMessage, doesHandleFetch);
});
}
void SWContextManager::startedServiceWorker(Optional<ServiceWorkerJobDataIdentifier> jobDataIdentifier, ServiceWorkerIdentifier serviceWorkerIdentifier, const String& exceptionMessage, bool doesHandleFetch)
{
if (m_serviceWorkerCreationCallback)
m_serviceWorkerCreationCallback(serviceWorkerIdentifier.toUInt64());
if (!exceptionMessage.isEmpty()) {
connection()->serviceWorkerFailedToStart(jobDataIdentifier, serviceWorkerIdentifier, exceptionMessage);
return;
}
connection()->serviceWorkerStarted(jobDataIdentifier, serviceWorkerIdentifier, doesHandleFetch);
}
ServiceWorkerThreadProxy* SWContextManager::serviceWorkerThreadProxy(ServiceWorkerIdentifier identifier) const
{
return m_workerMap.get(identifier);
}
void SWContextManager::postMessageToServiceWorker(ServiceWorkerIdentifier destination, MessageWithMessagePorts&& message, ServiceWorkerOrClientData&& sourceData)
{
auto* serviceWorker = m_workerMap.get(destination);
ASSERT(serviceWorker);
ASSERT(!serviceWorker->isTerminatingOrTerminated());
// FIXME: We should pass valid MessagePortChannels.
serviceWorker->thread().postMessageToServiceWorker(WTFMove(message), WTFMove(sourceData));
}
void SWContextManager::fireInstallEvent(ServiceWorkerIdentifier identifier)
{
auto* serviceWorker = m_workerMap.get(identifier);
if (!serviceWorker)
return;
serviceWorker->thread().fireInstallEvent();
}
void SWContextManager::fireActivateEvent(ServiceWorkerIdentifier identifier)
{
auto* serviceWorker = m_workerMap.get(identifier);
if (!serviceWorker)
return;
serviceWorker->thread().fireActivateEvent();
}
void SWContextManager::softUpdate(ServiceWorkerIdentifier identifier)
{
auto* serviceWorker = m_workerMap.get(identifier);
if (!serviceWorker)
return;
serviceWorker->thread().softUpdate();
}
void SWContextManager::terminateWorker(ServiceWorkerIdentifier identifier, Seconds timeout, Function<void()>&& completionHandler)
{
auto serviceWorker = m_workerMap.take(identifier);
if (!serviceWorker) {
if (completionHandler)
completionHandler();
return;
}
serviceWorker->setAsTerminatingOrTerminated();
m_pendingServiceWorkerTerminationRequests.add(identifier, makeUnique<ServiceWorkerTerminationRequest>(*this, identifier, timeout));
auto& thread = serviceWorker->thread();
thread.stop([this, identifier, serviceWorker = WTFMove(serviceWorker), completionHandler = WTFMove(completionHandler)]() mutable {
m_pendingServiceWorkerTerminationRequests.remove(identifier);
if (auto* connection = SWContextManager::singleton().connection())
connection->workerTerminated(identifier);
if (completionHandler)
completionHandler();
// Spin the runloop before releasing the worker thread proxy, as there would otherwise be
// a race towards its destruction.
callOnMainThread([serviceWorker = WTFMove(serviceWorker)] { });
});
}
void SWContextManager::forEachServiceWorkerThread(const WTF::Function<void(ServiceWorkerThreadProxy&)>& apply)
{
for (auto& workerThread : m_workerMap.values())
apply(*workerThread);
}
bool SWContextManager::postTaskToServiceWorker(ServiceWorkerIdentifier identifier, WTF::Function<void(ServiceWorkerGlobalScope&)>&& task)
{
auto* serviceWorker = m_workerMap.get(identifier);
if (!serviceWorker)
return false;
serviceWorker->thread().runLoop().postTask([task = WTFMove(task)] (auto& context) {
task(downcast<ServiceWorkerGlobalScope>(context));
});
return true;
}
void SWContextManager::serviceWorkerFailedToTerminate(ServiceWorkerIdentifier serviceWorkerIdentifier)
{
UNUSED_PARAM(serviceWorkerIdentifier);
RELEASE_LOG_ERROR(ServiceWorker, "Failed to terminate service worker with identifier %s, killing the service worker process", serviceWorkerIdentifier.loggingString().utf8().data());
_exit(EXIT_FAILURE);
}
SWContextManager::ServiceWorkerTerminationRequest::ServiceWorkerTerminationRequest(SWContextManager& manager, ServiceWorkerIdentifier serviceWorkerIdentifier, Seconds timeout)
: m_timeoutTimer([&manager, serviceWorkerIdentifier] { manager.serviceWorkerFailedToTerminate(serviceWorkerIdentifier); })
{
m_timeoutTimer.startOneShot(timeout);
}
void SWContextManager::stopAllServiceWorkers()
{
auto serviceWorkers = WTFMove(m_workerMap);
for (auto& serviceWorker : serviceWorkers.values()) {
serviceWorker->setAsTerminatingOrTerminated();
m_pendingServiceWorkerTerminationRequests.add(serviceWorker->identifier(), makeUnique<ServiceWorkerTerminationRequest>(*this, serviceWorker->identifier(), workerTerminationTimeout));
serviceWorker->thread().stop([] { });
}
}
} // namespace WebCore
#endif