A multithreaded C++ service that exposes data from multiple Modbus networks as MQTT topics.
Main features:
- Connects to multiple TCP and RTU modbus networks
- Handles state and availablity for each configured MQTT object
- Allows to read and write to MODBUS registers from MQTT side with custom data conversion
- Flexible MQTT state topic configuration:
- single modbus register published as string value
- multiple modbus registers as JSON object
- multiple modubs registers as JSON list
- registers from different slaves combined as single JSON list/object
- Data conversion:
- single register value to MQTT converters
- multiple registers values to single MQTT value converters
- support for exprtk expressions language when converting data
- support for custom conversion plugins
- Fast modbus frequency polling, configurable per newtork, per mqtt object and per register
- Optimized modbus pulling - registers used in multiple MQTT topics are polled only once
MQMGateway depends on libmodbus and Mosqutto MQTT library. See main CMakeLists.txt for full list of dependencies. It is developed under Linux, but it should be easy to port it to other platforms.
This software is dual-licensed:
- under the terms of AGPL-3.0 license as Open Source project.
- under commercial license.
For a commercial-friendly license and support please see http://mqmgateway.zork.pl.
This software includes "A single-producer, single-consumer lock-free queue for C++" written by Cameron Desrochers. See license terms in LICENSE.md
-
git clone https://github.com/BlackZork/mqmgateway.git
-
Install dependencies:
- boost
- libmodbus
- mosquitto
- yaml-cpp
- rapidJSON
-
Configure project:
cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr -S (project dir) -B (build dir) make make install
-
Copy config.template.yml to /etc/modmqttd.yml.
-
There is an example modmqttd.service file for systemd. Copy it to /etc/systemd/system.
-
Edit configuration and start service:
systemctl start modmqttd
modmqttd configuration file is in YAML format. It is divied in three main sections:
- modmqttd section contains information about custom plugins
- modbus section contains modbus network definitions
- mqtt section contains mqtt broker connection parameters and modbus register mappings
For quick example see config.template.yaml in source directory.
-
timespan format: "([0-9]+)(ms|s|min)"
Used for various timeout interval configuration entries
-
converter_search_path (optional)
List of paths where to search for converter plugins. If path does not start with '/' then it is treated as relative to current working directory
-
converter_plugins (optional)
List of converter plugins to load. Modmqttd search for plugins in all directories specified in converter_search_path list
Modbus section contains a list of modbus networks modmqttd should connect to. Modbus network configuration parameters are listed below:
-
name (required)
Unique name for network - referenced in mqtt mappings
-
response_timeout (optional, default ?)
A timeout interval used to wait for modbus response. This setting is propagated down mqtt object and register definitions. See modbus_set_response_timeout(3)
-
response_data_timeout (optional, default ?)
A timeout interval used to wait for data when reading response from modbus device. See modbus_set_byte_timeout(3).
-
RTU device settings For details, see modbus_new_rtu(3)
-
device (required)
A path to modbus RTU device
-
baud (required)
The baud rate of the communication
-
parity (required)
N for none, E for even, O for odd
-
data_bit (required)
the number of bits of data (5,6,7,8)
-
stop_bit (required)
the bits of stop
-
-
TCP/IP device settings
-
address
IP address of a device
-
port
TCP port of a device
-
The mqtt section contains broker definition and modbus register mappings. Mappings describe how modbus data should be published as mqtt topics.
-
client_id (required)
name used to connect to mqtt broker.
-
refresh (timespan, optional, default 5s)
A timespan used to poll modbus registers. This setting is propagated down to object and register definitions
-
broker (required)
This section contains configuration settings used to connect to MQTT broker.
-
host (required)
MQTT boker IP address
-
port (optional, default 1883)
MQTT broker TCP port
-
keepalive (optional, default 60s)
The number of seconds after which the bridge should send a ping if no other traffic has occurred.
-
username (optional)
The username to be used to connect to MQTT broker
-
password (optional)
The password to be used to connect to MQTT broker
-
-
objects (required)
A list of topics where modbus values are published to MQTT broker and subscribed for writing data received from MQTT broker to modbus registers.
-
topic (required)
The base name for modbus register mapping. A mapping must contain at least one of following sections:
- commands - for writing to modbus registers
- state - for reading modbus registers
- avaiability - for checking if modbus data is available.
- response_timeout (optional)
Overrides modbus.response_timeout for this topic
-
response_data_timeout (optional)
Overrides modbus.response_data_timeout for this topic
-
refresh
Overrides mqtt.refresh for all state and availability sections in this topic
A single command is defined using following settings.
-
name (required)
A command topic name to subscribe. Full name is created as
topic_name/command_name
-
register (required)
Modbus register address in the form of
<network_name>.<slave_id>.<register_address>
Where
register_address
is a decimal, octal or hexadecimal string (formally everythingstd::stoi(3)
with base=0 accepts). -
register_type (required)
Modbus register type: coil, input, holding
M2MGateway expects mqtt data as UTF-8 string value. It is converted to u_int16 and written to modbus register.
The state sections defines how to publish modbus data to MQTT broker. State can be mapped to a single register, an unnamed and a named list of registers. Following table shows what kind of output is generated for each type:
Value type | Default output |
---|---|
single register | u_int16 register data as string |
unnamed list | JSON array with u_int16 register data as string |
named list | JSON map with values as u_int16 register data as string |
It is also possible to combine and output an unnamed list of registers as a single value using converter. See converters section for details.
-
refresh
Overrides mqtt.refresh for this state topic
-
response_timeout (optional)
Overrides modbus.response_timeout when reading state registers
-
response_data_timeout (optional)
Overrides modbus.response_data_timeout when reading state registers
When state is a single modbus register value:
-
name
The last part of topic name where value should be published. Full topic name is created as
topic.name/state.name
-
register (required)
Modbus register address in the form of <network_name>.<slave_id>.
-
register_type (required)
Modbus register type: coil, input, holding
-
converter (optional)
The name of function that should be called to convert register u_int16 value to MQTT UTF-8 value. Format of function name is
plugin name.function name
. See converters for details.
The following examples show how to combine name, register, register_type, and converter to output different state values:
- single value
```
state:
name: mqtt_val
register: net.1.12
register_type: coil
```
- unnamed list
```
state:
name: mqtt_list
registers:
- register: net.1.12
register_type: input
- register: net.1.13
register_type: input
```
- multiple registers converted to single MQTT value
```
state:
name: mqtt_combined_val
converter: std.int32
registers:
- register: net.1.12
register_type: input
- register: net.1.13
register_type: input
```
- named list (map)
```
state:
registers:
- name: humidity
register: net.1.12
register_type: input
# optional
converter: std.divide(100,2)
- name: temp1
register: net.1.13
register_type: input
```
In all of above examples refresh, response_timeout and response_data_timeout can be added at any level to set different values to whole list or a single register.
For every state topic there is another availability topic defined by default. If all data from modbus registers needed for state is read without errors then by default value "1" is published. If there is any network or device error when polling register data value "0" is published. This is the default behavoiur if availability section is not defined.
Availablity flag is always published before state value.
Availability section extends this default behaviour by defining a single or list of modbus registers that should be readed to check if state data is valid. This could be i.e. some fault indicator or hardware switch state.
Configuration values:
-
name (required)
The last part of topic name where availability flag should be published. Full topic name is created as
topic.name/availability.name
-
register (required)
Modbus register address in the form of <network_name>.<slave_id>.
-
register_type (required)
Modbus register type: coil, input, holding
-
available_value (optional, default 1)
Expected u_int16 value readed from availablity register when availablity flag should be set to "1". If other value is readed then availability flag is set to "0".
register, register_type and available_value can form a list when multiple registers should be readed.
Data readed from modbus registers is by default converted to string and published to MQTT broker. To combine multiple modbus registers into single value, use mask to extract one bit or perform some simple divide operations a converter can be used.
Converter functions are defined in libraries dynamically loaded at startup. M2MGateway contains std library with basic converters ready to use:
-
divide
Arguments:
- divider (required)
- precision (optional)
Divides modbus value by divder and rounds to (precision) digits after the decimal.
-
int32
Arguments: none
For unnamed list. Combines two modbus registers into one 32bit value. The first register holds bits 17-32, the second holds 1-16.
-
bitmask
Arguments:
- bitmask in hex (default "0xffff")
Applies a mask to value readed from modbus register.
Converter can be added to modbus register in state section.
When state is a single modbus register:
state:
register: device1.slave2.12
register_type: input
converter: std.divide(10,2)
When state is a single MQTT value combined from multiple modbus registers:
state:
converter: std.int32()
registers:
- register: device1.slave2.12
register_type: input
- register: device1.slave2.13
register_type: input
Custom converters can be added by creating a C++ dynamically loaded library with conversion classes. There is a header only libmodmqttconv that provide base classes for plugin and converter implementations.
Here is a minimal example of custom conversion plugin with help of boost dll library loader:
#include <boost/config.hpp> // for BOOST_SYMBOL_EXPORT
#include "libmodmqttconv/converterplugin.hpp"
class MyConverter : public IStateConverter {
public:
//called by modmqttd to set coverter arguments
virtual void setArgs(const std::vector<std::string>& args) {
mShift = getIntArg(0, args);
}
// ModbusRegisters contains one register or as many as
// configured in unnamed register list.
virtual MqttValue toMqtt(const ModbusRegisters& data) const {
int val = data.getValue(0);
return MqttValue::fromInt(val << mShift);
}
virtual ~MyConverter() {}
private:
int mShift = 0;
};
class MyPlugin : ConverterPlugin {
public:
// name used in configuration as plugin prefix.
virtual std::string getName() const { return "myplugin"; }
virtual IStateConverter* getStateConverter(const std::string& name) {
if (name == "myconverter")
return new MyConverter();
return nullptr;
}
virtual ~MyPlugin() {}
};
// modmqttd search for "converter_plugin" C symbol in loaded dll
extern "C" BOOST_SYMBOL_EXPORT MyPlugin converter_plugin;
MyPlugin converter_plugin;
Compilation on linux:
g++ -I<path to mqmgateway source dir> -fPIC -shared myplugin.cpp -o myplugin.so
myconverter from this example can be used like this:
modmqttd:
loglevel: 5
converter_search_path:
- <mylugin.so dir>
converter_plugins:
- myplugin.so
mqtt:
objects:
- topic: test_topic
state:
name: test_val
register: device1.slave2.12
register_type: input
converter: myplugin.myconverter(1)
For more examples see libstdconv source code.