| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2021, Kyle Pereira <hey@xylepereira.me> | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:47 +01:00
										 |  |  |  * Copyright (c) 2022, kleines Filmröllchen <filmroellchen@serenityos.org> | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |  * Copyright (c) 2021-2023, Ali Mohammad Pur <mpfard@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  | #include <AK/Concepts.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | #include <LibCore/EventLoop.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-06 18:09:39 +02:00
										 |  |  | #include <LibCore/EventReceiver.h>
 | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Core { | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-11 10:58:49 +02:00
										 |  |  | class PromiseBase | 
					
						
							|  |  |  |     : public RefCounted<PromiseBase> | 
					
						
							|  |  |  |     , public Weakable<PromiseBase> { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     virtual ~PromiseBase() = default; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-11 10:58:49 +02:00
										 |  |  | template<typename Result, typename TError> | 
					
						
							|  |  |  | class Promise : public PromiseBase { | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     using ErrorType = TError; | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-11 10:58:49 +02:00
										 |  |  |     virtual ~Promise() = default; | 
					
						
							|  |  |  |     static NonnullRefPtr<Promise> construct() { return adopt_ref(*new Promise()); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-30 20:10:35 -05:00
										 |  |  |     Function<ErrorOr<void, TError>(Result&)> on_resolution; | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     Function<void(ErrorType&)> on_rejection; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-18 08:36:30 +02:00
										 |  |  |     template<typename U> | 
					
						
							|  |  |  |     static NonnullRefPtr<Promise> after(Vector<NonnullRefPtr<Promise<U>>> promises) | 
					
						
							| 
									
										
										
										
											2025-05-13 12:11:47 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         auto promise = Promise::construct(); | 
					
						
							|  |  |  |         struct Resolved : RefCounted<Resolved> { | 
					
						
							|  |  |  |             explicit Resolved(size_t n) | 
					
						
							|  |  |  |                 : needed(n) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             size_t count { 0 }; | 
					
						
							|  |  |  |             size_t needed { 0 }; | 
					
						
							| 
									
										
										
										
											2025-06-18 08:36:30 +02:00
										 |  |  |             Optional<Error> error; | 
					
						
							| 
									
										
										
										
											2025-05-13 12:11:47 +02:00
										 |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto resolved = make_ref_counted<Resolved>(promises.size()); | 
					
						
							|  |  |  |         auto weak_promise = promise->template make_weak_ptr<Promise>(); | 
					
						
							|  |  |  |         for (auto p : promises) { | 
					
						
							| 
									
										
										
										
											2025-06-18 08:36:30 +02:00
										 |  |  |             p->when_resolved([weak_promise, resolved](auto&&) -> ErrorOr<void> { | 
					
						
							| 
									
										
										
										
											2025-05-13 12:11:47 +02:00
										 |  |  |                 if (!weak_promise || weak_promise->is_rejected()) | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-18 08:36:30 +02:00
										 |  |  |                 if (++resolved->count == resolved->needed) { | 
					
						
							|  |  |  |                     if (resolved->error.has_value()) | 
					
						
							|  |  |  |                         weak_promise->reject(resolved->error.release_value()); | 
					
						
							|  |  |  |                     else | 
					
						
							|  |  |  |                         weak_promise->resolve({}); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-05-13 12:11:47 +02:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-18 08:36:30 +02:00
										 |  |  |             p->when_rejected([weak_promise, resolved](auto&& error) { | 
					
						
							|  |  |  |                 resolved->error = move(error); | 
					
						
							|  |  |  |                 if (!weak_promise) | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (++resolved->count == resolved->needed) | 
					
						
							|  |  |  |                     weak_promise->reject(resolved->error.release_value()); | 
					
						
							| 
									
										
										
										
											2025-05-13 12:11:47 +02:00
										 |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             promise->add_child(*p); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return promise; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     void resolve(Result&& result) | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         m_result_or_rejection = move(result); | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         if (on_resolution) { | 
					
						
							|  |  |  |             auto handler_result = on_resolution(m_result_or_rejection->value()); | 
					
						
							|  |  |  |             possibly_handle_rejection(handler_result); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     void reject(ErrorType&& error) | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         m_result_or_rejection = move(error); | 
					
						
							|  |  |  |         possibly_handle_rejection(*m_result_or_rejection); | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     bool is_rejected() | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         return m_result_or_rejection.has_value() && m_result_or_rejection->is_error(); | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     bool is_resolved() const | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         return m_result_or_rejection.has_value() && !m_result_or_rejection->is_error(); | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-24 01:41:18 +03:30
										 |  |  |     ErrorOr<Result, ErrorType> await() | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         while (!m_result_or_rejection.has_value()) | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |             Core::EventLoop::current().pump(); | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-24 01:41:18 +03:30
										 |  |  |         return m_result_or_rejection.release_value(); | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Converts a Promise<A> to a Promise<B> using a function func: A -> B
 | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							| 
									
										
										
										
											2023-08-29 14:32:41 +02:00
										 |  |  |     NonnullRefPtr<Promise<T>> map(Function<T(Result&)> func) | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-08-29 14:32:41 +02:00
										 |  |  |         NonnullRefPtr<Promise<T>> new_promise = Promise<T>::construct(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (is_resolved()) | 
					
						
							|  |  |  |             new_promise->resolve(func(m_result_or_rejection->value())); | 
					
						
							|  |  |  |         if (is_rejected()) | 
					
						
							|  |  |  |             new_promise->reject(m_result_or_rejection->release_error()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |         on_resolution = [new_promise, func = move(func)](Result& result) -> ErrorOr<void> { | 
					
						
							|  |  |  |             new_promise->resolve(func(result)); | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |         on_rejection = [new_promise](ErrorType& error) { | 
					
						
							|  |  |  |             new_promise->reject(move(error)); | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  |         }; | 
					
						
							|  |  |  |         return new_promise; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-31 23:38:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     template<CallableAs<void, Result&> F> | 
					
						
							|  |  |  |     Promise& when_resolved(F handler) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-03-08 12:17:55 -05:00
										 |  |  |         return when_resolved([handler = move(handler)](Result& result) mutable -> ErrorOr<void> { | 
					
						
							| 
									
										
										
										
											2023-12-31 09:40:43 -05:00
										 |  |  |             handler(result); | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<CallableAs<ErrorOr<void>, Result&> F> | 
					
						
							|  |  |  |     Promise& when_resolved(F handler) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         on_resolution = move(handler); | 
					
						
							|  |  |  |         if (is_resolved()) { | 
					
						
							|  |  |  |             auto handler_result = on_resolution(m_result_or_rejection->value()); | 
					
						
							|  |  |  |             possibly_handle_rejection(handler_result); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<CallableAs<void, ErrorType&> F> | 
					
						
							|  |  |  |     Promise& when_rejected(F handler) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         on_rejection = move(handler); | 
					
						
							|  |  |  |         if (is_rejected()) | 
					
						
							|  |  |  |             on_rejection(m_result_or_rejection->error()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-11 10:58:49 +02:00
										 |  |  |     void add_child(NonnullRefPtr<PromiseBase> child) { m_children.append(move(child)); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-31 23:38:04 +01:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     template<typename T> | 
					
						
							| 
									
										
										
										
											2025-09-30 20:10:35 -05:00
										 |  |  |     void possibly_handle_rejection(ErrorOr<T, TError>& result) | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     { | 
					
						
							|  |  |  |         if (result.is_error() && on_rejection) | 
					
						
							|  |  |  |             on_rejection(result.error()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-31 23:38:04 +01:00
										 |  |  |     Promise() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 01:30:27 +03:30
										 |  |  |     Optional<ErrorOr<Result, ErrorType>> m_result_or_rejection; | 
					
						
							| 
									
										
										
										
											2025-08-11 10:58:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Vector<NonnullRefPtr<PromiseBase>> m_children; | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2022-12-29 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-15 14:13:02 +10:00
										 |  |  | } |