Dashboards for IoT sensors, part 2/2

This article is a follow-up of part 1.

Node-Red: record data

Fetch MQTT messages…

As introduced, we’re going to record data coming up on the broker by using Node-Red on a PostgreSQL database. If setting-up Node-Red is successfull, you can reach it on port 1880 (e.g. raspberrypi.local:1880). Then over its user interface, you need to set-up: And that’s it! An MQTT node already exists. By placing a MQTT IN node on the current flow tab, you can configure the broker after double-clicking on it as you did previously in python. Then edit the node itself with the correct channel name under Topic to listen to this sensor. Node Red works with scopes. There are three:
  • msg: this scope points to objects shared among all nodes that are linked with each other. The object that interests us here is msg.payload, it holds the message coming from your source MQTT IN node.
  • flow: this scope points to all objects you wish to share with all nodes on that tab, whether they’re linked or not.
  • global: as its name indicates it, this scope is shared with all flows, thence all tabs.
Here is a flow using the office sensor as example, zigbee2mqtt/TH_bureau, which outputs to the debugger (printed on the right):
The output message from the MQTT canal is held in a javascript object.
As Output, use parsed JSON object to receive a javascript object as shown on the debugger. We find our 6 features: battery level, humidity, link quality, air pressure, temperature and the battery voltage in mV. Our objective here is to get the same message format for each similar sensor. We can then use a unique flow for all sensors: fewer nodes means less code, so also fewer bugs and less maintenance! Link quality is not always sent by the Xiaomi sensors. So we need to add it should it be missing. We can also add a friendly name in the msg rather than using the long Topic. We use a switch node to detect whether linkquality is lacking and change nodes to add missing elements.
The name object is added, linkquality as well if missing

…and insert them in SQL tables

As indicated in the introduction of the preceding article, we have created a th schema within which we will add a table per sensor. Our objective in our flow is to create the tables if they’re missing, then insert the new data. Add a PostgreSQL node to your flow. By double clicking, you can here as well indicate the server authentication parameters. You can also deactivate the Output function: we are only inserting. Between the MQTT IN node and the postgresql node, you need to add two nodes:
  • a switch node within which we inject the SQL query parameters inside the msg.data object.
  • a template node within which we write our SQL query and put it in the msg.query object.
In the first node, you create an empty object then declare your parameters.
An empty object is created, then we add sensor data within it
Behind msg.data.sensors is:
{"name":msg.name,
"battery": msg.payload.battery,
"humidity":msg.payload.humidity,
"pressure":msg.payload.pressure,
"temperature":msg.payload.temperature,
"voltage":msg.payload.voltage,
"linkquality":msg.payload.linkquality,
"msg_ts":$now()
}
Note msg_ts. It has been added to capture the datetime. We could have also used the default now value of PostgreSQL, but I prefer capturing a time as close as possible to the emitting event. In the second node, select the msg.query property, use the Mustache template format, select text as output format and write the following SQL query:
CREATE TABLE IF NOT EXISTS th.{{{data.sensors.name}}}(
ID SERIAL PRIMARY KEY NOT NULL,
battery INT,
humidity REAL NOT NULL,
pressure REAL,
temperature REAL NOT NULL,
voltage INT,
linkquality INT,
msg_ts TIMESTAMP NOT NULL
);
INSERT INTO th.{{{data.sensors.name}}}(battery,
humidity,
pressure,
temperature,
voltage,
linkquality,
msg_ts)
VALUES ({{{data.sensors.battery}}},
{{{data.sensors.humidity}}},
{{{data.sensors.pressure}}},
{{{data.sensors.temperature}}},
{{{data.sensors.voltage}}},
{{{data.sensors.linkquality}}},
'{{{data.sensors.msg_ts}}}');
COMMIT;
The first part of the query is explicit: if there is no table named after the sensor, it should be created. We find our 6 features, the datetime and a unique ID. The second part of the request is the insert. Timestamp is between single quotes as it is a string, not an int. Now all you need to do is to put everything together! We can link several channels to the same switch node after giving the sensor names as can be seen in the image herebelow.
‣ Here is an example to import (Sandwich menu → import)
[
{
"id": "150ff0cd.ac1c9f",
"type": "tab",
"label": "MQTT to SQL Example",
"disabled": false,
"info": ""
},
{
"id": "502ce981.c3c718",
"type": "debug",
"z": "150ff0cd.ac1c9f",
"name": "",
"active": true,
"tosidebar": true,
"console": false,
"tostatus": false,
"complete": "true",
"targetType": "full",
"statusVal": "",
"statusType": "auto",
"x": 1870,
"y": 240,
"wires": []
},
{
"id": "6028372b.6d199",
"type": "change",
"z": "150ff0cd.ac1c9f",
"name": "set name to roomname",
"rules": [
{
"t": "set",
"p": "name",
"pt": "msg",
"to": "room",
"tot": "str"
}
],
"action": "",
"property": "",
"from": "",
"to": "",
"reg": false,
"x": 910,
"y": 200,
"wires": [
[
"a73a032f.b631b"
]
]
},
{
"id": "a73a032f.b631b",
"type": "switch",
"z": "150ff0cd.ac1c9f",
"name": "",
"property": "payload.linkquality",
"propertyType": "msg",
"rules": [
{
"t": "null"
},
{
"t": "nnull"
}
],
"checkall": "true",
"repair": false,
"outputs": 2,
"x": 1050,
"y": 200,
"wires": [
[
"e37391f1.76d83"
],
[
"5e46f449.29575c"
]
]
},
{
"id": "e37391f1.76d83",
"type": "change",
"z": "150ff0cd.ac1c9f",
"name": "",
"rules": [
{
"t": "set",
"p": "payload.linkquality",
"pt": "msg",
"to": "NULL",
"tot": "str"
}
],
"action": "",
"property": "",
"from": "",
"to": "",
"reg": false,
"x": 1240,
"y": 160,
"wires": [
[
"5e46f449.29575c"
]
]
},
{
"id": "5e46f449.29575c",
"type": "change",
"z": "150ff0cd.ac1c9f",
"name": "Query parameters",
"rules": [
{
"t": "set",
"p": "data",
"pt": "msg",
"to": "{}",
"tot": "json"
},
{
"t": "set",
"p": "data.sensors",
"pt": "msg",
"to": "{\"name\":msg.name,\t \"battery\": msg.payload.battery,\t \"humidity\":msg.payload.humidity,\t \"pressure\":msg.payload.pressure,\t \"temperature\":msg.payload.temperature,\t \"voltage\":msg.payload.voltage,\t \"linkquality\":msg.payload.linkquality,\t \"msg_ts\":$now()\t}",
"tot": "jsonata"
}
],
"action": "",
"property": "",
"from": "",
"to": "",
"reg": false,
"x": 1470,
"y": 220,
"wires": [
[
"687d29ae.4ff0f"
]
]
},
{
"id": "687d29ae.4ff0f",
"type": "template",
"z": "150ff0cd.ac1c9f",
"name": "INSERT TH DATA",
"field": "query",
"fieldType": "msg",
"format": "handlebars",
"syntax": "mustache",
"template": "\nCREATE TABLE IF NOT EXISTS th.{{{data.sensors.name}}}(\n ID SERIAL PRIMARY KEY NOT NULL,\n battery INT,\n humidity REAL NOT NULL,\n pressure INT,\n temperature REAL NOT NULL,\n voltage INT,\n linkquality INT,\n msg_ts TIMESTAMP NOT NULL\n );\n\n\nINSERT INTO th.{{{data.sensors.name}}}(battery,\n humidity,\n pressure,\n temperature,\n voltage,\n linkquality,\n msg_ts)\nVALUES ({{{data.sensors.battery}}}, \n {{{data.sensors.humidity}}},\n {{{data.sensors.pressure}}},\n {{{data.sensors.temperature}}},\n {{{data.sensors.voltage}}},\n {{{data.sensors.linkquality}}},\n '{{{data.sensors.msg_ts}}}');\n\nCOMMIT;",
"output": "str",
"x": 1670,
"y": 200,
"wires": [
[
"502ce981.c3c718",
"5fbb8de4.19e0e4"
]
]
},
{
"id": "3c8b18ee.62d62",
"type": "mqtt in",
"z": "150ff0cd.ac1c9f",
"name": "",
"topic": "",
"qos": "2",
"datatype": "auto",
"x": 640,
"y": 220,
"wires": [
[
"6028372b.6d199"
]
]
},
{
"id": "5fbb8de4.19e0e4",
"type": "digitaloak-postgresql-query",
"z": "150ff0cd.ac1c9f",
"name": "",
"inputs": 1,
"outputs": 1,
"x": 1870,
"y": 200,
"wires": [
[]
]
}
]

