Nodered jsem instaloval v dockeru pomoci IOTstack. Instalace v dockeru má nějaká omezení - zejména co se týče nodu exec. Proto je velmi užitečné prostudovat dokumentaci k IOTstack k Node-RED.
Při rozběhnutí sériového portu v Noderedu jsem měl několik problémů. Nejprve mi Nodered psal, že se nepodařilo k /dev/ttyUSB0 připojit. To jsem vyřešil editací docker-compose.yml:
services: nodered: container_name: nodered build: context: ./services/nodered/. args: - DOCKERHUB_TAG=latest - EXTRA_PACKAGES= restart: unless-stopped user: "0" environment: - TZ=Etc/UTC ports: - "1880:1880" volumes: - ./volumes/nodered/data:/data - ./volumes/nodered/ssh:/root/.ssh - /var/run/docker.sock:/var/run/docker.sock - /var/run/dbus/system_bus_socket:/var/run/dbus/system_bus_socket devices: - "/dev/ttyUSB0:/dev/ttyAMA0" - "/dev/vcio:/dev/vcio" - "/dev/gpiomem:/dev/gpiomem"
Púvodně bylo v devices tohle: /dev/ttyAMA0:/dev/ttyAMA0. Do Noderedu jsem chtěl ale posílat ttyUSB0, tak jsem to změnil na /dev/ttyUSB0:/dev/ttyAMA0 a v dockeru budu přistupovat k zařízení /dev/ttyAMA0.
Další problém, který se objevil byl, že když jsem nastavil nodered, tak havaroval s hlášením segmentatin fault vždycky, když se měl připojit k /dev/ttyAMA0. Takže se docker neustále restartoval.
V této diskusi jsem našel pomoc.
Nejprve jsem se připojil do dockeru s noderedem.
docker exec -it nodered bash
Dále jsem zkusil vytvořil soubor test.js s tímto obsahem:
const { SerialPort } = require("serialport"); var port = new SerialPort({path: '/dev/ttyAMA0', baudRate: 115200});
A otestovat spuštění:
bash-5.1# node test.js
Problém se pořád projevoval. Když jsem podle doporučení provedl rebuild pluginů, testovací příkaz prošel a modul opravdu začal fungovat.
bash-5.1# npm rebuild --build-from-source
Najednou v logu noderedu vidím namísto segfault tohle:
3 Jan 16:50:22 - [info] [serialconfig:2268525fcaad5010] serial port /dev/ttyAMA0 opened at 115200 baud 8N1
K vyčítání dat z mého zařízení na sériovém portu z Noderedu mi posloužil tento návod.
Takto vypadá jednoduché propojení v Noderedu:
A takto jsem musel nakonfigurovat sériovou linku pro mé zařízení:
Samotnou zprávu pak posílám v json formátu:
Tutoriál popisuje jak vytvořit grafickou nástěnku pro Node RED.
Při zabezpečení jsem čerpal z manuálu Securing Node-Red with SSL and Username Authentication.
Nejprve je potřeba vytvořit certifikáty. Protože jsem nechtěl vytvářet další certifikační autoritu, využil jsem certifikáty, které mi připravil skript v dokumentu Mosquitto.
Skript mi vytvořil certifikát server-key.pem a server-cert.pem. Jen pro úplnost. Node RED i Mosquitto mi běží na jednom serveru. Proto si mohu dovolit použít stejné certifikáty. Pokud by každá služba běžela na jiném serveru, musel bych vygenerovat nové certifikáty pro konkrétní IP adresu.
Poté jsem editoval konfigurační soubor settings.js pro Node RED ve kterém jsem zadal absolutní cesty k certifikátům a zapnul vyžadování HTTPS. Bez absolutních cest nemohl Node RED najít certifikáty:
https: function() { // This function should return the options object, or a Promise // that resolves to the options object return { key: require("fs").readFileSync('/data/server-key.pem'), cert: require("fs").readFileSync('/data/server-cert.pem') } }, requireHttps: true,
Nejprve jsem si vygeneroval nové zaheshované heslo:
docker exec -it nodered bash bash-5.1# node-red-admin hash-pw Password: $2a$08$zZWtXTja0fB1pzD4sHCMyOCMYz2Z6dNbM6tl8sJogENOMcxWV9DN.
Poté v souboru settings.js odkomentuju následující řádky a nahradím hash hesla:
adminAuth: { type: "credentials", users: [{ username: "admin", password: "$2a$08$zZWtXTja0fB1pzD4sHCMyOCMYz2Z6dNbM6tl8sJogENOMcxWV9DN.", permissions: "*" }] },
Restartuju kontejner a při startu se mi už ke konfiguraci načítá přihlašovací obrazovka.
Protože Node RED běží v kontejneru, tak se všechny příkazy spustí uvnitř. V manuálu je popsáno, že když chci spouštět příkazy mimo kontejner, tak je řešení skrze SSH. Abych nemusel pokaždé vyplňovat heslo, vygeneruji klíč, kterým se přes ssh automaticky autentizuji.
Nejprve je potřeba na straně docker kontejneru spusti následující:
docker exec -it nodered ssh-keygen -q -t ed25519 -C "Node-RED container key-pair" -N ""
Poté přenést klíč na raspberry, na kterém budu chtít povolit, aby se Node RED automaticky připojil.
docker exec -it nodered ssh-copy-id myrpiuser@192.168.1.20
A nyní mohu otestovat, zda to funguje. Spustím v kontejneru v Node RED příkaz, který vypíše obsah adresáře mého Raspberry.
docker exec -it nodered ssh myrpiuser@192.168.1.20 ls -l
Pokud nechci zadávat dlouhé názvy serveru, mohu si vytvořit v kontejneru konfigurační soubor pro ssh. Soubor vytvořím v kontejneru v /root/.ssh
host iot-dev hostname 192.168.1.20 user myrpiuser IdentitiesOnly yes IdentityFile /root/.ssh/id_ed25519
Pro jistotu upravit práva:
chmod 644 config sudo chown root:root config
No a nyní mohu volat ssh příkazy jednodušeji.
docker exec -it nodered ssh iot-dev ls -l
A tady ukázka z Node RED:
Chtěl jsem využít node node-red-node-mysql pro ukládání dat do databáze na hostovaném počítači. Situace se má tak, že na počítači mi běží MySQL databáze, která není v Dockeru. Je to z historických důvodů - ještě jsem neznal Docker.
V Dockeru mi ale běží Node RED a chci, aby přistupoval k localhostu hostnovaného počítače, na kterém naslouchá MySQL databáze.
První překážka k tomu, aby se Node RED dokázal připojit k databázi je, že pokud zadám v Node RED localhost, tak se bude snažit připojit na port MySQL databáze v Node RED, kde žádná databáze neběží. O mém problému popisuje tato diskuse.
Zkoušel jsem nejdříve, aby MySQL naslouchala na IP adrese síťového bridge Node RED, ale pak bych musel vyřadit naslouchání MySQL na localhostu. A to jsme nechtěl. Proto jsem zvolil variantu, kdy kontejner v Node RED dostane k dispozici localhost hostitelského počítače. Není to úplně doporučované řešení, nicméně aktuálně mi to problém řeší a tím, že Node RED také může běžet pouze na localhostu, tak mi to až tak nevadí.
Konfigurace Node RED pro docker-composer tedy vypadá takto (přidán parametr network_mode: host):
version: '3.6' services: nodered: container_name: nodered ports: - '127.0.0.1:1880:1880' environment: - LANG=cs_CZ.UTF-8 - TZ=Europe/Prague volumes: - './volumes/nodered/data:/data' image: nodered/node-red network_mode: host
To mi pomohlo, aby byla databáze dostupná. Pak jsem ale narazil na to, že uživatel neměl do databáze přístup. Setkal jsem se s hlášením: Error: ER_ACCESS_DENIED_ERROR: Access denied for user 'root'@'localhost' (using password: YES)
Z hostitelského počítače jsem problém s připojením neměl, takže to byl problém z kontejneru s Node RED. Někdo přede mnou měl také podobný problém: Node JS and mysql not connecting (client does not support authentication protocol requested by server consider upgrading mysql client) a vytvořil skript v kontejneru, který posloužil i mě na odhalení problému:
const mysql = require('mysql2'); var connection = mysql.createConnection({ host : 'localhost', user : 'root', port : '3306', password : 'test12310', database : 'fys' }); connection.connect((err) => { if(!err) console.log('Database is geconnect!'); else console.log('Database connectie niet gelukt! : '+ JSON.stringify(err, undefined,2)); });
Skript jsem pak spustil v kontejneru:
node jmenoskriptu.js
Nakonec jsem přišel na to, že problém byl v MySQL databázi u uživatele. Vytvořil jsem uživatele s omezeným oprávněním přístupu do databáze. Namísto uživatele mujzivatel@localhost jsem měl vytvořil uživatele mujuzivatel@%. Tady se promítá to, že nepřistupuji jako uživatel z localhostu, ale kontejneru.
V mém případě jsem u modulu zjistil ještě jednu zradu. Pro každý příkaz, který jde na nod MySQL, se vytvoří zvlášť připojení do MySQL databáze. Protože mám počet připojení do databáze limitovaný, tak při velkém počtu připojení mi databáze spadla.
Dalším průzkumem jsem zjistil, že v pluginu je max connection nastaven na 50. Například tady je o tom diskuse.
Pro vypsání všech uživatelů a procesů v databázi slouží dotaz:
SHOW processlist;
V kódu pluginu jsem vyčetl tento řádek:
connectionLimit: RED.settings.mysqlConnectionLimit || 50,
Řádek říká, že pokud je nastavená proměnná RED.settings.mysqlConnectionLimit v Node RED, tak vezme číslo z ní a pokud nastavená není, tak defaultní connectionLimit je 50.
Řešením pro mě bylo nastavit proměnnou v Node RED v souboru settings.js:
module.exports = { // tady přidávám proměnnou pro limit spojení s databází // přidání musí být za definicí module.exports = { mysqlConnectionLimit: 5, }
Toto video popisuje práci s pluginem node-red-contrib-ui-svg, který vypadá poměrně zajímavě právě na tvorbu animací v NodeRED.
Nabízí se připravit si v Inkscape SVG grafiku a pak jí v NodeRED dodělat „animace“.
Zdá se, že je pouze pro 64 bit verzi operačního systému. Takže minimálně RPi4 a 64 bit verzi.