Guide for developing with events ================================ Signaling agents are services that send events to any clients that subscribed for receiving it. The sent events carry any data. To have a good understanding of how to write a signaling agent, the actions of subscribing, unsubscribing, producing, sending and receiving events must be described and explained. Overview of events ------------------ The basis of a signaling agent is shown in the following figure: ![scenario of using events](pictures/signaling-basis.svg) This figure shows the main role of the signaling framework for the events propagation. For people not familiar with the framework, a signaling agent and a “binding” are similar. ### Subscribing and unsubscribing Subscribing is the action that makes a client able to receive data from a signaling agent. Subscription must create resources for generating the data, and for delivering the data to the client. These two aspects are not handled by the same piece of software. Generating the data is the responsibility of the developer of the signaling agent while delivering the data is handled by the framework. When a client subscribes for data, the agent must: 1. check that the subscription request is correct; 2. establish the computation chain of the required data, if not already done; 3. create a named event for the computed data, if not already done; 4. ask the framework to establish the subscription to the event for the request; 5. optionally give indications about the event in the reply to the client. The first two steps are not involving the framework. They are linked to the business logic of the binding. The request can be any description of the requested data and the computing stream can be of any nature, this is specific to the binding. As said before, the framework uses and integrates **libsystemd** and its event loop. Within the framework, **libsystemd** is the standard API/library for bindings expecting to setup and handle I/O, timer or signal events. Steps 3 and 4 are bound to the framework. The agent must create an object for handling the propagation of produced data to its clients. That object is called “event” in the framework. An event has a name that allows clients to distinguish it from other events. Events are created using the ***afb\_daemon\_make\_event*** function that takes the name of the event. Example: ```C event = afb_daemon_make_event(afb_daemon, name); ``` Once created, the event can be used either to push data to its subscribers or to broadcast data to any listener. The event must be used to establish the subscription for the requesting client. This is done using the ***afb\_req\_subscribe*** function that takes the current request object and event and associates them together. Example: ```C rc = afb_req_subscribe(afb_req, event); ``` When successful, this function make the connection between the event and the client that emitted the request. The client becomes a subscriber of the event until it unsubscribes or disconnects. The ***afb\_req\_subscribe*** function will fail if the client connection is weak: if the request comes from a HTTP link. To receive signals, the client must be connected. The AGL framework allows connections using WebSocket. The name of the event is either a well known name or an ad hoc name forged for the use case. Let's see a basic example: client A expects to receive the speed in km/h every second while client B expects the speed in mph twice a second. In that case, there are two different events because it is not the same unit and it is not the same frequency. Having two different events allows to associate clients to the correct event. But this doesn't tell any word about the name of these events. The designer of the signaling agent has two options for naming: 1. names can be the same (“speed” for example) with sent data self describing itself or having a specific tag (requiring from clients awareness about requesting both kinds of speed isn't safe). 2. names of the event include the variations (by example: “speed-km/h-1Hz” and “speed-mph-2Hz”) and, in that case, sent data can self describe itself or not. In both cases, the signaling agent might have to send the name of the event and/or an associated tag to its client in the reply of the subscription. This is part of the step 5 above. The framework only uses the event (not its name) for subscription, unsubscription and pushing. When the requested data is already generated and the event used for pushing it already exists, the signaling agent must not instantiate a new processing chain and must not create a new event object for pushing data. The signaling agent must reuse the existing chain and event. Unsubscribing is made by the signaling agent on a request of its client. The ***afb\_req\_unsubscribe*** function tells the framework to remove the requesting client from the event's list of subscribers. Example: ```C afb_req_unsubscribe(afb_req, event); ``` Subscription count does not matter to the framework: subscribing the same client several times has the same effect that subscribing only one time. Thus, when unsubscribing is invoked, it becomes immediately effective. #### More on naming events Within the AGL framework, a signaling agent is a binding that has an API prefix. This prefix is meant to be unique and to identify the binding API. The names of the events that this signaling agent creates are automatically prefixed by the framework, using the API prefix of the binding. Thus, if a signaling agent of API prefix ***api*** creates an event of name ***event*** and pushes data to that event, the subs
BBLAYERS =+ "\
  ${METADIR}/meta-renesas-rcar-gen3/meta-rcar-gen3 \
  ${METADIR}/meta-agl/meta-agl-bsp/meta-rcar-gen3-adas \
  "
tracting and specializing The key is modularity versus cost of propagation. It can be partly solved when logical group of signaling agent are launched together in the same binder process. In that particular case, the cost of propagation of data between agents is reduced[^2] because there is no serialization. This reduction of the propagation cost (and of the resources used) precludes implementation of strong security between the agents because they share the same memory. ### Soft composition The soft composition implements the business logic of high-level signaling agents as libraries that can then be used directly by the low level signaling agents. Advantages: - No propagation: same memory, sharing of native structures Drawbacks: - Cannot be used for aggregation of several sources - Difficulties to abstract low-level signaling agent or to find a trade-off between abstracting and specializing - Source code binding not good for maintenance [^1]: There are two aspect in using JSON: the first is the flexible data structure that mixes common types (booleans, numbers, strings, arrays, dictionaries, nulls), the second, is the streaming specification. Streaming is often seen as the bottleneck of using JSON (see http://bjson.org). When the agent share the same process, there is no streaming at all. [^2]: Within the same process, there is not serialization, the propagation has the cost of wrapping a json data and calling callbacks with the benefit of having a powerful callback manager: the event mechanism of the framework.