Implementing a Driver

To create a new device driver from scratch, create a new Java class inherited from AbstractDeviceDriver. You will need to override at least some of the methods to provide driver functionality.

Global and Per-Account Configuration

Certain plugins may have global and per-user-account configuration settings.

To add global and per-user settings, implement globalInit() and userInit() methods respectively. Their implementations should call createGlobalConfigContext() and createUserConfigContext() and provide VariableDefinition's of global and per-user settings. These settings will be exposed to system administrators, their values get persisted in the database. The variables must belong to "default" group, available as ContextUtils.GROUP_DEFAULT constant.

To access setting values from any method of the plugin, retrieve global or per-user config contexts using getGlobalConfigContext() or getUserConfigContext(), then call Context.getVariable() to fetch instances of DataTable representing setting values.

Setting up Device Context

Iotellect Server calls the setupDeviceContext() method of the driver when a new device account is created or the server is started. The method receives an instance of DeviceContext interface that is a Device Context representing this device account. Call super.setupDeviceContext() to store its reference inside AbstractDeviceDriver.  It will be available via getDeviceContext() method.

The main purpose of setupDeviceContext() is adding device-specific communication settings. They are added in the form of context variables belonging to ContextUtils.GROUP_ACCESS variable group. System operator is prompted to edit these settings upon device account creation, and they can be later accessed using Edit Device Properties action.

Communication setting variables should be added to device context using Context.addVariableDefinition() method.

The actual user-specified values can be later fetched from DeviceContext using Context.getVariable(), e.g. from DeviceDriver.startSynchronization() or DeviceDriver.connect().

Starting and Finishing Synchronization

Device driver has two methods that are called in the beginning and in the end of synchronization respectively: startSynchronization() and endSynchronization(). These methods may be used to initialize and clear some driver's internal caches or other data structures.

Handling Connections and Disconnections

DeviceDriver.connect() method should establish a connection with the hardware device. It usually uses communication settings described in Setting Up Device Context (see above). In most cases, connection includes several steps:

  • Fetch device communication properties by calling getDeviceContext.getVariable(). Use getDeviceContext().getCallerController() as a caller controller in these calls to ensure proper permission level.

  • Bringing up a link (opening TCP socket, serial port, etc.)

  • Sending authentication/authorization credentials to the device and processing its reply

  • Switching the device to "configuration mode", if applicable

The connect() method should throw an exception if connection has failed at any phase. Otherwise, if should call super.connect() method of AbstractDeviceDriver in the end of method body to mark device as connected. AbstractDeviceDriver will then provide a correct result of isConnected() method.

DeviceDriver.disconnect() should correctly shutdown the link to hardware. It's called on server shutdown, account removal, or when reconnection to the device is requested by the system.

To force reconnection during driver testing/debugging, use Reset Device Driver action.

Reading Device Metadata

There are three methods that are called by the Iotellect Server core during full synchronization:

  • readVariableDefinitions() that should examine available device settings (or communication properties if the device protocol does not support self-documentation) and create a VariableDefinition object for every available device setting. depending on the communication protocol, device settings may be also called I/O channels, tags, configuration items, etc. See variable definition for details.

  • readFunctionDefinitions() that should provide one FunctionDefinition object per every operation provided by the hardware. In different communication standards, device operations may be also called device methods, functions, actions, procedures, etc. See function definition for details.

  • readEventDefinitions() that provides one EventDefinition object for every type of event that can be generated by hardware. In some device protocols, events may be also called messages, notifications, etc. See event definition for details.

Definitions of variables, functions and events provided by the driver are added to the <%dt%> context and cached. The driver may not be asked to re-read them until the next full synchronization even if the server was restarted.

Variables, functions and events belonging to a device-side (relied on communication with the device) must be declared in readVariableDefinitions(), readFunctionDefinitions() and readEventDefinitions() methods to avoid interference and synchronization sequence disruption.

To force re-reading of device metadata during driver testing/debugging, use Reset Device Driver action.

Reading and Writing Device Settings

If the driver has provided a non-empty list of device settings from readVariableDefinitions() method, server core will call readVariableValue() and writeVariableValue() methods of the driver. readVariableValue() should read up-to-date setting value from the hardware and convert it to the form of DataTable matching format defined in its VariableDefinition. On the contrary, writeVariableDefinition() takes value encoded as a DataTable, converts it to the device native format and writes to the hardware.

Setting is being read from the hardware:

  • During first synchronization, and

  • During any other synchronization if there are no server-side changes of its value

Setting is being written to the hardware during synchronization if it's writable according to the definition and there were some server-side changes of its value.

Setting values will be cached inside the <%dt%> context. The driver may not be asked to re-read them until the next full synchronization even if the server was restarted.

To force re-reading of device settings during driver testing/debugging, use Synchronize action.

Executing Device Operations

If the driver has provided a non-empty list of device settings from readFuncitonDefinitions() method, Iotellect Server will add a Call Function action per every device operation. Being executed by system operator, this action prompts to specify input data (according to function input format, and only if it's not "empty"), then calls executeFunction() method of device driver and shows its output to the operator (also only if it's not "empty").

Thus, driver-specific implementation of executeFunction() method should basically:

  • Convert input Data Table to the device native format

  • Send input data to the hardware

  • Trigger device-side operation

  • Read operation output from the device

  • Convert device-side output to the Data Table and return it

Working with Device Events

When dealing with events, device driver:

  • Returns the list of event definitions from readEventDefinitions() method. The list should include every type of event that may be produced by the hardware.

  • Asynchronously calls DeviceContext.fireEvent() method when new event instance is received from the hardware.

Device driver can start firing context events only after the end of first synchronization, i.e. after its finishSynchronization() method was called at least once.

When firing an event, the driver must ensure that format of event's Data Table containing event-specific data matches format stored in corresponding Event Definition.

Any event provided by device driver is persistent. It will be stored in event history if non-zero history storage time is defined for it.

Advanced Features

See javadocs of DeviceDriver, DeviceContext and parent interfaces for description of all their methods.

Access Setting Reinitializers

Once you've added some device-specific communication settings (see Setting Up Device Context), you can define actions that will be triggered if any of those settings were changed.

To do that, use DeviceContext.setAccessSettingReinitializer(String variable, AccessSettingReinitializer reinitializer). The reinitializer's reinitialize() method will be called once the specified variable was changed. The reinializer can perform some driver-specific actions. Its output is a Boolean value that defines whether device re-synchronization is required after the change.

Was this page helpful?