The base package provides the basic building blocks for an asynchronous/event-based application and, more often than not, will be used by the rest of the packages.
The backbone is the event loop provided by kouta::base::asio
, which aliases either asio::
or boost::asio
depending on whether the KOUTA_STANDALONE_ASIO
option is set in CMake.. Kouta builds on top of that to provide a tree-like architecture with the following elements.
Implemented in kouta::base::Component
.
This is the base type for any asynchronous element that requires access to the event loop. A Component
does not own the I/O context, and instead is expected to have a parent from which it can reference said context (hence the aforementioned tree-like architecture).
Apart from providing access to the event loop, the parent also keeps track of its children. This allows for components to be allocated in the heap via the new
keyword if desired so that, when a Component
is deleted, it will automatically attempt to delete
its children and remove itself from its parent.
In the case of objects allocated in the stack, each object will be deleted in reverse creation order, which should prevent any issues with calling delete
on non-heap objects due to objects removing themselves from the parent's children list.
Note: take extra care when mixing stack and heap allocations, as this may cause invalid deletions.
These components can be chained as many times as needed, as long as there is one parent at the root of the tree which provides the I/O context (see kouta::base::Component::context()
).
Note that all components deriving from a single parent will be executed in the same thread.
Method calls can be posted to the event loop via the kouta::base::Component::post()
method. This is the core of the event-based architecture, as it allows to defer method execution in a single thread in an efficient manner. In addition, it is also possible to post calls to lambdas and free functions.
A simple component could be:
Implemented in kouta::base::Root
.
As opposed to the base Component
explained above, the Root
owns the I/O context and is supposed to serve as the parent for other components that want to share the same event loop.
While a parent can be provided to the root, it is only used when dealing with heap-allocated objects in order to guarantee that the Root
is deleted with its parent.
Note that starting the event loop of a Root
(kouta::base::Root::run()
) blocks the current thread and will only be unblocked after the event loop is terminated (e.g. via the kouta::base::Root::stop()
method).
Implemented in kouta::base::Branch
.
The problem with the Root
type is that there is a one-to-one relation between a Root
and a thread, considering that the event loop must block in order to handle events. Even though this is probably fine in most cases, there might be situations where another "asynchronous tree" needs to be introduced in the application, for instance to handle communications with an I/O device with high throughput.
The Branch
type was introduced for such cases. It is a Root
(owns the I/O context), but manages an std::thread
internally. As opposed to the Root
, starting the event loop via kouta::base::Branch::run()
will not block, and instead start the event loop in the child thread.
It is supposed to wrap a Component
, allowing external components to post events to the wrapped Component
, as well as to the Branch
itself. In addition, the Branch
assumes that the wrapped Component
expects a pointer to a parent as its first argument (which will be set to the Branch
itself).
The parent provided to the Branch
is only used when dealing with heap-allocated objects in order to guarantee that the Root
is deleted with its parent.
Implemented in kouta::base::callback
.
Callbacks are wrappers around std::function
that allow safely registering a method to call.
There are different types of callbacks:
kouta::base::callback::BaseCallback
, aliased as kouta::base::Callback
): Empty callback type which is supposed to be used as an implementation-agnostic callback. If called as-is, whill throw an exception, helping identify non-defined callbacks.kouta::base::callback::DirectCallback
): Can be seen as a direct method call within the same thread. Can be used with any type of object.kouta::base::callback::DeferredCallback
): Posts a method/functor call to the event loop of a Component
, effectively running it in that component's thread.kouta::base::callback::CallbackList
): Container for any of the aforementioned callbacks. When invoked, will in turn invoke the contained callbacks, regardless of their type, allowing for a one-to-many model.Callbacks cannot return anything.
Implemented in kouta::base::Timer
.
The Timer
allows performing actions after a certain amount of time. Its duration can be modified at any point, and can also be stopped before it elapses (for instance to cancel a timeout in a request).
Once the Timer
elapses, it is within the context of its parent component, so it is up to the developer to make sure that proper synchronization occurs if calling something outside of that context.