gecko-dev/dom/quota/MozPromiseUtils.h
Jan Varga 6ba3d89eda Bug 1962129 - QM: Extend Map to support mapping from non-exclusive to exclusive MozPromise; r=dom-storage-reviewers,hsingh
Extends the Map helper to support mapping from a non-exclusive MozPromise to
an exclusive MozPromise. All combinations of promise exclusivity are handled
explicitly, with careful control over lambda capture and mutability, ensuring
that lambdas are only marked `mutable` when strictly necessary. This improves
consistency and flexibility in future promise chains while preserving full
compatibility with existing usage.

Differential Revision: https://phabricator.services.mozilla.com/D246609
2025-06-09 06:32:50 +00:00

109 lines
3.7 KiB
C++

/* -*- 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 DOM_QUOTA_MOZPROMISEUTILS_H_
#define DOM_QUOTA_MOZPROMISEUTILS_H_
#include "mozilla/MozPromise.h"
#include "nsThreadUtils.h"
namespace mozilla::dom::quota {
namespace detail {
template <typename T>
struct IsExclusiveMozPromise {
static constexpr bool value = false;
};
// Specialization for MozPromise
template <typename ResolveValueT, typename RejectValueT, bool IsExclusive>
struct IsExclusiveMozPromise<
MozPromise<ResolveValueT, RejectValueT, IsExclusive>> {
static constexpr bool value = IsExclusive;
};
} // namespace detail
template <typename T, typename U, typename F>
auto Map(RefPtr<U> aPromise, F&& aFunc) -> std::enable_if_t<
!detail::IsExclusiveMozPromise<RemoveSmartPointer<T>>::value &&
detail::IsExclusiveMozPromise<RemoveSmartPointer<U>>::value,
RefPtr<T>> {
return aPromise->Then(
GetCurrentSerialEventTarget(), __func__,
[func =
std::forward<F>(aFunc)](typename U::ResolveOrRejectValue&& aValue) {
if (aValue.IsReject()) {
return T::CreateAndReject(aValue.RejectValue(), __func__);
}
auto value = func(std::move(aValue));
return T::CreateAndResolve(value, __func__);
});
}
template <typename T, typename U, typename F>
auto Map(RefPtr<U> aPromise, F&& aFunc) -> std::enable_if_t<
!detail::IsExclusiveMozPromise<RemoveSmartPointer<T>>::value &&
!detail::IsExclusiveMozPromise<RemoveSmartPointer<U>>::value,
RefPtr<T>> {
return aPromise->Then(GetCurrentSerialEventTarget(), __func__,
[func = std::forward<F>(aFunc)](
const typename U::ResolveOrRejectValue& aValue) {
if (aValue.IsReject()) {
return T::CreateAndReject(aValue.RejectValue(),
__func__);
}
auto value = func(aValue);
return T::CreateAndResolve(value, __func__);
});
}
template <typename T, typename U, typename F>
auto Map(RefPtr<U> aPromise, F&& aFunc) -> std::enable_if_t<
detail::IsExclusiveMozPromise<RemoveSmartPointer<T>>::value &&
detail::IsExclusiveMozPromise<RemoveSmartPointer<U>>::value,
RefPtr<T>> {
return aPromise->Then(
GetCurrentSerialEventTarget(), __func__,
[func = std::forward<F>(aFunc)](
typename U::ResolveOrRejectValue&& aValue) mutable {
if (aValue.IsReject()) {
return T::CreateAndReject(aValue.RejectValue(), __func__);
}
auto value = func(std::move(aValue));
return T::CreateAndResolve(std::move(value), __func__);
});
}
template <typename T, typename U, typename F>
auto Map(RefPtr<U> aPromise, F&& aFunc) -> std::enable_if_t<
detail::IsExclusiveMozPromise<RemoveSmartPointer<T>>::value &&
!detail::IsExclusiveMozPromise<RemoveSmartPointer<U>>::value,
RefPtr<T>> {
return aPromise->Then(
GetCurrentSerialEventTarget(), __func__,
[func = std::forward<F>(aFunc)](
const typename U::ResolveOrRejectValue& aValue) mutable {
if (aValue.IsReject()) {
return T::CreateAndReject(aValue.RejectValue(), __func__);
}
auto value = func(aValue);
return T::CreateAndResolve(std::move(value), __func__);
});
}
} // namespace mozilla::dom::quota
#endif // DOM_QUOTA_MOZPROMISEUTILS_H_