IEventsAdapter#

Fully qualified name: carb::events::IEventsAdapter

Defined in carb/events/IEventsAdapter.h

class IEventsAdapter#

Interface for creating IEventStream objects that adapt IEventStream functionality to carb::eventdispatcher systems.

The IEventsAdapter interface can be used similarly to IEvents to create a specialized type of IEventStream that can be used to map and redirect events through the newer carb::eventdispatcher::IEventDispatcher system.

IEventsAdapter is meant to ease the transition from using IEventStream by switching to eventdispatcher::IEventDispatcher. The IEventStream created by IEventsAdapter can be used by existing APIs (Python and C++) and it will translate events to-and-from IEvents.

Typical Transition Case

Assume that you have current code which uses IEventStream, but the desire is to convert it to eventdispatcher::IEventDispatcher. Of course, this could be done but is likely to break any existing extensions or plugins that use the current IEventStream API.

Instead, code can switch to using eventdispatcher::IEventDispatcher but also use IEventsAdapter to create a special IEventStream that observes eventdispatcher::Event distribution, but translates them back to a IEvent for existing code. Conversely, if existing code pushes or dispatches events using IEventStream, the adapter will translate these to eventdispatcher::Event and notify observers.

IEventStream is complicated and used in a variety of ways. The following AdapterType values correspond to the typical IEventStream uses:

AdapterType::eDispatch

AdapterType::eDispatch - The simplest adapter type. Most functionality for IEventStream is disabled. This is for adapting IEventStream instances that only call IEventStream::dispatch; IEventStream::push and IEventStream::pop functionality is disabled and will result in error logging. Push-side subscribers will never be called.

Conversion to use this mode simply requires using eventdispatcher::IEventDispatcher::dispatchEvent to dispatch events. The AdapterDesc::mappings should contain entries for all event names that should be converted to IEvent by the adapter.

AdapterType::ePushPump

Another common use case for IEventStream is a stream to which objects are pushed and then immediately pumped from the same thread. From the eventdispatcher::IEventDispatcher perspective, this is likely to convert to a single eventdispatcher::IEventDispatcher::dispatchEvent call. However, this mode correctly supports both push- and pop-side subscribers as they would be correctly called in legacy code.

In this mode, when eventdispatcher::IEventDispatcher::dispatchEvent is called, if any push-side subscribers exist on the IEventStream, they are notified prior to pop-side subscribers (and event observers).

WARNING: If multiple events were pushed before pumping, changing to single dispatchEvent calls will change the push-side notification order from multiple push notifications in a row, to push, pop, push, pop notification instead. If this is problematic, consider using the adapter to push multiple events before pumping or switch to the more complex AdapterType::eFull type.

If instead the adapter IEventStream is used to push and pump, eventdispatcher::IEventDispatcher observers are only called during the pop-side phase (i.e. pump) to maintain compatibility.

AdapterType::eFull

The most complex use of legacy IEventStream is as both a message queue and event dispatcher. In this case, IEventStream::push is called for potentially several events, but IEventStream::pump (or IEventStream::pop) is called later and possibly even from a different thread. Converting to eventdispatcher::IEventDispatcher is a bit more complex as a queue must now be involved. This adapter type must be combined with eventdispatcher::IMessageQueue to provide the message queuing.

For conversion, the paradigm is typically:

  1. When an event is queued, call eventdispatcher::IEventDispatcher::dispatchEvent with a pre-delivery notification event. This should be specified as MappingEntry::pushName during adapter creation and will function as the push-side IEventStream notification. Typically the name of this event will be similar to the MappingEntry::dispatchName (such as adding :push or :pre for instance).

  2. Immediately after the above event is dispatched, push a message with the corresponding MappingEntry::dispatchName to the eventdispatcher::IMessageQueue in use (this must also be given to adapter creation via AdapterDesc::messageQueue).

  3. At a later point, pop entries from the eventdispatcher::IMessageQueue and dispatch them via eventdispatcher::IEventDispatcher::dispatchEvent.

The adapter will observe the pre-delivery event and call push-side IEventListener subscribers when notified. When the event is dequeued and dispatched, the adapter will call pop-side IEventListener subscribers.

Conversely, if IEventStream::push is called, the adapter dispatches the pre-delivery notification and adds the event to the eventdispatcher::IMessageQueue. If the adapter’s IEventStream::pump is called (or IEventStream::pop), the event is popped from the eventdispatcher::IMessageQueue and immediately dispatched.

AdapterType::eFullAlias

The eFullAlias AdapterType is similar to eFull, but instead of using a mapping of events for translation, legacy events are renamed and dispatched and rely upon eventdispatcher::IEventDispatcher aliasing to provide the mapping between IEventStream events and eventdispatcher::IEventDispatcher events.

Like eFull, this must be combined with eventdispatcher::IMessageQueue to provide the queuing aspect.

EventType for IEvent::type is a 64-bit unsigned integer value. While it is recommended that a string hash is used for this value, this value is not reversible to a string. Therefore before any event is sent to this adapter type, it is required that eventdispatcher::IEventDispatcher::addEventAlias is called to register an alias to a target event. In the typical case of pop-side events, the alias will be named AdapterAlias:<64-bit number> and the target will be the corresponding event name dispatched through eventdispatcher::IEventDispatcher. To support push-side events, the alias will be named AdapterAlias:<64-bit number>:immediate and the target will be the corresponding immediate event name dispatched through eventdispatcher::IEventDispatcher. The <64-bit number> is converted to a string as if by the PRIu64 printf format specifier.

This produces the following behavior:

Unlike the other AdapterType values, this adapter does not listen to any events; instead aliases are relied upon. Creating this adapter type fails if any mappings were provided in the AdapterDesc. Aliases for EventType can be created by using the helper function registerEventAlias().

CAVEAT: Since mappings are ignored and aliases are used, only subscriptions by event type are honored. Subscriptions that do not provide an event type will not be called. Attempting to subscribe without specifying an event type will emit an error log message.

Event Variants -> IEvent Payload Translation

The RStringKey is converted to a string via RStringKey::toString and used as the child item name in the payload.

IEvent Payload -> Event Variants Translation

If the item name is numeric only, then it is used as the number portion of RStringKey. Otherwise, the string name is converted directly to RStringKey and used as the key.

Any objects attached to the IEvent as via IEvent::attachObject are directly converted to variant::Variant of carb::IObjectPtr and stored as their attached object name.

Public Functions

inline omni::expected<IEventStreamPtr, omni::core::Result> createAdapter(
const AdapterDesc &desc,
) noexcept#

Creates an IEventStream that is used for simple event dispatch streams.

Thread Safety

This function is thread-safe.

Errors

Accessible via the omni::unexpected return value.

Parameters:

desc – The AdapterDesc describing the adapter type.

Returns:

An omni::expected type containing the pointer to the created IEventStream, or if the IEventStream could not be created, a omni::unexpected containing the error.

Public Static Functions

static inline constexpr carb::InterfaceDesc getInterfaceDesc(
) noexcept#

Returns information about this interface.

Auto-generated by CARB_PLUGIN_INTERFACE() or CARB_PLUGIN_INTERFACE_EX.

Returns:

The carb::InterfaceDesc struct with information about this interface.

static inline constexpr carb::InterfaceDesc getLatestInterfaceDesc(
) noexcept#

Returns information about the latest version of this interface.

Auto-generated by CARB_PLUGIN_INTERFACE() or CARB_PLUGIN_INTERFACE_EX.

Returns:

The carb::InterfaceDesc struct with information about the latest version of this interface.