Standard Driver Toolkit

Iotellect Server Driver Development Kit also includes a so-called standard driver toolkit which is a set of classes simplifying creation of typical drivers for network devices, PLCs, IoT gateways and sensors, satellite trackers, etc.

The standard toolkit is technically enabling easy development of drivers with the following characteristics:

  • Communication over TCP, UDP or serial link

  • Parsing incoming data stream into separate commands

  • Support for notifications (asynchronous incoming commands) and replies (synchronous responses to previously sent commands)

  • Custom parsing and processing of incoming commands by user code

  • Encoding and sending of outgoing commands

  • Matching incoming commands to previously sent outgoing commands via custom IDs

  • Support for sending multiple outgoing commands concurrently and receiving replies out-of-order

Implementing Drivers Based on the Standard Toolkit

To implement a Iotellect Server device driver based on a standard toolkit and using fully custom connection configuration and device communication code, you should:

  • Create a new generic device driver (plugin class + driver class) as described in other articles of this section

  • Create a new device command class inherited from Command and, optionally, more inherited classes for various types of incoming and outgoing commands. The command classes essentially contain both raw and parsed device data

  • Create a new device controller class inherited from AbstractDeviceController and parameterize it by base incoming/outgoing command classes. The device controller is responsible to handling device communications and data

  • Create a new command parser class either implementing CommandParser interface or (better) inherited from AbstractCommandParser

  • For the second approach (using AbstractCommandParser), the system will add incoming data to your parser's buffer byte by byte and then call readCommand() method. If you see that the buffer contains a fully formed command just create a Command object and return it, otherwise return null. Currently collected data is internally stored as a ByteArrayOutputStream and available via byte[] getData()

  • Override connectImpl(), loginImpl(), and disconnectImpl() methods of your device controller and put device connection, login and disconnection logic into them

  • Make sure you're calling setCommandParser() method from your custom connectImpl() to provide a correct command parser for the device controller

  • Override send(Command cmd) method of your device controller and implement command sending logic in it, i.e. send command data to a socket created during connection

  • As always, override device driver's setupDeviceContext(DeviceContext deviceContext) methods and add device controller's properties to the ContextUtils.GROUP_ACCESS group of device context variables

  • Override connect() method of your device driver class to create your custom device controller and call its connect() method. Implement driver's disconnect() by delegating logic to disconnect() call of the device controller

  • Put necessary device communication logic into driver's readVariableDefinitions(), readFunctionDefinitions(), readEventDefinitions(), readVariableValue(), writeVariableValue() and executeFunction(). Those methods are normally interfacing the device controller by sending various commands, getting replies and normalizing device protocol values, i.e. converting them to/from DataTable objects

  • If your device is sending asynchronous commands (events), make sure to implement AbstractDeviceController's processAsyncCommand() method. Its implementation should normally fire a custom event in a device context. Make sure the event type was previously introduced to the core by returning it from driver's readEventDefinitions()

Implementing Drivers Based on the Standard Toolkit and Device Communications

This section describes how to implement device drivers that:

  • Use TCP, UDP or serial communications and unified device configuration approach

  • Use binary or text device commands separated by one or two custom characters ("separators")

In this case, you should generally follow the guidelines of Implementing Drivers Based on Standard Toolkit section. However, implementing such a driver is even easier. The list of differences is described below:

  • Implement a connection properties object either implementing ConnectionProperties interface or extending DefaultConnectionProperties class

  • If inheriting from DefaultConnectionProperties, clone its FORMAT property in a static block, make necessary modifications, add your custom fields and pass the new format to DefaultConnectionProperties constructor

  • Base your device controller on StandardDeviceController instead of AbstractDeviceController. You won't need to implement connection/disconnection logic, as well as command sending logic. Just make sure you still override connectImpl() to set the command parser via setCommandParser() after calling super.connectImpl().

To recap, in this scenario you're not handling device connectivity issues and, thus, can concentrate directly on device data exchange.

The amount of code to be developed for creating a new device driver according to this scenario is absolutely minimal.

Was this page helpful?