22#include "util/async/AnyOperation.hpp"
23#include "util/async/AnyStopToken.hpp"
24#include "util/async/AnyStrand.hpp"
25#include "util/async/Concepts.hpp"
26#include "util/async/impl/ErasedOperation.hpp"
51 template <NotSameAs<AnyExecutionContext> CtxType>
65 template <RValueNotSameAs<AnyExecutionContext> CtxType>
67 AnyExecutionContext(CtxType&& ctx) : pimpl_{std::make_shared<Model<CtxType>>(std::forward<CtxType>(ctx))}
88 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn)>>;
89 static_assert(not std::is_same_v<RetType, std::any>);
91 return AnyOperation<RetType>(pimpl_->execute([fn = std::forward<
decltype(fn)>(fn)] mutable -> std::any {
92 if constexpr (std::is_void_v<RetType>) {
93 std::invoke(std::forward<decltype(fn)>(fn));
96 return std::make_any<RetType>(std::invoke(std::forward<decltype(fn)>(fn)));
112 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
113 static_assert(not std::is_same_v<RetType, std::any>);
116 pimpl_->execute([fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
117 if constexpr (std::is_void_v<RetType>) {
118 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
121 return std::make_any<RetType>(std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken)));
139 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
140 static_assert(not std::is_same_v<RetType, std::any>);
143 [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
144 if constexpr (std::is_void_v<RetType>) {
145 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
148 return std::make_any<RetType>(std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken)));
151 std::chrono::duration_cast<std::chrono::milliseconds>(timeout)
167 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
168 static_assert(not std::is_same_v<RetType, std::any>);
170 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(delay);
172 pimpl_->scheduleAfter(millis, [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
173 if constexpr (std::is_void_v<RetType>) {
174 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
177 return std::make_any<RetType>(std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken)));
196 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken,
bool>>;
197 static_assert(not std::is_same_v<RetType, std::any>);
199 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(delay);
201 millis, [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken,
auto cancelled) mutable -> std::any {
202 if constexpr (std::is_void_v<RetType>) {
203 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken), cancelled);
206 return std::make_any<RetType>(
207 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken), cancelled)
224 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn)>>;
225 static_assert(not std::is_same_v<RetType, std::any>);
227 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(interval);
229 pimpl_->executeRepeatedly(millis, [fn = std::forward<
decltype(fn)>(fn)] mutable -> std::any {
230 std::invoke(std::forward<decltype(fn)>(fn));
245 pimpl_->submit(std::forward<
decltype(fn)>(fn));
259 return pimpl_->makeStrand();
282 virtual ~Concept() =
default;
287 std::optional<std::chrono::milliseconds> timeout = std::nullopt
291 scheduleAfter(std::chrono::milliseconds, std::function<std::any(
AnyStopToken)>) = 0;
293 scheduleAfter(std::chrono::milliseconds, std::function<std::any(
AnyStopToken,
bool)>) = 0;
294 virtual impl::ErasedOperation executeRepeatedly(std::chrono::milliseconds, std::function<std::any()>) = 0;
295 virtual void submit(std::function<
void()>) = 0;
304 template <
typename CtxType>
305 struct Model : Concept {
308 template <
typename Type>
309 Model(Type&& ctx) : ctx(std::forward<Type>(ctx))
313 impl::ErasedOperation
314 execute(std::function<std::any(AnyStopToken)> fn, std::optional<std::chrono::milliseconds> timeout)
override
316 return ctx.execute(std::move(fn), timeout);
319 impl::ErasedOperation
320 execute(std::function<std::any()> fn)
override
322 return ctx.execute(std::move(fn));
325 impl::ErasedOperation
326 scheduleAfter(std::chrono::milliseconds delay, std::function<std::any(AnyStopToken)> fn)
override
328 return ctx.scheduleAfter(delay, std::move(fn));
331 impl::ErasedOperation
332 scheduleAfter(std::chrono::milliseconds delay, std::function<std::any(AnyStopToken,
bool)> fn)
override
334 return ctx.scheduleAfter(delay, std::move(fn));
337 impl::ErasedOperation
338 executeRepeatedly(std::chrono::milliseconds interval, std::function<std::any()> fn)
override
340 return ctx.executeRepeatedly(interval, std::move(fn));
344 submit(std::function<
void()> fn)
override
346 return ctx.submit(std::move(fn));
350 makeStrand()
override
352 return ctx.makeStrand();
356 stop()
const override
362 join()
const override
369 std::shared_ptr<Concept> pimpl_;
A type-erased execution context.
Definition AnyExecutionContext.hpp:41
auto execute(SomeHandlerWith< AnyStopToken > auto &&fn)
Execute a function on the execution context.
Definition AnyExecutionContext.hpp:110
auto execute(SomeHandlerWith< AnyStopToken > auto &&fn, SomeStdDuration auto timeout)
Execute a function with a timeout.
Definition AnyExecutionContext.hpp:137
auto scheduleAfter(SomeStdDuration auto delay, SomeHandlerWith< AnyStopToken > auto &&fn)
Schedule a function for execution.
Definition AnyExecutionContext.hpp:165
auto makeStrand()
Make a strand for this execution context.
Definition AnyExecutionContext.hpp:257
void stop() const
Stop the execution context.
Definition AnyExecutionContext.hpp:266
auto execute(SomeHandlerWithoutStopToken auto &&fn)
Execute a function on the execution context.
Definition AnyExecutionContext.hpp:86
auto scheduleAfter(SomeStdDuration auto delay, SomeHandlerWith< AnyStopToken, bool > auto &&fn)
Schedule a function for execution.
Definition AnyExecutionContext.hpp:194
AnyExecutionContext(CtxType &ctx)
Construct a new type-erased Execution Context object.
Definition AnyExecutionContext.hpp:53
void submit(SomeHandlerWithoutStopToken auto &&fn)
Schedule an operation on the execution context without expectations of a result.
Definition AnyExecutionContext.hpp:243
AnyExecutionContext(CtxType &&ctx)
Construct a new type-erased Execution Context object.
Definition AnyExecutionContext.hpp:67
void join() const
Join the execution context.
Definition AnyExecutionContext.hpp:275
auto executeRepeatedly(SomeStdDuration auto interval, SomeHandlerWithoutStopToken auto &&fn)
Schedule a repeating operation on the execution context.
Definition AnyExecutionContext.hpp:222
A type-erased operation that can be executed via AnyExecutionContext.
Definition AnyOperation.hpp:44
A type-erased stop token.
Definition AnyStopToken.hpp:37
A type-erased execution context.
Definition AnyStrand.hpp:40
Definition ErasedOperation.hpp:34
Specifies the interface for a handler that can be invoked with the specified args.
Definition Concepts.hpp:162
Specifies the interface for a handler that can be stopped.
Definition Concepts.hpp:154
Specifies that the type must be some std::duration.
Definition Concepts.hpp:170
This namespace implements an async framework built on top of execution contexts.
Definition AnyExecutionContext.hpp:36