/* -*- 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/. */ #ifndef mozilla_dom_serviceworkercontainer_h__ #define mozilla_dom_serviceworkercontainer_h__ #include "mozilla/DOMEventTargetHelper.h" #include "mozilla/ErrorResult.h" #include "mozilla/dom/ServiceWorkerUtils.h" class nsIGlobalWindow; namespace mozilla { namespace dom { class ClientPostMessageArgs; struct MessageEventInit; class Promise; struct RegistrationOptions; class ServiceWorker; // Lightweight serviceWorker APIs collection. class ServiceWorkerContainer final : public DOMEventTargetHelper { public: class Inner { public: virtual void AddContainer(ServiceWorkerContainer* aOuter) = 0; virtual void RemoveContainer(ServiceWorkerContainer* aOuter) = 0; virtual void Register(const ClientInfo& aClientInfo, const nsACString& aScopeURL, const nsACString& aScriptURL, ServiceWorkerUpdateViaCache aUpdateViaCache, ServiceWorkerRegistrationCallback&& aSuccessCB, ServiceWorkerFailureCallback&& aFailureCB) const = 0; virtual void GetRegistration( const ClientInfo& aClientInfo, const nsACString& aURL, ServiceWorkerRegistrationCallback&& aSuccessCB, ServiceWorkerFailureCallback&& aFailureCB) const = 0; virtual void GetRegistrations( const ClientInfo& aClientInfo, ServiceWorkerRegistrationListCallback&& aSuccessCB, ServiceWorkerFailureCallback&& aFailureCB) const = 0; virtual void GetReady(const ClientInfo& aClientInfo, ServiceWorkerRegistrationCallback&& aSuccessCB, ServiceWorkerFailureCallback&& aFailureCB) const = 0; NS_INLINE_DECL_PURE_VIRTUAL_REFCOUNTING }; NS_DECL_ISUPPORTS_INHERITED NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(ServiceWorkerContainer, DOMEventTargetHelper) IMPL_EVENT_HANDLER(controllerchange) IMPL_EVENT_HANDLER(error) IMPL_EVENT_HANDLER(messageerror) // Almost a manual expansion of IMPL_EVENT_HANDLER(message), but // with the additional StartMessages() when setting the handler, as // required by the spec. inline mozilla::dom::EventHandlerNonNull* GetOnmessage() { return GetEventHandler(nsGkAtoms::onmessage); } inline void SetOnmessage(mozilla::dom::EventHandlerNonNull* aCallback) { SetEventHandler(nsGkAtoms::onmessage, aCallback); StartMessages(); } static bool IsEnabled(JSContext* aCx, JSObject* aGlobal); static already_AddRefed Create( nsIGlobalObject* aGlobal); virtual JSObject* WrapObject(JSContext* aCx, JS::Handle aGivenProto) override; already_AddRefed Register(const nsAString& aScriptURL, const RegistrationOptions& aOptions, const CallerType aCallerType, ErrorResult& aRv); already_AddRefed GetController(); already_AddRefed GetRegistration(const nsAString& aDocumentURL, ErrorResult& aRv); already_AddRefed GetRegistrations(ErrorResult& aRv); void StartMessages(); Promise* GetReady(ErrorResult& aRv); // Testing only. void GetScopeForUrl(const nsAString& aUrl, nsString& aScope, ErrorResult& aRv); // DOMEventTargetHelper void DisconnectFromOwner() override; // Invalidates |mControllerWorker| and dispatches a "controllerchange" // event. void ControllerChanged(ErrorResult& aRv); void ReceiveMessage(const ClientPostMessageArgs& aArgs); private: ServiceWorkerContainer( nsIGlobalObject* aGlobal, already_AddRefed aInner); ~ServiceWorkerContainer(); // Utility method to get the global if its present and if certain // additional validaty checks pass. One of these additional checks // verifies the global can access storage. Since storage access can // vary based on user settings we want to often provide some error // message if the storage check fails. This method takes an optional // callback that can be used to report the storage failure to the // devtools console. nsIGlobalObject* GetGlobalIfValid( ErrorResult& aRv, const std::function&& aStorageFailureCB = nullptr) const; struct ReceivedMessage; // Dispatch a Runnable that dispatches the given message on this // object. When the owner of this object is a Window, the Runnable // is dispatched on the corresponding TabGroup. void EnqueueReceivedMessageDispatch(RefPtr aMessage); template void RunWithJSContext(F&& aCallable); void DispatchMessage(RefPtr aMessage); // When it fails, returning boolean means whether it's because deserailization // failed or not. static Result FillInMessageEventInit(JSContext* aCx, nsIGlobalObject* aGlobal, ReceivedMessage& aMessage, MessageEventInit& aInit, ErrorResult& aRv); RefPtr mInner; // This only changes when a worker hijacks everything in its scope by calling // claim. RefPtr mControllerWorker; RefPtr mReadyPromise; MozPromiseRequestHolder mReadyPromiseHolder; // Set after StartMessages() has been called. bool mMessagesStarted = false; // Queue holding messages posted from service worker as long as // StartMessages() hasn't been called. nsTArray> mPendingMessages; }; } // namespace dom } // namespace mozilla #endif /* mozilla_dom_serviceworkercontainer_h__ */