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):
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.
…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.
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:
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)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
"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;",
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:
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):
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 ).