summaryrefslogtreecommitdiffstats
path: root/dom/indexedDB/IDBResult.h
diff options
context:
space:
mode:
Diffstat (limited to 'dom/indexedDB/IDBResult.h')
-rw-r--r--dom/indexedDB/IDBResult.h158
1 files changed, 158 insertions, 0 deletions
diff --git a/dom/indexedDB/IDBResult.h b/dom/indexedDB/IDBResult.h
new file mode 100644
index 0000000000..424eda9bff
--- /dev/null
+++ b/dom/indexedDB/IDBResult.h
@@ -0,0 +1,158 @@
+/* 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_indexeddb_idbresult_h__
+#define mozilla_dom_indexeddb_idbresult_h__
+
+#include "mozilla/ErrorResult.h"
+#include "mozilla/ResultVariant.h"
+#include "mozilla/Variant.h"
+
+#include <type_traits>
+#include <utility>
+
+namespace mozilla::dom::indexedDB {
+
+// IDBSpecialValue represents two special return values, distinct from any other
+// value, used in several places in the IndexedDB spec.
+enum class IDBSpecialValue {
+ Failure,
+ Invalid,
+};
+
+namespace detail {
+
+template <IDBSpecialValue Value>
+using SpecialConstant = std::integral_constant<IDBSpecialValue, Value>;
+using FailureType = SpecialConstant<IDBSpecialValue::Failure>;
+using InvalidType = SpecialConstant<IDBSpecialValue::Invalid>;
+struct ExceptionType final {};
+} // namespace detail
+
+// Put these in a subnamespace to avoid conflicts from the combination of 1.
+// using namespace mozilla::dom::indexedDB; in cpp files, 2. the unified build
+// and 3. mozilla::dom::Exception
+namespace SpecialValues {
+constexpr const detail::FailureType Failure;
+constexpr const detail::InvalidType Invalid;
+constexpr const detail::ExceptionType Exception;
+} // namespace SpecialValues
+
+namespace detail {
+template <IDBSpecialValue... Elements>
+struct IsSortedSet;
+
+template <IDBSpecialValue First, IDBSpecialValue Second,
+ IDBSpecialValue... Rest>
+struct IsSortedSet<First, Second, Rest...>
+ : std::integral_constant<bool, IsSortedSet<First, Second>::value &&
+ IsSortedSet<Second, Rest...>::value> {};
+
+template <IDBSpecialValue First, IDBSpecialValue Second>
+struct IsSortedSet<First, Second>
+ : std::integral_constant<bool, (First < Second)> {};
+
+template <IDBSpecialValue First>
+struct IsSortedSet<First> : std::true_type {};
+
+template <>
+struct IsSortedSet<> : std::true_type {};
+
+template <IDBSpecialValue... S>
+class IDBError {
+ // This assertion ensures that permutations of the set of possible special
+ // values don't create distinct types.
+ static_assert(IsSortedSet<S...>::value,
+ "special value list must be sorted and unique");
+
+ template <IDBSpecialValue... U>
+ friend class IDBError;
+
+ public:
+ MOZ_IMPLICIT IDBError(nsresult aRv) : mVariant(ErrorResult{aRv}) {}
+
+ IDBError(ExceptionType, ErrorResult&& aErrorResult)
+ : mVariant(std::move(aErrorResult)) {}
+
+ template <IDBSpecialValue Special>
+ MOZ_IMPLICIT IDBError(SpecialConstant<Special>)
+ : mVariant(SpecialConstant<Special>{}) {}
+
+ IDBError(IDBError&&) = default;
+ IDBError& operator=(IDBError&&) = default;
+
+ // Construct an IDBResult from another IDBResult whose set of possible special
+ // values is a subset of this one's.
+ template <IDBSpecialValue... U>
+ MOZ_IMPLICIT IDBError(IDBError<U...>&& aOther)
+ : mVariant(aOther.mVariant.match(
+ [](auto& aVariant) { return VariantType{std::move(aVariant)}; })) {}
+
+ bool Is(ExceptionType) const { return mVariant.template is<ErrorResult>(); }
+
+ template <IDBSpecialValue Special>
+ bool Is(SpecialConstant<Special>) const {
+ return mVariant.template is<SpecialConstant<Special>>();
+ }
+
+ ErrorResult& AsException() { return mVariant.template as<ErrorResult>(); }
+
+ template <typename... SpecialValueMappers>
+ ErrorResult ExtractErrorResult(SpecialValueMappers... aSpecialValueMappers) {
+#if defined(__clang__) || (defined(__GNUC__) && __GNUC__ >= 8)
+ return mVariant.match(
+ [](ErrorResult& aException) { return std::move(aException); },
+ [aSpecialValueMappers](const SpecialConstant<S>& aSpecialValue) {
+ return ErrorResult{aSpecialValueMappers(aSpecialValue)};
+ }...);
+#else
+ // gcc 7 doesn't accept the kind of parameter pack expansion above,
+ // probably due to https://gcc.gnu.org/bugzilla/show_bug.cgi?id=47226
+ return mVariant.match([aSpecialValueMappers...](auto& aValue) {
+ if constexpr (std::is_same_v<ErrorResult&, decltype(aValue)>) {
+ return std::move(aValue);
+ } else {
+ return ErrorResult{aSpecialValueMappers(aValue)...};
+ }
+ });
+#endif
+ }
+
+ template <typename... SpecialValueMappers>
+ nsresult ExtractNSResult(SpecialValueMappers... aSpecialValueMappers) {
+ return mVariant.match(
+ [](ErrorResult& aException) { return aException.StealNSResult(); },
+ aSpecialValueMappers...);
+ }
+
+ protected:
+ using VariantType = Variant<ErrorResult, SpecialConstant<S>...>;
+
+ VariantType mVariant;
+};
+} // namespace detail
+
+// Represents a return value of an IndexedDB algorithm. T is the type of the
+// regular return value, while S is a list of special values that can be
+// returned by the particular algorithm.
+template <typename T, IDBSpecialValue... S>
+using IDBResult = Result<T, detail::IDBError<S...>>;
+
+template <nsresult E>
+nsresult InvalidMapsTo(const indexedDB::detail::InvalidType&) {
+ return E;
+}
+
+inline detail::IDBError<> IDBException(nsresult aRv) {
+ return {SpecialValues::Exception, ErrorResult{aRv}};
+}
+
+template <IDBSpecialValue Special>
+detail::IDBError<Special> IDBError(detail::SpecialConstant<Special> aResult) {
+ return {aResult};
+}
+
+} // namespace mozilla::dom::indexedDB
+
+#endif // mozilla_dom_indexeddb_idbresult_h__