aboutsummaryrefslogtreecommitdiff
path: root/src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h
diff options
context:
space:
mode:
authorGravatar Jarred Sumner <jarred@jarredsumner.com> 2022-04-09 19:53:17 -0700
committerGravatar Jarred Sumner <jarred@jarredsumner.com> 2022-04-09 19:53:17 -0700
commitc5d637fbb15874caa0d9669ac8cc79912ed4d5d9 (patch)
treef20a9b0047ba763b0003a60032d6c4ae1f9dcb47 /src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h
parentfb82e2bf86a13211f08b407c3ec80fc0db45a77c (diff)
downloadbun-jarred/workers.tar.gz
bun-jarred/workers.tar.zst
bun-jarred/workers.zip
Diffstat (limited to 'src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h')
-rw-r--r--src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h350
1 files changed, 350 insertions, 0 deletions
diff --git a/src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h b/src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h
new file mode 100644
index 000000000..2b8706e76
--- /dev/null
+++ b/src/javascript/jsc/bindings/webcore/DOMPromiseProxy.h
@@ -0,0 +1,350 @@
+/*
+ * Copyright (C) 2017-2021 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.
+ */
+
+#pragma once
+
+#include "ExceptionOr.h"
+#include "JSDOMGlobalObject.h"
+#include "JSDOMPromiseDeferred.h"
+#include <wtf/Function.h>
+#include <wtf/Vector.h>
+
+namespace WebCore {
+
+template<typename IDLType>
+class DOMPromiseProxy {
+ WTF_MAKE_FAST_ALLOCATED;
+public:
+ using Value = typename IDLType::StorageType;
+
+ DOMPromiseProxy() = default;
+ ~DOMPromiseProxy() = default;
+
+ JSC::JSValue promise(JSC::JSGlobalObject&, JSDOMGlobalObject&);
+
+ void clear();
+
+ bool isFulfilled() const;
+
+ void resolve(typename IDLType::StorageType);
+ void resolveWithNewlyCreated(typename IDLType::StorageType);
+ void reject(Exception, RejectAsHandled = RejectAsHandled::No);
+
+private:
+ JSC::JSValue resolvePromise(JSC::JSGlobalObject&, JSDOMGlobalObject&, const Function<void(DeferredPromise&)>&);
+
+ std::optional<ExceptionOr<Value>> m_valueOrException;
+ Vector<Ref<DeferredPromise>, 1> m_deferredPromises;
+};
+
+template<>
+class DOMPromiseProxy<IDLUndefined> {
+ WTF_MAKE_FAST_ALLOCATED;
+public:
+ DOMPromiseProxy() = default;
+ ~DOMPromiseProxy() = default;
+
+ JSC::JSValue promise(JSC::JSGlobalObject&, JSDOMGlobalObject&);
+
+ void clear();
+
+ bool isFulfilled() const;
+
+ void resolve();
+ void reject(Exception, RejectAsHandled = RejectAsHandled::No);
+
+private:
+ std::optional<ExceptionOr<void>> m_valueOrException;
+ Vector<Ref<DeferredPromise>, 1> m_deferredPromises;
+};
+
+// Instead of storing the value of the resolution directly, DOMPromiseProxyWithResolveCallback
+// allows the owner to specify callback to be called when the resolved value is needed. This is
+// needed to avoid reference cycles when the resolved value is the owner, such as is the case with
+// FontFace and FontFaceSet.
+template<typename IDLType>
+class DOMPromiseProxyWithResolveCallback {
+ WTF_MAKE_FAST_ALLOCATED;
+public:
+ using ResolveCallback = Function<typename IDLType::ParameterType()>;
+
+ template <typename Class, typename BaseClass>
+ DOMPromiseProxyWithResolveCallback(Class&, typename IDLType::ParameterType (BaseClass::*)());
+ DOMPromiseProxyWithResolveCallback(ResolveCallback&&);
+ ~DOMPromiseProxyWithResolveCallback() = default;
+
+ JSC::JSValue promise(JSC::JSGlobalObject&, JSDOMGlobalObject&);
+
+ void clear();
+
+ bool isFulfilled() const;
+
+ void resolve(typename IDLType::ParameterType);
+ void resolveWithNewlyCreated(typename IDLType::ParameterType);
+ void reject(Exception, RejectAsHandled = RejectAsHandled::No);
+
+private:
+ ResolveCallback m_resolveCallback;
+ std::optional<ExceptionOr<void>> m_valueOrException;
+ Vector<Ref<DeferredPromise>, 1> m_deferredPromises;
+};
+
+// MARK: - DOMPromiseProxy<IDLType> generic implementation
+
+template<typename IDLType>
+inline JSC::JSValue DOMPromiseProxy<IDLType>::resolvePromise(JSC::JSGlobalObject& lexicalGlobalObject, JSDOMGlobalObject& globalObject, const Function<void(DeferredPromise&)>& resolvePromiseCallback)
+{
+ UNUSED_PARAM(lexicalGlobalObject);
+ for (auto& deferredPromise : m_deferredPromises) {
+ if (deferredPromise->globalObject() == &globalObject)
+ return deferredPromise->promise();
+ }
+
+ // DeferredPromise can fail construction during worker abrupt termination.
+ auto deferredPromise = DeferredPromise::create(globalObject, DeferredPromise::Mode::RetainPromiseOnResolve);
+ if (!deferredPromise)
+ return JSC::jsUndefined();
+
+ if (m_valueOrException) {
+ if (m_valueOrException->hasException())
+ deferredPromise->reject(m_valueOrException->exception());
+ else
+ resolvePromiseCallback(*deferredPromise);
+ }
+
+ auto result = deferredPromise->promise();
+ m_deferredPromises.append(deferredPromise.releaseNonNull());
+ return result;
+}
+
+template<typename IDLType>
+inline JSC::JSValue DOMPromiseProxy<IDLType>::promise(JSC::JSGlobalObject& lexicalGlobalObject, JSDOMGlobalObject& globalObject)
+{
+ return resolvePromise(lexicalGlobalObject, globalObject, [this](auto& deferredPromise) {
+ deferredPromise.template resolve<IDLType>(m_valueOrException->returnValue());
+ });
+}
+
+template<>
+inline JSC::JSValue DOMPromiseProxy<IDLAny>::promise(JSC::JSGlobalObject& lexicalGlobalObject, JSDOMGlobalObject& globalObject)
+{
+ return resolvePromise(lexicalGlobalObject, globalObject, [this](auto& deferredPromise) {
+ deferredPromise.resolveWithJSValue(m_valueOrException->returnValue().get());
+ });
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxy<IDLType>::clear()
+{
+ m_valueOrException = std::nullopt;
+ m_deferredPromises.clear();
+}
+
+template<typename IDLType>
+inline bool DOMPromiseProxy<IDLType>::isFulfilled() const
+{
+ return m_valueOrException.has_value();
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxy<IDLType>::resolve(typename IDLType::StorageType value)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<Value> { std::forward<typename IDLType::StorageType>(value) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->template resolve<IDLType>(m_valueOrException->returnValue());
+}
+
+template<>
+inline void DOMPromiseProxy<IDLAny>::resolve(typename IDLAny::StorageType value)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<Value> { std::forward<typename IDLAny::StorageType>(value) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->resolveWithJSValue(m_valueOrException->returnValue().get());
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxy<IDLType>::resolveWithNewlyCreated(typename IDLType::StorageType value)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<Value> { std::forward<typename IDLType::StorageType>(value) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->template resolveWithNewlyCreated<IDLType>(m_valueOrException->returnValue());
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxy<IDLType>::reject(Exception exception, RejectAsHandled rejectAsHandled)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<Value> { WTFMove(exception) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->reject(m_valueOrException->exception(), rejectAsHandled);
+}
+
+
+// MARK: - DOMPromiseProxy<IDLUndefined> specialization
+
+inline JSC::JSValue DOMPromiseProxy<IDLUndefined>::promise(JSC::JSGlobalObject& lexicalGlobalObject, JSDOMGlobalObject& globalObject)
+{
+ UNUSED_PARAM(lexicalGlobalObject);
+ for (auto& deferredPromise : m_deferredPromises) {
+ if (deferredPromise->globalObject() == &globalObject)
+ return deferredPromise->promise();
+ }
+
+ // DeferredPromise can fail construction during worker abrupt termination.
+ auto deferredPromise = DeferredPromise::create(globalObject, DeferredPromise::Mode::RetainPromiseOnResolve);
+ if (!deferredPromise)
+ return JSC::jsUndefined();
+
+ if (m_valueOrException) {
+ if (m_valueOrException->hasException())
+ deferredPromise->reject(m_valueOrException->exception());
+ else
+ deferredPromise->resolve();
+ }
+
+ auto result = deferredPromise->promise();
+ m_deferredPromises.append(deferredPromise.releaseNonNull());
+ return result;
+}
+
+inline void DOMPromiseProxy<IDLUndefined>::clear()
+{
+ m_valueOrException = std::nullopt;
+ m_deferredPromises.clear();
+}
+
+inline bool DOMPromiseProxy<IDLUndefined>::isFulfilled() const
+{
+ return m_valueOrException.has_value();
+}
+
+inline void DOMPromiseProxy<IDLUndefined>::resolve()
+{
+ ASSERT(!m_valueOrException);
+ m_valueOrException = ExceptionOr<void> { };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->resolve();
+}
+
+inline void DOMPromiseProxy<IDLUndefined>::reject(Exception exception, RejectAsHandled rejectAsHandled)
+{
+ ASSERT(!m_valueOrException);
+ m_valueOrException = ExceptionOr<void> { WTFMove(exception) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->reject(m_valueOrException->exception(), rejectAsHandled);
+}
+
+// MARK: - DOMPromiseProxyWithResolveCallback<IDLType> implementation
+
+template<typename IDLType>
+template <typename Class, typename BaseClass>
+inline DOMPromiseProxyWithResolveCallback<IDLType>::DOMPromiseProxyWithResolveCallback(Class& object, typename IDLType::ParameterType (BaseClass::*function)())
+ : m_resolveCallback(std::bind(function, &object))
+{
+}
+
+template<typename IDLType>
+inline DOMPromiseProxyWithResolveCallback<IDLType>::DOMPromiseProxyWithResolveCallback(ResolveCallback&& function)
+ : m_resolveCallback(WTFMove(function))
+{
+}
+
+template<typename IDLType>
+inline JSC::JSValue DOMPromiseProxyWithResolveCallback<IDLType>::promise(JSC::JSGlobalObject& lexicalGlobalObject, JSDOMGlobalObject& globalObject)
+{
+ UNUSED_PARAM(lexicalGlobalObject);
+ for (auto& deferredPromise : m_deferredPromises) {
+ if (deferredPromise->globalObject() == &globalObject)
+ return deferredPromise->promise();
+ }
+
+ // DeferredPromise can fail construction during worker abrupt termination.
+ auto deferredPromise = DeferredPromise::create(globalObject, DeferredPromise::Mode::RetainPromiseOnResolve);
+ if (!deferredPromise)
+ return JSC::jsUndefined();
+
+ if (m_valueOrException) {
+ if (m_valueOrException->hasException())
+ deferredPromise->reject(m_valueOrException->exception());
+ else
+ deferredPromise->template resolve<IDLType>(m_resolveCallback());
+ }
+
+ auto result = deferredPromise->promise();
+ m_deferredPromises.append(deferredPromise.releaseNonNull());
+ return result;
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxyWithResolveCallback<IDLType>::clear()
+{
+ m_valueOrException = std::nullopt;
+ m_deferredPromises.clear();
+}
+
+template<typename IDLType>
+inline bool DOMPromiseProxyWithResolveCallback<IDLType>::isFulfilled() const
+{
+ return m_valueOrException.has_value();
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxyWithResolveCallback<IDLType>::resolve(typename IDLType::ParameterType value)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<void> { };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->template resolve<IDLType>(value);
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxyWithResolveCallback<IDLType>::resolveWithNewlyCreated(typename IDLType::ParameterType value)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<void> { };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->template resolveWithNewlyCreated<IDLType>(value);
+}
+
+template<typename IDLType>
+inline void DOMPromiseProxyWithResolveCallback<IDLType>::reject(Exception exception, RejectAsHandled rejectAsHandled)
+{
+ ASSERT(!m_valueOrException);
+
+ m_valueOrException = ExceptionOr<void> { WTFMove(exception) };
+ for (auto& deferredPromise : m_deferredPromises)
+ deferredPromise->reject(m_valueOrException->exception(), rejectAsHandled);
+}
+
+}