NEWS
MQTT Bluetooth BLE Anwesenheitserkennung mit ESP32
-
@GiuseppeS Muss nochmal blöd fragen, weil ich grad irgendwie auf dem Schlauch stehe. Zum testen hatte ich mir die MQTT-Messages an einen Mosquitto-Testserver geschickt, den ich mit FHEM abfrage. Dort bekam ich für jedes gefundene Gerät einen eigenen Datenpunkt und konnte direkt darauf reagieren.
Nun wollte ich mir den Umweg über FHEM sparen und die Daten direkt an den MQTT-Broker in iobroker schicken, der sowieso läuft. Die Daten kommen auch an, aber leider habe ich nur einen Datenpunkt, der alle enthält:[{"manuData": "8f3a109e405c6fe27c23448103fa010ff3f11000","mac": "44:23:7c:e2:6f:5e","rssi": -86}, {"manuData": "8f3a1055a08b33dfce55c8103f545ff3f11000","mac": "5c:e5:0c:df:33:8d","rssi": -82}, {"manuData": "8f3a109e40d279bc190648103fd46ff3f11000","mac": "64:90:c1:9b:27:0f","rssi": -83}, {"manuData": "8f3a109e40522da2c190648103fa47ff3f11000","mac": "64:90:c1:a2:2d:54","rssi": -73}, {"manuData": "00789ce77308b0a2c190648103f748ff3f11000","mac": "78:9c:e7:07:30:8b","rssi": -80}, {"manuData": "4c0215fda5693a4e24fb1afcfc6eb7647825275165c1","mac": "dc:0d:7f:05:26:47","rssi": -83}, {"manuData": "0000e77308b0a2c190648103ffcaff3f11000","mac": "e8:f2:e2:cd:2f:05","rssi": -68}]
Wie kann ich denn da auf ein ebstimmtes Gerät triggern? Also, wenn es Daten schickt, setze einen Datenpunkt "Anwesenheit" auf true, wenn keine Daten kommen, setzte ihn auf false. Ich habe auch schon mehrere Scripte hier im Forum getestet, die JSON-Strings in einzelne Datenpunkte aufteilen, aber irgendwie bin ich zu doof. Muss glaub doch noch Javascript lernen...
Gruss, Jürgen
-
@wildbill
Ich mache das wie folgt:
In meinem Skript ist die MAC oder die Manu Data notiert und in dem JSON suche ich danach. Das Ergebnis, wenn vorhanden, schreibe ich in ein separates State extra für dieses Gerät. Ist in Javascript geschrieben. Könnte ich auch zur Verfügung stellen.Bzgl JSON musste ich zu Beginn auch viel Try&Error durchmachen, inkl Google. Aber ich sag mal so: hier ist es eine klare kleine Aufgabe. Mit einer entsprechenden Vorlage sollte man klar kommen.
-
@giuseppes Ja gerne. Ich tippsel mir gerade die Finger wund und versuche aus verschiedenen Scripten, die ich hier im Forum zum Thema JSON parsen finde, was zusammenzustupfen, aber irgendwie will es nicht so ganz. Wenn Du mir da einen Denkanstoß geben könntest, würde ich das sicher angepasst bekommen.
Mir fehlen halt die Grundlagen der Javascript-Befehle und dann wird es da schon schwer. Mit Blockly fühle ich mich sehr wohl, aber das ist hier fehl am Platz.
Worin ich aber recht gut bin, ist Scripte anderer dann zu verstehen und für meine Zwecke anzupassen.Gruss, Jürgen
-
@wildbill
Geht klar, dann setze ich mich gleich mal an den PC. Nur vorab als Tip: Es ist eigentlich ein Array mit mehreren JSON Objekten. -
@giuseppes Das habe ich ziemlich am Anfang schon festgestellt. Erster Schritt war, den Inhalt des Datenpunktes mal in einen Online JSON Parser zu schieben und zu schauen, wie die Struktur genau ist. Array machte es dann für mich nicht leichter. Das meiste, was ich gefunden habe, waren JSON ohne Array und meist noch mit fester Struktur. Hier im Array ändern sich ja auch immer mal die Nummern der einzelnen Geräte, je nachdem, was ankommt und gerade nicht gehört wird...
Gruss, Jürgen
-
// erstelle States für Zielwerte createState("VIS.bleScan.Pino.RawWemos"); createState("VIS.bleScan.Pino.AvgWemos"); // nachfolgend kann auch manuData statt MAC eingetragen werden // dann müsste unten der Vergleich auf manuData statt mac erfolgen // z.B. val[dev].manuData.includes(pino) statt val[dev].mac.includes(pino) let pino = "ed:4b:90:40:e7:ba"; // für Berechnung gleitender Mittelwert let arrPinoWemos = [0,0,0,0,0,0]; on({id: "mqtt-client.0.bleScan.bett_192_168_192_49", change: "ne"}, function (obj) { // Array in val schreiben let val = JSON.parse(obj.state.val); // Array durchlaufen for ( let dev=0; dev < val.length; dev++){ //console.log(val[dev].mac + " | " + val[dev].rssi + " | " + val[dev].manuData); // Wenn MAC des betrachteten Array Elements oben definierte MAC enthält... if( val[dev].mac.includes(pino) ) { // Schreibe RSSI Wert direkt in das RAW State (oben definiert) setState("javascript." + instance + ".VIS.bleScan.Pino.RawWemos", val[dev].rssi); // ab hier gleitender Mittelwert über letzte 6 Elemente (kann oben auch größer definiert werden) arrPinoWemos.shift(); arrPinoWemos.push(val[dev].rssi); let sum = arrPinoWemos.reduce((a, b) => a + b, 0); let avg = (sum / arrPinoWemos.length) || 0; setState("javascript." + instance + ".VIS.bleScan.Pino.AvgWemos", Math.round(avg)); } }; });
Habe ergänzende Kommentare hinzugefügt. Sollte hoffentlich selbsterklärend sein. Wenn nicht, einfach nachfragen
-
@giuseppes Perfekt, läuft.
Vielen vielen Dank. Da hätte ich noch eine Weile rumgemacht. Ich glaube auch, ich habe im Array eine Ebene übersehen gehabt. Aber nun kommen die Daten einwandfrei.Gruss, Jürgen
-
@wildbill
Gern geschehen -
@giuseppes Habe heute ein kleines Problem festgestellt. Wenn ich den MQTT-Adapter im iobroker neu starte, dann verbindet sich der ESP nicht mehr automatisch. Es kommt dann im Log
Client [bleScan/oben_192_168_100_94] has invalid password(undefined) or username(undefined)
Scheint, als würde er keine neue Verbindung aushandeln. Ich muss ihn dann einmal neu starten (aus- und wieder einstecken) und schon läuft er wieder. Bevor ich ihn jetzt umflashe und den Umweg über meinen Mosquitto nehmen lasse, bringt mir das überhaupt was? Der ist auch mit user/password versehen. Handelt der das anders?
Gruss, Jürgen
-
@wildbill
Ich verwende im Heimnetz kein User/Pass über mqtt. Außerdem nutze ich auf meinem PC direkt Mosquitto, im ioBroker nur den mqtt-Client. Dass sich der esp32 nicht neu verbindet, ist mir bisher nicht aufgefallen. Es ist in den letzten Monaten 1 mal passiert, dass er nicht verbunden war. Grund war unklar. Jedenfalls hängt es nicht mit den PC Neustarts zusammen, die finden allein wegen Backup min. 1/Woche statt. -
@giuseppes Ja, es wird mit user/password zusammenhängen. Komischerweise hat nur der ESP mit dem blescan das Problem. Ein paar andere, unter Anderem mit ESPEasy verbinden sofort wieder.
Aber, ich frage mich gerade auch, warum ich im MQTT vom iobroker user/pass drin habe. Der ist nicht im Internet, da hängt nur der Mosquitto. Dann nehme ich es im iobroker einfach raus und sollte passen. Danke.Gruss, Jürgen
EDIT: Das wars auch. User password raus und sofort haben sich alle Geräte inclusive dem ESP mit blescan wieder verbunden. Wenn nur alles so einfach wäre...
-
Hallo zusammen, da ich nicht programmieren kann wollte ich am Wochenende mal versuchen dies hier umzusetzen:
https://docs.openmqttgateway.com/
Kennt das villeicht jemand hier? -
Habe jetzt mehrere ESP 32 mit OpenMQTTGateway bestückt und drei Mi Band 5 auf die withelist gesetzt.
Funktioniert soweit erstmal gut.
Habt ihr einen Tipp für mich, wie die Zuordnung zu den Zimmern gemacht wird.Außerdem muss ich an zwei der drei handys Bluetooth ausschalten, damit die Bänder überhaupt gesehen werden.
Wie habt ihr das Problem gelöst?
Habe testweise Nachdem die Bander von de ESP32 erkannt werden, Bluetooth wieder eingeschaltet und die Bänder werden weiterhin erkannt.Ach ja, falls es jemanden interessiert, es werden einzelne Datenpunkte angelegt:
-
@giuseppes said in MQTT Bluetooth BLE Anwesenheitserkennung mit ESP32:
// erstelle States für Zielwerte createState("VIS.bleScan.Pino.RawWemos"); createState("VIS.bleScan.Pino.AvgWemos"); // nachfolgend kann auch manuData statt MAC eingetragen werden // dann müsste unten der Vergleich auf manuData statt mac erfolgen // z.B. val[dev].manuData.includes(pino) statt val[dev].mac.includes(pino) let pino = "ed:4b:90:40:e7:ba"; // für Berechnung gleitender Mittelwert let arrPinoWemos = [0,0,0,0,0,0]; on({id: "mqtt-client.0.bleScan.bett_192_168_192_49", change: "ne"}, function (obj) { // Array in val schreiben let val = JSON.parse(obj.state.val); // Array durchlaufen for ( let dev=0; dev < val.length; dev++){ //console.log(val[dev].mac + " | " + val[dev].rssi + " | " + val[dev].manuData); // Wenn MAC des betrachteten Array Elements oben definierte MAC enthält... if( val[dev].mac.includes(pino) ) { // Schreibe RSSI Wert direkt in das RAW State (oben definiert) setState("javascript." + instance + ".VIS.bleScan.Pino.RawWemos", val[dev].rssi); // ab hier gleitender Mittelwert über letzte 6 Elemente (kann oben auch größer definiert werden) arrPinoWemos.shift(); arrPinoWemos.push(val[dev].rssi); let sum = arrPinoWemos.reduce((a, b) => a + b, 0); let avg = (sum / arrPinoWemos.length) || 0; setState("javascript." + instance + ".VIS.bleScan.Pino.AvgWemos", Math.round(avg)); } }; });
Habe ergänzende Kommentare hinzugefügt. Sollte hoffentlich selbsterklärend sein. Wenn nicht, einfach nachfragen
Ich versuche gerade die Anwesenheitserkennung nachzubauen. Leider bekomme ich das Script von GiuseppeS nicht ans Laufen... Die Datenpunkte, die das Script anlegt, bleiben leider leer. Wenn ich das Script debugge erhalten ich den Fehler "Cannot redeclare block-scoped variable".
Habt ihr eine Idee, wo der Fehler liegt?! -
Anscheinend wurde die Variable pino bereits vorher deklariert? Hast du das Skript aus Versehen doppelt eingefügt?
-
@giuseppes Danke für deine schnelle Rückmeldung.
Nein ich habe das Skript lediglich einmal eingefügt. Ich habe pino nun mal gegen Alf ausgetauscht und bekommen leider den gleichen Fehler.
-
@spezialtrick
Dann poste hier bitte dein komplettes Skript. Kann mir noch vorstellen dass woanders Fehler sind und das hier ein Folgefehler. -
So sieht mein Skript aus:
// erstelle States für Zielwerte createState("Präsenzmelder.Miro.Raw"); createState("Präsenzmelder.Miro.Avg"); // nachfolgend kann auch manuData statt MAC eingetragen werden // dann müsste unten der Vergleich auf manuData statt mac erfolgen // z.B. val[dev].manuData.includes(miro) statt val[dev].mac.includes(miro) let miro = "7C:2F:80:C7:FB:1B"; // für Berechnung gleitender Mittelwert let arrMiroWemos = [0,0,0,0,0,0]; on({id: "mqtt.0.SmartHome.Küche.Präsenzmelder_192_168_188_75", change: "ne"}, function (obj) { // Array in val schreiben let val = JSON.parse(obj.state.val); // Array durchlaufen for ( let dev=0; dev < val.length; dev++){ //console.log(val[dev].mac + " | " + val[dev].rssi + " | " + val[dev].manuData); // Wenn MAC des betrachteten Array Elements oben definierte MAC enthält... if( val[dev].mac.includes(miro) ) { // Schreibe RSSI Wert direkt in das RAW State (oben definiert) setState("javascript." + instance + ".Präsenzmelder.Miro.Raw", val[dev].rssi); // ab hier gleitender Mittelwert über letzte 6 Elemente (kann oben auch größer definiert werden) arrMiroWemos.shift(); arrMiroWemos.push(val[dev].rssi); let sum = arrMiroWemos.reduce((a, b) => a + b, 0); let avg = (sum / arrMiroWemos.length) || 0; setState("javascript." + instance + ".Präsenzmelder.Miro.Avg", Math.round(avg)); } }; });
-
@spezialtrick
Ich kann mir die Fehler nicht erklären. Allerdings würde ich immer davon abraten, deutsche Umlaute in Programm Codes zu verwenden. Zwar ist es hier ein "String". Aber ioBroker muss später damit klarkommen und entsprechende States anlegen. Habe gerade explizit auf fehlerhafte Semikolon o.ä. geachtet, konnte aber nichts feststellen.Dieses WE wollte ich in ioBroker noch was machen. Werde dein Skript mal übernehmen und testen.
-
Ich teste es heute Abend nochmals ohne Umlaute und gebe dir eine Rückmeldung. Kann es sein, dass ich an irgendeiner Stelle im IOBroker eine fehlende Konfiguration habe? Ich bin erst kürzlich von FHEM umgestiegen, sodass mein IOBroker noch sehr jungfräulich ist.