3#include "util/async/AnyOperation.hpp"
4#include "util/async/AnyStopToken.hpp"
5#include "util/async/AnyStrand.hpp"
6#include "util/async/Concepts.hpp"
7#include "util/async/impl/ErasedOperation.hpp"
32 template <NotSameAs<AnyExecutionContext> CtxType>
46 template <RValueNotSameAs<AnyExecutionContext> CtxType>
49 : pimpl_{std::make_shared<Model<CtxType>>(std::forward<CtxType>(ctx))}
70 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn)>>;
71 static_assert(not std::is_same_v<RetType, std::any>);
74 pimpl_->execute([fn = std::forward<
decltype(fn)>(fn)] mutable -> std::any {
75 if constexpr (std::is_void_v<RetType>) {
76 std::invoke(std::forward<decltype(fn)>(fn));
79 return std::make_any<RetType>(std::invoke(std::forward<decltype(fn)>(fn)));
96 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
97 static_assert(not std::is_same_v<RetType, std::any>);
100 [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
101 if constexpr (std::is_void_v<RetType>) {
102 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
105 return std::make_any<RetType>(
106 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken))
125 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
126 static_assert(not std::is_same_v<RetType, std::any>);
129 [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
130 if constexpr (std::is_void_v<RetType>) {
131 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
134 return std::make_any<RetType>(
135 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken))
139 std::chrono::duration_cast<std::chrono::milliseconds>(timeout)
155 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken>>;
156 static_assert(not std::is_same_v<RetType, std::any>);
158 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(delay);
160 millis, [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken) mutable -> std::any {
161 if constexpr (std::is_void_v<RetType>) {
162 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken));
165 return std::make_any<RetType>(
166 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken))
186 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn),
AnyStopToken,
bool>>;
187 static_assert(not std::is_same_v<RetType, std::any>);
189 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(delay);
192 [fn = std::forward<
decltype(fn)>(fn)](
auto stopToken,
auto cancelled) mutable
194 if constexpr (std::is_void_v<RetType>) {
195 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken), cancelled);
198 return std::make_any<RetType>(
199 std::invoke(std::forward<decltype(fn)>(fn), std::move(stopToken), cancelled)
216 using RetType = std::decay_t<std::invoke_result_t<
decltype(fn)>>;
217 static_assert(not std::is_same_v<RetType, std::any>);
219 auto const millis = std::chrono::duration_cast<std::chrono::milliseconds>(interval);
221 pimpl_->executeRepeatedly(
222 millis, [fn = std::forward<
decltype(fn)>(fn)] mutable -> std::any {
223 std::invoke(std::forward<decltype(fn)>(fn));
239 pimpl_->submit(std::forward<
decltype(fn)>(fn));
253 return pimpl_->makeStrand();
276 virtual ~Concept() =
default;
281 std::optional<std::chrono::milliseconds> timeout = std::nullopt
285 scheduleAfter(std::chrono::milliseconds, std::function<std::any(
AnyStopToken)>) = 0;
287 std::chrono::milliseconds,
291 executeRepeatedly(std::chrono::milliseconds, std::function<std::any()>) = 0;
292 virtual void submit(std::function<
void()>) = 0;
301 template <
typename CtxType>
302 struct Model : Concept {
305 template <
typename Type>
306 Model(Type&& ctx) : ctx(std::forward<Type>(ctx))
310 impl::ErasedOperation
312 std::function<std::any(AnyStopToken)> fn,
313 std::optional<std::chrono::milliseconds> timeout
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
327 std::chrono::milliseconds delay,
328 std::function<std::any(AnyStopToken)> fn
331 return ctx.scheduleAfter(delay, std::move(fn));
334 impl::ErasedOperation
336 std::chrono::milliseconds delay,
337 std::function<std::any(AnyStopToken,
bool)> fn
340 return ctx.scheduleAfter(delay, std::move(fn));
343 impl::ErasedOperation
344 executeRepeatedly(std::chrono::milliseconds interval, std::function<std::any()> fn)
override
346 return ctx.executeRepeatedly(interval, std::move(fn));
350 submit(std::function<
void()> fn)
override
352 return ctx.submit(std::move(fn));
356 makeStrand()
override
358 return ctx.makeStrand();
362 stop()
const override
368 join()
const override
375 std::shared_ptr<Concept> pimpl_;
A type-erased execution context.
Definition AnyExecutionContext.hpp:22
auto execute(SomeHandlerWith< AnyStopToken > auto &&fn)
Execute a function on the execution context.
Definition AnyExecutionContext.hpp:94
auto execute(SomeHandlerWith< AnyStopToken > auto &&fn, SomeStdDuration auto timeout)
Execute a function with a timeout.
Definition AnyExecutionContext.hpp:123
auto scheduleAfter(SomeStdDuration auto delay, SomeHandlerWith< AnyStopToken > auto &&fn)
Schedule a function for execution.
Definition AnyExecutionContext.hpp:153
auto makeStrand()
Make a strand for this execution context.
Definition AnyExecutionContext.hpp:251
void stop() const
Stop the execution context.
Definition AnyExecutionContext.hpp:260
auto execute(SomeHandlerWithoutStopToken auto &&fn)
Execute a function on the execution context.
Definition AnyExecutionContext.hpp:68
auto scheduleAfter(SomeStdDuration auto delay, SomeHandlerWith< AnyStopToken, bool > auto &&fn)
Schedule a function for execution.
Definition AnyExecutionContext.hpp:184
AnyExecutionContext(CtxType &ctx)
Construct a new type-erased Execution Context object.
Definition AnyExecutionContext.hpp:34
void submit(SomeHandlerWithoutStopToken auto &&fn)
Schedule an operation on the execution context without expectations of a result.
Definition AnyExecutionContext.hpp:237
AnyExecutionContext(CtxType &&ctx)
Construct a new type-erased Execution Context object.
Definition AnyExecutionContext.hpp:48
void join() const
Join the execution context.
Definition AnyExecutionContext.hpp:269
auto executeRepeatedly(SomeStdDuration auto interval, SomeHandlerWithoutStopToken auto &&fn)
Schedule a repeating operation on the execution context.
Definition AnyExecutionContext.hpp:214
A type-erased operation that can be executed via AnyExecutionContext.
Definition AnyOperation.hpp:25
A type-erased stop token.
Definition AnyStopToken.hpp:18
A type-erased execution context.
Definition AnyStrand.hpp:21
Definition ErasedOperation.hpp:15
Specifies the interface for a handler that can be invoked with the specified args.
Definition Concepts.hpp:165
Specifies the interface for a handler that can be stopped.
Definition Concepts.hpp:157
Specifies that the type must be some std::duration.
Definition Concepts.hpp:173
This namespace implements an async framework built on top of execution contexts.
Definition AnyExecutionContext.hpp:17