Panel: watch historical data

As we did for live data, we can build a Panel application to visualize sensor measurements held in the database. This time the Dashboard will be more classic and easier to implement. Herebelow a notebook describing step after step how to build it. Http iframes are not shown in https pages in many major browsers. Please read this post for details. As previously, you can use this notebook as code source for your application:
panel serve yournotebook.ipynb --port 5006 --address localhost --show
above command will open a webpage showing only the last element, called with the .servable() method. It would be smarter to clean the notebook a little as every cell will be executed upon calling above command. Or you could directly use the notebook available on GitHub.

Implement

You can now add this application to your raspberry pi service. Bokeh, the library behind Panel, let you serve several applications in parallel. It automatically generates a home webpage letting your users pick the application. Simply modify the ExecStart line from your home_on_bokeh.service file by adding the application (bold is the modification):
sudo nano /etc/systemd/system/home_on_bokeh.service
Description=Serve panel to access zigbee sensor data
After=multi-user.target
[Service]
Type=simple
ExecStart=/home/pi/.local/bin/panel serve /home/pi/home_on_bokeh/plotting_live_data.py /home/pi/home_on_bokeh/plotting_hist_data.py --port 80 --address yourrpiaddress
Restart=on-abort
User=pi
[Install]
WantedBy=multi-user.target
If you’ve used environment variables for authenticating to your database, you must add them:
sudo systemctl edit home_on_bokeh.service
[Service]
Environment="MQTT_ADDRESS=localhost"
Environment="MQTT_USER=pi"
Environment="MQTT_PASSWORD=verycomplexpassphrase"
Environment="SQL_ADDRESS=localhost"
Environment="SQL_USER=pi"
Environment="SQL_PASSWORD=unguessablepassword"
Environment="BOKEH_ALLOW_WS_ORIGIN=raspberrypi.local:80,192.168.10.10:80"
Now restart your service:
sudo systemctl restart home_on_bokeh.service
There you go! This second application is slower to start than the first one. On top of it, the poor RPi must run a Node-Red server and a postgresql server. Swaping to a RPi 4 will certainly help. But if all you do concerns domotic applications, I strongly suggest you to use Home Assistant and add the Grafana and InfluxDB modules to reach a similar result (well, without all the fun 😉 ).

Leave a Reply

Your email address will not be published. Required fields are marked *