/* -*- Mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 40 -*- */ /* vim: set ts=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_Promise_h #define mozilla_dom_Promise_h #include "mozilla/Attributes.h" #include "mozilla/ErrorResult.h" #include "mozilla/TypeTraits.h" #include "mozilla/dom/BindingDeclarations.h" #include "nsCycleCollectionParticipant.h" #include "mozilla/dom/PromiseBinding.h" #include "mozilla/dom/ToJSValue.h" #include "nsWrapperCache.h" #include "nsAutoPtr.h" #include "js/TypeDecls.h" #include "mozilla/dom/workers/bindings/WorkerFeature.h" class nsIGlobalObject; namespace mozilla { namespace dom { class AnyCallback; class DOMError; class PromiseCallback; class PromiseInit; class PromiseNativeHandler; class PromiseDebugging; class Promise; class PromiseReportRejectFeature : public workers::WorkerFeature { // The Promise that owns this feature. Promise* mPromise; public: PromiseReportRejectFeature(Promise* aPromise) : mPromise(aPromise) { MOZ_ASSERT(mPromise); } virtual bool Notify(JSContext* aCx, workers::Status aStatus) MOZ_OVERRIDE; }; class Promise MOZ_FINAL : public nsISupports, public nsWrapperCache { friend class NativePromiseCallback; friend class PromiseResolverMixin; friend class PromiseResolverTask; friend class PromiseTask; friend class PromiseReportRejectFeature; friend class PromiseWorkerProxy; friend class PromiseWorkerProxyRunnable; friend class RejectPromiseCallback; friend class ResolvePromiseCallback; friend class ThenableResolverMixin; friend class WorkerPromiseResolverTask; friend class WorkerPromiseTask; friend class WrapperPromiseCallback; ~Promise(); public: NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS(Promise) // Promise creation tries to create a JS reflector for the Promise, so is // fallible. Furthermore, we don't want to do JS-wrapping on a 0-refcount // object, so we addref before doing that and return the addrefed pointer // here. static already_AddRefed Create(nsIGlobalObject* aGlobal, ErrorResult& aRv); typedef void (Promise::*MaybeFunc)(JSContext* aCx, JS::Handle aValue); void MaybeResolve(JSContext* aCx, JS::Handle aValue); void MaybeReject(JSContext* aCx, JS::Handle aValue); // Helpers for using Promise from C++. // Most DOM objects are handled already. To add a new type T, add a // ToJSValue overload in ToJSValue.h. // aArg is a const reference so we can pass rvalues like integer constants template void MaybeResolve(const T& aArg) { MaybeSomething(aArg, &Promise::MaybeResolve); } inline void MaybeReject(nsresult aArg) { MOZ_ASSERT(NS_FAILED(aArg)); MaybeSomething(aArg, &Promise::MaybeReject); } // DO NOT USE MaybeRejectBrokenly with in new code. Promises should be // rejected with Error instances. // Note: MaybeRejectBrokenly is a template so we can use it with DOMError // without instantiating the DOMError specialization of MaybeSomething in // every translation unit that includes this header, because that would // require use to include DOMError.h either here or in all those translation // units. template void MaybeRejectBrokenly(const T& aArg); // Not implemented by default; see // specializations in the .cpp for // the T values we support. // WebIDL nsIGlobalObject* GetParentObject() const { return mGlobal; } virtual JSObject* WrapObject(JSContext* aCx) MOZ_OVERRIDE; static already_AddRefed Constructor(const GlobalObject& aGlobal, PromiseInit& aInit, ErrorResult& aRv); static already_AddRefed Resolve(const GlobalObject& aGlobal, JS::Handle aValue, ErrorResult& aRv); static already_AddRefed Resolve(nsIGlobalObject* aGlobal, JSContext* aCx, JS::Handle aValue, ErrorResult& aRv); static already_AddRefed Reject(const GlobalObject& aGlobal, JS::Handle aValue, ErrorResult& aRv); static already_AddRefed Reject(nsIGlobalObject* aGlobal, JSContext* aCx, JS::Handle aValue, ErrorResult& aRv); already_AddRefed Then(JSContext* aCx, AnyCallback* aResolveCallback, AnyCallback* aRejectCallback, ErrorResult& aRv); already_AddRefed Catch(JSContext* aCx, AnyCallback* aRejectCallback, ErrorResult& aRv); static already_AddRefed All(const GlobalObject& aGlobal, const Sequence& aIterable, ErrorResult& aRv); static already_AddRefed Race(const GlobalObject& aGlobal, const Sequence& aIterable, ErrorResult& aRv); void AppendNativeHandler(PromiseNativeHandler* aRunnable); private: // Do NOT call this unless you're Promise::Create. I wish we could enforce // that from inside this class too, somehow. Promise(nsIGlobalObject* aGlobal); friend class PromiseDebugging; enum PromiseState { Pending, Resolved, Rejected }; enum PromiseTaskSync { SyncTask, AsyncTask }; void SetState(PromiseState aState) { MOZ_ASSERT(mState == Pending); MOZ_ASSERT(aState != Pending); mState = aState; } void SetResult(JS::Handle aValue) { mResult = aValue; } // This method processes promise's resolve/reject callbacks with promise's // result. It's executed when the resolver.resolve() or resolver.reject() is // called or when the promise already has a result and new callbacks are // appended by then(), catch() or done(). void RunTask(); void RunResolveTask(JS::Handle aValue, Promise::PromiseState aState, PromiseTaskSync aAsynchronous); void AppendCallbacks(PromiseCallback* aResolveCallback, PromiseCallback* aRejectCallback); // If we have been rejected and our mResult is a JS exception, // report it to the error console. // Use MaybeReportRejectedOnce() for actual calls. void MaybeReportRejected(); void MaybeReportRejectedOnce() { MaybeReportRejected(); RemoveFeature(); mResult = JS::UndefinedValue(); } void MaybeResolveInternal(JSContext* aCx, JS::Handle aValue, PromiseTaskSync aSync = AsyncTask); void MaybeRejectInternal(JSContext* aCx, JS::Handle aValue, PromiseTaskSync aSync = AsyncTask); void ResolveInternal(JSContext* aCx, JS::Handle aValue, PromiseTaskSync aSync = AsyncTask); void RejectInternal(JSContext* aCx, JS::Handle aValue, PromiseTaskSync aSync = AsyncTask); template void MaybeSomething(T& aArgument, MaybeFunc aFunc) { ThreadsafeAutoJSContext cx; JSObject* wrapper = GetWrapper(); MOZ_ASSERT(wrapper); // We preserved it! JSAutoCompartment ac(cx, wrapper); JS::Rooted val(cx); if (!ToJSValue(cx, aArgument, &val)) { HandleException(cx); return; } (this->*aFunc)(cx, val); } // Static methods for the PromiseInit functions. static bool JSCallback(JSContext *aCx, unsigned aArgc, JS::Value *aVp); static bool ThenableResolverCommon(JSContext* aCx, uint32_t /* PromiseCallback::Task */ aTask, unsigned aArgc, JS::Value* aVp); static bool JSCallbackThenableResolver(JSContext *aCx, unsigned aArgc, JS::Value *aVp); static bool JSCallbackThenableRejecter(JSContext *aCx, unsigned aArgc, JS::Value *aVp); static JSObject* CreateFunction(JSContext* aCx, JSObject* aParent, Promise* aPromise, int32_t aTask); static JSObject* CreateThenableFunction(JSContext* aCx, Promise* aPromise, uint32_t aTask); void HandleException(JSContext* aCx); void RemoveFeature(); nsRefPtr mGlobal; nsTArray > mResolveCallbacks; nsTArray > mRejectCallbacks; JS::Heap mResult; PromiseState mState; bool mTaskPending; bool mHadRejectCallback; bool mResolvePending; // If a rejected promise on a worker has no reject callbacks attached, it // needs to know when the worker is shutting down, to report the error on the // console before the worker's context is deleted. This feature is used for // that purpose. nsAutoPtr mFeature; }; } // namespace dom } // namespace mozilla #endif // mozilla_dom_Promise_h