NEWS
[Gelöst] Helios KWL - Zugriff auf xml
-
Moin zusammen,
ich habe kürzlich hier schon mein Problem geschildert. Jetzt geht es um einen anderen Ansatz: Ich möchte die Daten der Helios aus den xmls, befindliche auf dem lokalen Webserver des Gerätes, auslesen.
Dazu nutze ich:
var request = require('request'); function doWork() { request('http://192.168.178.22/data/werte8.xml', function (error, response, body) { if (error) { log("Fehler beim Herunterladen: " + error); return; } const regex = /v00104<\/ID>\n<VA>(\d+.\d+)/gm; try { const atemp = regex.exec(body); createState("HeliosKWL.Aussentemperatur", atemp, true, { type: "number", unit: "°C", role: "value.temperature" }); } catch (e) { log("Fehler beim Auswerten: " + error); } }); }; schedule('*/5 * * * * *', doWork);
Dank an AlCalzone, dessen Code ich hier nutze
Angeblich sei es mit xml2js einfacher, aber das bekomme ich nicht hin. Wer sich damit auseinandersetzen mag, gerne2 Probleme:
-
Der lokale Zugriff auf die Helios ist passwortgeschützt. Kein Nutzername. Wie kann ich das Script so erweitern, dass es sich vorab auf einer anderen Seite einloggt und danach die o.g. xml verarbeitet?
-
Mir ist aufgefallen, dass ich die genannte Datei "werte8.xml" nicht im Browser aufrufen kann. Auch keine andere "werteX.xml". Die sind für mich nur per Debugger (Fiddler oder Mozilla) sichtbar und lesbar. Sonstige xmls aus derselben Quelle kann ich aber aufrufen. Kann das sein? Könnte das zu einem Problem werden? Wie kann ich damit umgehen?
Danke und Grüße!
=================================================
Edit: Titel angepasst=================================================
Edit 2: Um dem Suchenden weiteres Leid zu ersparen: Die Lösung verbirgt sich im Script von @KLVN und findet sich hier: https://github.com/KLVN/ioBroker_Helios-KWL=================================================
Edit 3: Aus dem o.g. Script hat @tombox einen Adapter gebaut, weitere Details hier: https://forum.iobroker.net/topic/47762/test-helios-kwl-v0-0-x -
-
OK, ich bin einen Schritt zurück gegangen und habe ersteinmal nur den Zugriff auf die xml probiert: Klappt. Zumindest mit anderen xmls vom Webserver der Helios. Denn für die "werte8.xml" bekomme ich mittels "request" einen 404.
Wenn ich nun per Debugger (hier Mozilla) statt einer GET-Anfraeg eine POST-Anfrage mit request-body "xml=/data/werte8.xml" stelle, dann bekomme ich meine xml auch im Browser angezeigt! Jetzt die Quizfrage: Irgendeine Chance, das in ein Script zu packen?
-
Nach viel herumexperimentieren und zusammenklauben von Code, auch hier aus dem Forum, habe ich eine Lösung gefunden:
Der Login läuft über einen post-request, ebenso die Abfrage der Daten aus den xml. Dazu werden jeweils konkrete Header- und Body-daten angefragt.
Ausgelesen werden die Werte mittels stumpfer Positionsbestimmung. Das klappt, weil Helios für jeden Wert eine einmalige Variable sowie feste Zeichenanzahl vorsieht.Zur besseren Übersicht ist unten nur die Aussentemperatur exemplarisch aufgeführt. Zur Erweiterung auf andere Werte müsste der request der Datenabfrage kopiert, die Summanden aus den Zeilen 48 + 50 angepasst und - falls Werte aus anderen xmls ausgelesen werden sollen - eine weitere Variable mit neuen Headerdaten angelegt und im neuen request ebenfalls angepasst werden.
Die Variablen zur Zuordnung der Werte sowie die Zeichenanzahl gibts direkt bei Helios.Falls jemand Vorschläge/Kommentare/Anregungen hat, freue ich mich über Input aller Art. Auch über Hinweise zu Denkfehlern.
Und wenn jemand dieses Gewurschtel in Form bringen möchte - nur zuvar position_anfang, result, position_ende; var request= require('request'); var Hlogin = {headers: { 'Host': '192.168.178.22', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Firefox/68.0', 'Accept': '*/*', 'Accept-Language': 'de,en-US;q=0.7,en;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Referer': 'http://192.168.178.22/', 'Content-Type': 'text/plain;charset=UTF-8', 'Content-Length': '15', 'DNT': '1', 'Connection': 'keep-alive', }, url: 'http://192.168.178.22/info.htm', body: 'v00402=PASSWORT', method: 'POST',}; var Hwerte8 = {headers: { 'Host': '192.168.178.22', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Firefox/68.0', 'Accept': '*/*', 'Accept-Language': 'de,en-US;q=0.7,en;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Referer': 'http://192.168.178.22/anzeig.htm', 'Content-Type': 'text/plain;charset=UTF-8', 'Content-Length': '20', 'DNT': '1', 'Connection': 'keep-alive' }, url: 'http://192.168.178.22/data/werte8.xml', body: 'xml=/data/werte8.xml', method: 'POST'}; //Datenpunkte erzeugen createState("HeliosKWL.Aussentemperatur"); //Login alle 5 Min setInterval(function() { request(Hlogin); }, 300000); //Datenabfrage alle 5 Sek setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00104') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Aussentemperatur', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000);
Viele Grüße!
Johannes -
Entschuldigt bitte, dass ich dieses alte Thema wieder ausgrabe aber ich habe versucht bei meiner Helios KWL die Werte auszulesen (klappt super, danke hierfür!) aber ich möchte auch die Lüfterstufe ändern können. Dazu habe ich den Teil für die Passwort Eingabe kopiert und setze dazu die Lüfterstufe bei "body" entsprechend.
var Hstufe = {headers: { 'Host': '192.168.178.127', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Firefox/68.0', 'Accept': '*/*', 'Accept-Language': 'de,en-US;q=0.7,en;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Referer': 'http://192.168.178.127/', 'Content-Type': 'text/plain;charset=UTF-8', 'Content-Length': '8', 'DNT': '1', 'Connection': 'keep-alive', }, url: 'http://192.168.178.127/info.htm', body: 'v00102=' + getState("Meine_Geräte.0.Lueftung").val, method: 'POST',};
Nun wird es leider ganz merkwürdig. Wenn ich die entsprechende Variable ändere, passiert bei der Lüftungsanlage manchmal nichts und manchmal ändert sich die Lüfterstufe. die neue Lüfterstufe entspricht aber oft nicht dem Wert den ich eingegeben habe. Ein Muster konnte ich hier leider noch nicht erkennen. Hat einer von euch eine Idee? Meine Javascript Kenntnisse sind leider nicht gerade gut. Eventuell habe ich hier auch nur eine Banalität falsch.
Edit: Ich muss wohl den Eintrag mit "\0" abschließen. Das hat aber leider auch noch keinen Erfolg gebracht. Auch habe ich mal mit der Content-Length gespeilt und dachte zwischenzeitlich Erfolg gehabt zu haben, war aber leider nicht von Dauer. Es ist mir absolut schleierhaft warum die Kiste ab und zu reagiert aber nicht das macht was ich will...
-
So, habe mich mal wieder mit dem Thema beschäftigt.
In der grafischen Webseite kann ich die Lüfterstufe nicht direkt eingeben, sondern nur auf + und - drücken
Die zugehörigen code Zeilen müssten diese hier sein:
<div id="div3" style="visibility:hidden;display:none;"> <input type="hidden" name="v00020" id="v00020" value="" disabled="disabled"/> <figure style="float:left;margin-left:5px;margin-right:5px;"><img src="images/minus.png" id="minus" onclick="clearInterval(interval);setstufe(document.getElementById('v00102'),-1,document.getElementById('v00020').value,4,1);" style="visibility:hidden;"></figure> <div id="akt" style="float:left;margin-top:20px;width:130px;"> <label for="v00102" id="lv00102">Lüfterstufe:</label> <input type="text" name="v00102" id="v00102" style="width:20px;margin-top:0px;margin-bottom:3px;"> <input type="hidden" name="v00001" id="v00001"> </div> <figure style="float:left;margin-left:5px;margin-right:5px;"><img src="images/plus.png" id="plus" onclick="clearInterval(interval);setstufe(document.getElementById('v00102'),1,document.getElementById('v00020').value,4,1);" style="visibility:hidden;"></figure>
Könntet ihr mir behilflich sein, wie ich das Skriptabändern muss, damit er mir die Lüfterstufe nach oben und unten ändert? Im Element v00102 steht die aktuelle Lüfterstufe und das Element v00020 enthält die minimal zulässige Lüfterstufe.
-
@chrisB bist du schon weitergekommen beim Setzen von Werten?
und könntest du bitte dein Script zum Auslesen der Werte veröffentlichen?Danke
-
Hier mal mein Skript für mehrere Daten aus meiner Helios Lüftung auszulesen.
var position_anfang, result, position_ende; var request= require('request'); var Hlogin = {headers: { 'Host': '192.168.178.xx', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Firefox/68.0', 'Accept': '*/*', 'Accept-Language': 'de,en-US;q=0.7,en;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Referer': 'http://192.168.178.xx/', 'Content-Type': 'text/plain;charset=UTF-8', 'Content-Length': '15', 'DNT': '1', 'Connection': 'keep-alive', }, url: 'http://192.168.178.xx/info.htm', body: 'v00402=Passwort', method: 'POST',}; var Hwerte8 = {headers: { 'Host': '192.168.178.xx', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:68.0) Gecko/20100101 Firefox/68.0', 'Accept': '*/*', 'Accept-Language': 'de,en-US;q=0.7,en;q=0.3', 'Accept-Encoding': 'gzip, deflate', 'Referer': 'http://192.168.178.xx/anzeig.htm', 'Content-Type': 'text/plain;charset=UTF-8', 'Content-Length': '20', 'DNT': '1', 'Connection': 'keep-alive' }, url: 'http://192.168.178.xx/data/werte8.xml', body: 'xml=/data/werte8.xml', method: 'POST'}; //Datenpunkte erzeugen createState("HeliosKWL.Aussentemperatur"); createState("HeliosKWL.Zulufttemperatur"); createState("HeliosKWL.Fortlufttemperatur"); createState("HeliosKWL.Ablufttemperatur"); createState("HeliosKWL.Zuluftdrehzahl"); createState("HeliosKWL.Abluftdrehzahl"); createState("HeliosKWL.Lüfterstufe"); createState("HeliosKWL.Filterwechsel"); createState("HeliosKWL.Betriebsart"); createState("HeliosKWL.Abluftfeuchte"); //Login alle 5 Min setInterval(function() { request(Hlogin); }, 300000); //Datenabfrage alle 5 Sek setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00104') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Aussentemperatur', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00105') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Zulufttemperatur', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00106') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Fortlufttemperatur', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00107') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Ablufttemperatur', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00348') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Zuluftdrehzahl', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00349') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Abluftdrehzahl', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00102') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Lüfterstufe', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v01033') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Filterwechsel', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00101') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 0; setState('HeliosKWL.Betriebsart', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000); setInterval(function() { request(Hwerte8, function (error, response, result) { position_anfang = result.indexOf('v00033') + 1; position_anfang = (typeof position_anfang == 'number' ? position_anfang : 0) + 16; position_ende = position_anfang; position_ende = (typeof position_ende == 'number' ? position_ende : 0) + 3; setState('HeliosKWL.Abluftfeuchte', (result.slice((position_anfang - 1), position_ende)), true, parseInt(0, 10), false); }); }, 5000);
Hier muss die Ip der Anlage und natürlich auch das Passwort eingetraen werden.
-
Nächster Schritt soll sein
"Partymodus einschlaten" + Zeiten erhöhen
und
"Filterwechsel" resetten -
@Ritschy2000 das Skript ist oben gepostet, bzw. hat Timmerx es auch gepostet. Ich hatte zwischenzeitlich aufgegeben, da ich weder die Zeit hatte mich detailliert in das Thema einzuarbeiten und keine Erfahrung mit den notwendigen Programmiersprachen habe. Die Reaktion des Skriptes beim Schreiben der Werte war einfach in keiner Weise nachvollziehbar, so dass ich mit try and error nicht mehr weitergekommen bin.
@Timmerx hast du es schon geschafft Werte an die KWL zu übergeben?
-
Nein das habe ich noch nicht geschafft.
-
@chrisB @Timmerx ich bekomme das Script leider nicht zum laufen, bekomme immer folgende Fehlermeldung:
javascript.0 2020-10-13 23:42:06.238 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Abluftfeuchte, state={"val":"tle>","ack":true}) - wurde nicht ausgeführt, während der Debug-Modus aktiv ist javascript.0 2020-10-13 23:42:06.232 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Betriebsart, state={"val":"t","ack":true}) - wurde nicht ausgeführt, während der Debug-Modus aktiv ist javascript.0 2020-10-13 23:42:06.230 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Filterwechsel, state={"val":"tle>","ack":true}) - wurde nicht ausgeführt, während der Debug-Modus aktiv ist javascript.0 2020-10-13 23:42:06.220 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Lüfterstufe, state={"val":"tle>","ack":true}) - wurde nicht ausgeführt, während der Debug-Modus aktiv ist javascript.0 2020-10-13 23:42:06.219 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Abluftdrehzahl, state={"val":"tle>","ack":true}) - wurde nicht ausgeführt, während der Debug-Modus aktiv ist javascript.0 2020-10-13 23:42:06.214 warn (20887) script.js.common.Java.Helios: setForeignState(id=javascript.0.HeliosKWL.Zuluftdrehzahl, state={"val":"tle>","ack":true}) - wurde nicht ausgeführt, während der
habe ich irgendwo etwas vergessen einzutragen?
Bin für jede Hilfe dankbar.
-
Es freut mich, dass hier noch andere mit einer Helios KWL unterwegs sind - da klinke ich mich gleich mal ein
Gestern konnte ich schon mal über eine direkte LAN-Verbindung, zwischen Laptop und Helios, mir die Steuerung ansehen und habe auch bisschen in den Code geguckt (der ist größtenteils auf deutsch, leicht verständlich und ab und zu gibt es auch Kommentare).Zum Thema Lüfterstufe einstellen: Die Funktion dafür im Code nimmt nicht die gewünschte Stufe als Argument, sondern die Differenz zur aktuellen Stufe. Wenn also Stufe 4 eingestellt ist und man Stufe 2 haben will, muss man "-2" übergeben.
Bis jetzt habe ich noch keinen konkreten Code schreiben können und auch noch nichts mit ioBroker versucht. Zuerst muss ich den Helios-Kasten ins Netzwerk bringen, doch der hängt leider ungünstig.
Habt ihr Helios über LAN direkt an den Router anschließen können oder eine Art WLAN-Brücke aufgebaut? Ich würde mir jetzt einen portablen Router holen und es so aufbauen:Helios (LAN) <=> Mini-Router (LAN) - Mini-Router (WLAN) <=> Fritzbox (WLAN)
EDIT: Hat alles so funktioniert, wie ich es vorhatte und jetzt ganz ich mich am Code probieren. Ich habe diesen Router genommen: https://www.amazon.de/dp/B00TQEX8BO/
-
@Ritschy2000 sagte in [Gelöst] Helios KWL - Zugriff auf xml:
habe ich irgendwo etwas vergessen einzutragen?
Bin für jede Hilfe dankbar.Klicke mal rechts auf den Schraubenschlüssel und deaktiviere "Debug".
-
Ich habe es jetzt geschafft, dass man den Lüfter auf eine bestimmte Stufe einstellen kann. Die Tage werde ich noch weitere Funktionen hinzufügen und den Code dann auch veröffentlichen.
-
Hier ist mein Code: https://github.com/KLVN/ioBroker_Helios-KWL
Läuft jetzt seit fast einer Woche und ist auch in mein VIS integriert. Sollte leicht zu erweitern sein, allerdings kann ich nicht mehr auf die Datei zugreifen, in der alle Register und Werte aufgezählt sein sollen.Features
- Werte auslesen
- Lüfterstufe und andere Werte setzen
- Vereinfachter Partybetrieb
- Automatisches Zurücksetzen des Filterwechsels
Wichtig: Es muss mindestens Node.js v12.x installiert sein! Anleitung zum Aktualisieren: https://forum.iobroker.net/topic/22867/how-to-node-js-für-iobroker-richtig-updaten
-
@KLVN Absolut genial. Endlich hat es jemand geschafft, nicht nur die Werte auszulesen, sondern auch zu setzen. DANKE.
komme heute leider nicht mehr zum Testen, aber ich werde berichten. -
@KLVN Vielen Dank für den Code.
Ich habe helios_ip und helios_passwort angepasst. Muss ich sonst noch was ändern?
Nach dem Starten bekomme ich folgende Fehlermeldung:
javascript.0 2020-11-18 08:30:37.043 error (5931) Error in request callback: TypeError: xml.matchAll is not a functionKannst du da weiterhelfen?
Besten Dank
mfg Andreas -
@Andyth Moin, welche Version von Node.js hast du installiert? Ich sehe gerade, dass
matchAll()
erst ab Version 12.0.0. vorhanden ist.
Hier ist eine Anleitung, wie man Node.js auf die neuste Version aktualisiert: https://forum.iobroker.net/topic/22867/how-to-node-js-für-iobroker-richtig-updaten(Sorry für die Umstände)
-
@KLVN Danke. Das war es gewesen. Nach dem Update von Node.js funktioniert das Script. Endlich kann man auch die Werte an die Lüftung übergeben. Darauf habe ich gewartet.
mfG Andreas
-
@KLVN Es wäre super wenn du noch deine VIS zur Verfügung stellen könntest.
Danke
mfG Andreas