/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "ServiceWorkerContainer.h" #include "nsIDocument.h" #include "nsIServiceWorkerManager.h" #include "nsNetUtil.h" #include "nsPIDOMWindow.h" #include "mozilla/Preferences.h" #include "mozilla/Services.h" #include "nsCycleCollectionParticipant.h" #include "nsServiceManagerUtils.h" #include "mozilla/dom/Navigator.h" #include "mozilla/dom/Promise.h" #include "mozilla/dom/ServiceWorkerContainerBinding.h" #include "mozilla/dom/workers/bindings/ServiceWorker.h" #include "ServiceWorker.h" namespace mozilla { namespace dom { NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(ServiceWorkerContainer) NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper) NS_IMPL_ADDREF_INHERITED(ServiceWorkerContainer, DOMEventTargetHelper) NS_IMPL_RELEASE_INHERITED(ServiceWorkerContainer, DOMEventTargetHelper) NS_IMPL_CYCLE_COLLECTION_INHERITED(ServiceWorkerContainer, DOMEventTargetHelper, mControllerWorker, mReadyPromise) /* static */ bool ServiceWorkerContainer::IsEnabled(JSContext* aCx, JSObject* aGlobal) { MOZ_ASSERT(NS_IsMainThread()); JS::Rooted global(aCx, aGlobal); nsCOMPtr window = Navigator::GetWindowFromGlobal(global); if (!window) { return false; } nsIDocument* doc = window->GetExtantDoc(); if (!doc || nsContentUtils::IsInPrivateBrowsing(doc)) { return false; } return Preferences::GetBool("dom.serviceWorkers.enabled", false); } ServiceWorkerContainer::ServiceWorkerContainer(nsPIDOMWindow* aWindow) : DOMEventTargetHelper(aWindow) { } ServiceWorkerContainer::~ServiceWorkerContainer() { RemoveReadyPromise(); } void ServiceWorkerContainer::DisconnectFromOwner() { RemoveReadyPromise(); DOMEventTargetHelper::DisconnectFromOwner(); } void ServiceWorkerContainer::ControllerChanged(ErrorResult& aRv) { mControllerWorker = nullptr; aRv = DispatchTrustedEvent(NS_LITERAL_STRING("controllerchange")); } void ServiceWorkerContainer::RemoveReadyPromise() { nsCOMPtr window = GetOwner(); if (window) { nsCOMPtr swm = mozilla::services::GetServiceWorkerManager(); if (!swm) { // If the browser is shutting down, we don't need to remove the promise. return; } swm->RemoveReadyPromise(window); } } JSObject* ServiceWorkerContainer::WrapObject(JSContext* aCx, JS::Handle aGivenProto) { return ServiceWorkerContainerBinding::Wrap(aCx, this, aGivenProto); } already_AddRefed ServiceWorkerContainer::Register(const nsAString& aScriptURL, const RegistrationOptions& aOptions, ErrorResult& aRv) { nsCOMPtr promise; nsCOMPtr swm = mozilla::services::GetServiceWorkerManager(); if (!swm) { aRv.Throw(NS_ERROR_FAILURE); return nullptr; } nsCOMPtr window = GetOwner(); MOZ_ASSERT(window); nsresult rv; nsCOMPtr scriptURI; rv = NS_NewURI(getter_AddRefs(scriptURI), aScriptURL, nullptr, window->GetDocBaseURI()); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.ThrowTypeError(MSG_INVALID_URL, &aScriptURL); return nullptr; } // In ServiceWorkerContainer.register() the scope argument is parsed against // different base URLs depending on whether it was passed or not. nsCOMPtr scopeURI; // Step 4. If none passed, parse against script's URL if (!aOptions.mScope.WasPassed()) { nsresult rv = NS_NewURI(getter_AddRefs(scopeURI), NS_LITERAL_CSTRING("./"), nullptr, scriptURI); MOZ_ALWAYS_TRUE(NS_SUCCEEDED(rv)); } else { // Step 5. Parse against entry settings object's base URL. nsresult rv = NS_NewURI(getter_AddRefs(scopeURI), aOptions.mScope.Value(), nullptr, window->GetDocBaseURI()); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.ThrowTypeError(MSG_INVALID_URL, &aOptions.mScope.Value()); return nullptr; } } aRv = swm->Register(window, scopeURI, scriptURI, getter_AddRefs(promise)); if (aRv.Failed()) { return nullptr; } nsRefPtr ret = static_cast(promise.get()); MOZ_ASSERT(ret); return ret.forget(); } already_AddRefed ServiceWorkerContainer::GetController() { if (!mControllerWorker) { nsresult rv; nsCOMPtr swm = mozilla::services::GetServiceWorkerManager(); if (!swm) { return nullptr; } nsCOMPtr serviceWorker; rv = swm->GetDocumentController(GetOwner(), getter_AddRefs(serviceWorker)); if (NS_WARN_IF(NS_FAILED(rv))) { return nullptr; } mControllerWorker = static_cast(serviceWorker.get()); } nsRefPtr ref = mControllerWorker; return ref.forget(); } already_AddRefed ServiceWorkerContainer::GetRegistrations(ErrorResult& aRv) { nsresult rv; nsCOMPtr swm = do_GetService(SERVICEWORKERMANAGER_CONTRACTID, &rv); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(rv); return nullptr; } nsCOMPtr promise; aRv = swm->GetRegistrations(GetOwner(), getter_AddRefs(promise)); if (aRv.Failed()) { return nullptr; } nsRefPtr ret = static_cast(promise.get()); MOZ_ASSERT(ret); return ret.forget(); } already_AddRefed ServiceWorkerContainer::GetRegistration(const nsAString& aDocumentURL, ErrorResult& aRv) { nsresult rv; nsCOMPtr swm = do_GetService(SERVICEWORKERMANAGER_CONTRACTID, &rv); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(rv); return nullptr; } nsCOMPtr promise; aRv = swm->GetRegistration(GetOwner(), aDocumentURL, getter_AddRefs(promise)); if (aRv.Failed()) { return nullptr; } nsRefPtr ret = static_cast(promise.get()); MOZ_ASSERT(ret); return ret.forget(); } Promise* ServiceWorkerContainer::GetReady(ErrorResult& aRv) { if (mReadyPromise) { return mReadyPromise; } nsCOMPtr swm = mozilla::services::GetServiceWorkerManager(); if (!swm) { aRv.Throw(NS_ERROR_FAILURE); return nullptr; } nsCOMPtr promise; aRv = swm->GetReadyPromise(GetOwner(), getter_AddRefs(promise)); mReadyPromise = static_cast(promise.get()); return mReadyPromise; } // Testing only. void ServiceWorkerContainer::GetScopeForUrl(const nsAString& aUrl, nsString& aScope, ErrorResult& aRv) { nsCOMPtr swm = mozilla::services::GetServiceWorkerManager(); if (!swm) { aRv.Throw(NS_ERROR_FAILURE); return; } aRv = swm->GetScopeForUrl(GetOwner()->GetExtantDoc()->NodePrincipal(), aUrl, aScope); } } // namespace dom } // namespace mozilla