123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341 |
- /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
- /* 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 "nsContentUtils.h"
- #include "nsIDocument.h"
- #include "nsIServiceWorkerManager.h"
- #include "nsIURL.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<JSObject*> global(aCx, aGlobal);
- nsCOMPtr<nsPIDOMWindowInner> 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(nsPIDOMWindowInner* aWindow)
- : DOMEventTargetHelper(aWindow)
- {
- }
- ServiceWorkerContainer::~ServiceWorkerContainer()
- {
- RemoveReadyPromise();
- }
- void
- ServiceWorkerContainer::DisconnectFromOwner()
- {
- mControllerWorker = nullptr;
- RemoveReadyPromise();
- DOMEventTargetHelper::DisconnectFromOwner();
- }
- void
- ServiceWorkerContainer::ControllerChanged(ErrorResult& aRv)
- {
- mControllerWorker = nullptr;
- aRv = DispatchTrustedEvent(NS_LITERAL_STRING("controllerchange"));
- }
- void
- ServiceWorkerContainer::RemoveReadyPromise()
- {
- if (nsCOMPtr<nsPIDOMWindowInner> window = GetOwner()) {
- nsCOMPtr<nsIServiceWorkerManager> 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<JSObject*> aGivenProto)
- {
- return ServiceWorkerContainerBinding::Wrap(aCx, this, aGivenProto);
- }
- static nsresult
- CheckForSlashEscapedCharsInPath(nsIURI* aURI)
- {
- MOZ_ASSERT(aURI);
- // A URL that can't be downcast to a standard URL is an invalid URL and should
- // be treated as such and fail with SecurityError.
- nsCOMPtr<nsIURL> url(do_QueryInterface(aURI));
- if (NS_WARN_IF(!url)) {
- return NS_ERROR_DOM_SECURITY_ERR;
- }
- nsAutoCString path;
- nsresult rv = url->GetFilePath(path);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- return rv;
- }
- ToLowerCase(path);
- if (path.Find("%2f") != kNotFound ||
- path.Find("%5c") != kNotFound) {
- return NS_ERROR_DOM_TYPE_ERR;
- }
- return NS_OK;
- }
- already_AddRefed<Promise>
- ServiceWorkerContainer::Register(const nsAString& aScriptURL,
- const RegistrationOptions& aOptions,
- ErrorResult& aRv)
- {
- nsCOMPtr<nsISupports> promise;
- nsCOMPtr<nsIServiceWorkerManager> swm = mozilla::services::GetServiceWorkerManager();
- if (!swm) {
- aRv.Throw(NS_ERROR_FAILURE);
- return nullptr;
- }
- nsCOMPtr<nsIURI> baseURI;
- nsIDocument* doc = GetEntryDocument();
- if (doc) {
- baseURI = doc->GetBaseURI();
- } else {
- // XXXnsm. One of our devtools browser test calls register() from a content
- // script where there is no valid entry document. Use the window to resolve
- // the uri in that case.
- nsCOMPtr<nsPIDOMWindowInner> window = GetOwner();
- nsCOMPtr<nsPIDOMWindowOuter> outerWindow;
- if (window && (outerWindow = window->GetOuterWindow()) &&
- outerWindow->GetServiceWorkersTestingEnabled()) {
- baseURI = window->GetDocBaseURI();
- }
- }
- nsresult rv;
- nsCOMPtr<nsIURI> scriptURI;
- rv = NS_NewURI(getter_AddRefs(scriptURI), aScriptURL, nullptr, baseURI);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- aRv.ThrowTypeError<MSG_INVALID_URL>(aScriptURL);
- return nullptr;
- }
- aRv = CheckForSlashEscapedCharsInPath(scriptURI);
- if (NS_WARN_IF(aRv.Failed())) {
- return nullptr;
- }
- // In ServiceWorkerContainer.register() the scope argument is parsed against
- // different base URLs depending on whether it was passed or not.
- nsCOMPtr<nsIURI> scopeURI;
- // Step 4. If none passed, parse against script's URL
- if (!aOptions.mScope.WasPassed()) {
- NS_NAMED_LITERAL_STRING(defaultScope, "./");
- rv = NS_NewURI(getter_AddRefs(scopeURI), defaultScope,
- nullptr, scriptURI);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- nsAutoCString spec;
- scriptURI->GetSpec(spec);
- NS_ConvertUTF8toUTF16 wSpec(spec);
- aRv.ThrowTypeError<MSG_INVALID_SCOPE>(defaultScope, wSpec);
- return nullptr;
- }
- } else {
- // Step 5. Parse against entry settings object's base URL.
- rv = NS_NewURI(getter_AddRefs(scopeURI), aOptions.mScope.Value(),
- nullptr, baseURI);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- nsIURI* uri = baseURI ? baseURI : scriptURI;
- nsAutoCString spec;
- uri->GetSpec(spec);
- NS_ConvertUTF8toUTF16 wSpec(spec);
- aRv.ThrowTypeError<MSG_INVALID_SCOPE>(aOptions.mScope.Value(), wSpec);
- return nullptr;
- }
- aRv = CheckForSlashEscapedCharsInPath(scopeURI);
- if (NS_WARN_IF(aRv.Failed())) {
- return nullptr;
- }
- }
- // The spec says that the "client" passed to Register() must be the global
- // where the ServiceWorkerContainer was retrieved from.
- aRv = swm->Register(GetOwner(), scopeURI, scriptURI, getter_AddRefs(promise));
- if (NS_WARN_IF(aRv.Failed())) {
- return nullptr;
- }
- RefPtr<Promise> ret = static_cast<Promise*>(promise.get());
- MOZ_ASSERT(ret);
- return ret.forget();
- }
- already_AddRefed<workers::ServiceWorker>
- ServiceWorkerContainer::GetController()
- {
- if (!mControllerWorker) {
- nsCOMPtr<nsIServiceWorkerManager> swm = mozilla::services::GetServiceWorkerManager();
- if (!swm) {
- return nullptr;
- }
- // TODO: What should we do here if the ServiceWorker script fails to load?
- // In theory the DOM ServiceWorker object can exist without the worker
- // thread running, but it seems our design does not expect that.
- nsCOMPtr<nsISupports> serviceWorker;
- nsresult rv = swm->GetDocumentController(GetOwner(),
- getter_AddRefs(serviceWorker));
- if (NS_WARN_IF(NS_FAILED(rv))) {
- return nullptr;
- }
- mControllerWorker =
- static_cast<workers::ServiceWorker*>(serviceWorker.get());
- }
- RefPtr<workers::ServiceWorker> ref = mControllerWorker;
- return ref.forget();
- }
- already_AddRefed<Promise>
- ServiceWorkerContainer::GetRegistrations(ErrorResult& aRv)
- {
- nsresult rv;
- nsCOMPtr<nsIServiceWorkerManager> swm = do_GetService(SERVICEWORKERMANAGER_CONTRACTID, &rv);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- aRv.Throw(rv);
- return nullptr;
- }
- nsCOMPtr<nsISupports> promise;
- aRv = swm->GetRegistrations(GetOwner(), getter_AddRefs(promise));
- if (aRv.Failed()) {
- return nullptr;
- }
- RefPtr<Promise> ret = static_cast<Promise*>(promise.get());
- MOZ_ASSERT(ret);
- return ret.forget();
- }
- already_AddRefed<Promise>
- ServiceWorkerContainer::GetRegistration(const nsAString& aDocumentURL,
- ErrorResult& aRv)
- {
- nsresult rv;
- nsCOMPtr<nsIServiceWorkerManager> swm = do_GetService(SERVICEWORKERMANAGER_CONTRACTID, &rv);
- if (NS_WARN_IF(NS_FAILED(rv))) {
- aRv.Throw(rv);
- return nullptr;
- }
- nsCOMPtr<nsISupports> promise;
- aRv = swm->GetRegistration(GetOwner(), aDocumentURL, getter_AddRefs(promise));
- if (aRv.Failed()) {
- return nullptr;
- }
- RefPtr<Promise> ret = static_cast<Promise*>(promise.get());
- MOZ_ASSERT(ret);
- return ret.forget();
- }
- Promise*
- ServiceWorkerContainer::GetReady(ErrorResult& aRv)
- {
- if (mReadyPromise) {
- return mReadyPromise;
- }
- nsCOMPtr<nsIServiceWorkerManager> swm = mozilla::services::GetServiceWorkerManager();
- if (!swm) {
- aRv.Throw(NS_ERROR_FAILURE);
- return nullptr;
- }
- nsCOMPtr<nsISupports> promise;
- aRv = swm->GetReadyPromise(GetOwner(), getter_AddRefs(promise));
- mReadyPromise = static_cast<Promise*>(promise.get());
- return mReadyPromise;
- }
- // Testing only.
- void
- ServiceWorkerContainer::GetScopeForUrl(const nsAString& aUrl,
- nsString& aScope,
- ErrorResult& aRv)
- {
- nsCOMPtr<nsIServiceWorkerManager> swm = mozilla::services::GetServiceWorkerManager();
- if (!swm) {
- aRv.Throw(NS_ERROR_FAILURE);
- return;
- }
- nsCOMPtr<nsPIDOMWindowInner> window = GetOwner();
- if (NS_WARN_IF(!window)) {
- aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
- return;
- }
- nsCOMPtr<nsIDocument> doc = window->GetExtantDoc();
- if (NS_WARN_IF(!doc)) {
- aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
- return;
- }
- aRv = swm->GetScopeForUrl(doc->NodePrincipal(),
- aUrl, aScope);
- }
- } // namespace dom
- } // namespace mozilla
|