Listening and Processing Events
This section explains how to programmatically list, subscribe to, and process context events.
Finding Out Available Events
There are several ways to get a list of events from a Context
:
getEventDefinitions()
- this method will return the list of all events in a contextgetEventDefinitions(CallerController caller)
- this method will return the list of events accessible by the callergetEventDefinitions(String group)
- this method will return the list of events belonging to the certain group of its subgroupsgetEventDefinitions(CallerController caller, String group)
- this method will return the list of events belonging to the certain group and accessible by the caller
List<EventDefinition> events = context.getEventDefinitions(ContextUtils.GROUP_REMOTE); // Finding all events in group "remote"
To retrieve a single event definition by its name call getEventDefinition(String name)
method. The method getEventDefinition(String name, CallerController caller)
will return a definition or null if event is not accessible by caller.
Firing Events
Once a certain system component (e.g. a device driver) detects that something has happened, it may fire a context event. The event may be fired asynchronously from any thread. Note, that there is no generic way to fire any event remotely (from Iotellect Client or when using Iotellect Server API).
To fire an event, call Context.fireEvent()
method. This method accepts the following parameters:
- Event name
- Event level
- Event parameters (in form of pre-built
DataTable
or list of its cell values)
Example 1 - firing event by supplying event DataTable
:
DataTable eventData = new SimpleDataTable(context.getEventDefinition("myEvent").getFormat()); // Creating empty value
eventData.addRecord().addString("str1").addInt(111); // Adding a record
context.fireEvent("myEvent", EventLevel.INFO, eventData);
Example 2: firing event by supplying event table cell value:
context.fireEvent("randomValue", EventLevel.INFO, new Float(Math.random() * 1000000));
Subscribing to Events
Server modules may subscribe to context events in order to receive and process their instances. In most cases, this functionality is required when implementing custom server plugins.
To subscribe to an event:
- Create an instance of class implementing
ContextEventListener
interface. - Pass this instance to
Context.addEventListener()
method.
ContextEventListener
interface has a lot of methods, and in most cases it's more reasonable to inherit your event listener from an existing implementation:
- If writing server-side code (e.g. a plugin), inherit your event listener from
ServerEventListener
to ensure its correct permission level. - In other cases, use
DefaultContextEventListener
as a base.
![]() | Using |
Mass Subscription via Context Manager
It's also possible to subscribe to events via ContextManager
's void addMaskEventListener(String mask, String event, ContextEventListener listener)
method. This way of subscription has a number of differences from subscription via Context
interface:
- It's possible to receive events from multiple sources by specifying a mask of contexts
- If subscription was made via
ContextManager
and a certain context was deleted and then a new context with the same name was created, the same listener will be automatically added to the new context
Weak Listeners
Most subscription methods have signatures accepting a boolean weak
flag. If a certain listener was added as weak, the subscriber will be automatically garbage collected if there are no other strong references to it.
All listeners are non-weak by default.
Fingerprints
To speed up event handling fingerprints can be used. Fingerprints allow to match fired event and appropriate listeners more effectively, without traversing over listeners and evaluating their filter expressions. If an event definition has a fingerprint expression it is evaluated when the event is fired. Then listeners with the same fingerprint are directly selected among other fingerprinted listeners. Afterward the rest listeners are traversed in a standard manner.
Processing Events
Once you've subscribed to an event by adding a listener, the system will call listener's handle()
method each time when an event occurs. Thus, all processing logic should be added to the implementation of this method:
DefaultContextEventListener listener = new DefaultContextEventListener()
{
@Override
public void handle(Event event) throws EventHandlingException
{
// Handling event here
int level = event.getLevel();
List<Acknowledgement> acknowledgements = event.getAcknowledgements();
// Accessing event data
DataTable eventData = event.getData();
}
};
context.addEventListener("eventName", listener);
Accessing Historical Events
There are two ways to access historical events stored in the server database:
- By calling get function from Events context. This method will work both locally (inside server drivers/plugins) and remotely (via server API).
- By calling
Server.getDaoFactory().getEventDao().getEvents()
Java method. This way is slightly faster than previous one, but it will work inside the server JVM only (e.g. in drivers/plugins).
Calling Get Function
The get
function of Events context returns a Data Table each row of that contains information about one historic event. See its description here.
DataTable history = eventsContext.callFunction(EventsContextConstants.F_GET, getCallerController(), con.getPath(), vd.getName()); // Other parameters omitted
for (DataRecord rec : history)
{
// Process historical events
}
Using Direct Database Access
Method of EventDao
class that performs loading of events from the database has the following signature:
public Iterator<Event> getEvents(ContextManager cm, CallerController caller, EntityList eventList, EventsFilter filter, Date startDate, Date endDate, Integer maxResults, String sortBy, boolean sortAscending, Object... additionalCriteria) throws DaoException;
This method accept a list of event types to load, pre-filter (additionalCriteria
) and post-filter (filter
), start/end dates of loaded events occurrence period, maximum number of events to load, and sort options. startDate
, endDate
and maxResults
parameters can be set to null to disable time range and count limit.
It returns an iterator of Event
objects representing historical events.
Usage example:
EntityList events = new EntityList("users.admin.devices.my_device", "my_device_event");
Iterator<Event> eventItr = Server.getDaoFactory().getEventDao().getEvents(cm, caller, entities, null, null, null, null, EventDao.EVENT_FIELD_CREATIONTIME, true);
while (eventItr.hasNext())
{
Event ev = eventItr.next();
Date eventTime = ev.getCreationtime();
DataTable eventData = ev.getData();
// Processing
}
Using Helper Constants
Interfaces located in com.tibbo.aggregate.common.server
package provide string constants matching names of most server context events and their fields.
Using those constants is always preferred to using string constants defined in your own code. This will ensure error-proof code if some events or fields will get renamed or relocated in the future versions of Iotellect Server.
Was this page helpful?