NEWS
[gelöst] Datenpunt aus JSON String erstellen und verarbeiten
-
Hey genau sowas suche ich auch und häng mich mal kurz dran. Mein Json sieht nur leider etwas anders aus und es funktioniert deshalb wohl nicht. Gibst du mir den notwendigen Tipp?
const id = "mqtt.0.solar.inverter.2000668170"; // bei Bedarf korrigieren const scheune2 = "mqtt.0.solar.inverter.2000364195";const scheune3 = "mqtt.0.solar.inverter.2000395543"; const haus1 = "mqtt.0.solar.inverter.2000415588"; const haus2 = "mqtt.0.solar.inverter.2000415597"; const path = 'javascript.' + instance + '.PV.Scheune1.'; const idSpannungDC = path + 'Spannung_DC'; const idStromstaerkeAC = path + 'Stromstärke_AC'; const idSpannungAC = path + 'Spannung_AC'; const idFrequenzAC = path + 'Freuqenz_AC'; const idLeistungAC = path + 'Leistung_AC'; const idEtotal = path + 'Eingespeiste_Leistung'; const idHtotal = path + 'Betriebsdauer'; if(!existsState(idHtotal)) { // Datenpunkte erzeugen createState(idSpannungDC, '', {type: 'number',unit: 'V'}); createState(idStromstaerkeAC, '', {type: 'number',unit: 'A'}); createState(idSpannungAC, '', {type: 'number',unit: 'V'}); createState(idFrequenzAC, '', {type: 'number',unit: 'Hz'}); createState(idLeistungAC, '', {type: 'number',unit: 'W'}); createState(idEtotal, '', {type: 'number',unit: 'kWh'}); createState(idHtotal, '', {type: 'number',unit: 'h'}); }; on(id, function(dp) { // triggert bei Wertänderung let obj = JSON.parse(getAttr(dp.state.val, 'values')); setState(idSpannungDC, obj.Upv-Ist, true); setState(idStromstaerkeAC, obj.Iac-Ist, true); setState(idSpannungAC, obj.Uac, true); setState(idFrequenzAC, obj.Fac, true); setState(idLeistungAC, obj.Pac, true); setState(idEtotal, obj.E-total, true); setState(idHtotal, obj.h-Total, true); });
Mein Objekt sieht so aus:
{"sn":000,"time":1586608779,"values": { "Iac":12580, "Uac":239, "Fac":50.019998881965876, "Pac":3006, "Zac":0.17200000816956162, "Riso":10000, "dI":4, "Upv-Ist":416, "PPV":3013, "E-Total":45358.538154414855, "h-Total":47797.772832013434, "h-On":51654.766385075491, "Netz-Ein":9012, "Seriennummer":000, "E-Total DC":45694.108978657052, "Status":"Mpp", "Fehler":"-------" }}
-
@iobroker_Alex sagte:
let obj = JSON.parse(getAttr(dp.state.val, 'values'));
Versuche es mal so:
let obj = JSON.parse(dp.state.val).values;
-
@paul53 wir nähern und dem Ergebnis:
In Obj steht nun:{'Upv-Ist':432,'Upv-Soll':433,'Iac-Ist':11.118000528076664,'Uac':242,'Fac':49.979998882859945,'Pac':2691,'Zac':0.5040000239387155,'Riso':3000,'Ipv':6.52900031011086,'E-Total':44230.93310085649,'h-Total':26578.838428355462,'h-On':27670.649680297298,'Netz-Ein':6759,'Seriennummer':2000668170,'Status':'Mpp','Balancer':'Off','Fehler':'-------'}
Jetzt meckert er bei ersten setState:
setState(idSpannungDC, obj.Upv-Ist, true);
Kann es sein, dass er "Upv-Ist" nicht mag / findet?
Fehler:
javascript.0 (24587) at Object.<anonymous> (script.js.Strom.Inverter1:25:36)
-
@iobroker_Alex sagte:
Kann es sein, dass er "Upv-Ist" nicht mag / findet?
Ja, enthält ein unzulässiges Zeichen: -
setState(idSpannungDC, obj['Upv-Ist'], true);
-
@paul53 Besten Dank funktioniert!!!
Sag mal wie du siehst habe ich mehrere Inverter die auf gleiche Weise ausgelesen werden sollen => 5 Stück um genau zu sein.
Um es mir am einfachsten zu machen würde ich nun das Skript 5 mal kopieren und unter anderem Namen speichern und nur oben den Path und das Objekt abändern. => AnfängermäßigWie würdest du das lösen?
Schleife drum und vorher nen Array mit den 5 Invertern und den 5 Paths anlegen? -
@iobroker_Alex sagte:
das Skript 5 mal kopieren und unter anderem Namen speichern und nur oben den Path und das Objekt abändern. => Anfängermäßig
Wie würdest du das lösen?Genauso.
-
@paul53 Vielen Dank hab echt was gelernt heute
-
Vielen Dank @paul53
Da haben wir echt etwas gelernt
zum JSON string auslesen und in Datenpunkte speichern hätte ich da noch zwei Fragen:1.) wie kann man diesen JSON string auslesen wobei die werte mit einem Semicolon getrennt sind?
{"idx":788,"RSSI":7,"nvalue":0,"svalue":"22.9;15.4;2"}2.) was kann man machen wenn sich in einer Objektzeile die Werte immer ändern ?
wie bekommt man die werte getrennt voneinander in einen Datenpunkt.
Das ESPeasy Projekt versendet den JSON string auf diese weise. Alle Werte ok aber eben immer in die selbe Objekt zeile, die Werte und der "idx" sind narürlich unterschiedlich.
LG
-
on(idJson, function(dp) { let obj = JSON.parse(dp.state.val); if(obj.idx == 788) { let arr = obj.svalue.split(';'); setState(idWert1 , parseFloat(arr[0]), true); setState(idWert2 , parseFloat(arr[1]), true); setState(idWert3 , parseFloat(arr[2]), true); } });
-
Kann ich auch jeden Tag einen automatisch einen neuen DP erstellen mit dem Datum des jeweilligen Tages?
Ich würde gerne die maximale Leistung meiner PV-Anlage speichern....
Hab es mit Blockly versucht, geht aber scheinbar nicht. Mit JS stehe ich "noch" auf dem Kriegspfad! -
@guergen sagte:
Kann ich auch jeden Tag einen automatisch einen neuen DP erstellen mit dem Datum des jeweilligen Tages?
Ja, unter "javascript.0" ist es möglich.
-
@paul53 Wie übergebe ich denn als Datenpunkt den Tag?
Anstatt des Test1 soll das Datum stehen...schedule("0 23 * * *", function () { createState("PV.Test1", getState("0_userdata.0.PV.Max").val, JSON.parse('{"type":"number","unit":"W"}'), function () { }); });
-
@guergen sagte:
Anstatt des Test1 soll das Datum stehen
schedule("0 23 * * *", function () { let datum = formatDate(new Date(), 'YYYY.O.D'); createState("PV." + datum, getState("0_userdata.0.PV.Max").val, {type: "number", unit: "W"}); });
Die Punkte im Datum erzeugen jedes Jahr und jeden Monat einen Ordner im Tab "Objekte". Wenn das nicht gewünscht ist, ersetze die Punkte durch Bindestriche. Doku. Datum heute bei der angegebenen Formatierung:
2020.Mai.7
-
@paul53 OK, verstanden...!
DP wird auch schon angelegt. jetzt muss ich nur noch den alten DP löschen, aber das bekomme ich hin!
Danke! -
@guergen sagte:
DP wird auch schon angelegt.
Ist denn schon 23:00 Uhr ?
Wenn die Sortier-Reihenfolge im Tab "Objekte" stimmen soll, dann verwende das Format "YYYY.MM.DD". -
@paul53 Na ich kann doch schonmal mit einer anderen Uhrzeit testen...
Habe es eben in YYYY.MM.DD geändert -
@paul53 Hmmmh... der soll den Max-Wert danach auf 0 setzen, habe ein Delay benutzt... geht nicht:
schedule("27 19 * * *", function () { let datum = formatDate(new Date(), 'YYYY.MM.DD'); createState("PV-Max." + datum, getState("0_userdata.0.PV.Max").val, {type: "number", unit: "W"}); setStateDelayed("0_userdata.0.PV.Max", 0, 30000); });
Ich befürchte, ich habe das mit den geschweiften Klammern noch nicht verstanden
Es schreibt den blöden Wert nicht in den neuen DP -
OK, wenn der neue DP PV-Max heisst geht es nicht, lasse ich das-Max weg, geht es!
-
-
@paul53 sagte in Datenpunt aus JSON String erstellen und weiter verarbeiten:
on(idJson, function(dp) { let obj = JSON.parse(dp.state.val); if(obj.idx == 788) { let arr = obj.svalue.split(';'); setState(idWert1 , parseFloat(arr[0]), true); setState(idWert2 , parseFloat(arr[1]), true); setState(idWert3 , parseFloat(arr[2]), true); } });
@paul53
Wie kann ich den in diesem fall dazugehörige Datenpunkte erzeugen ?const idVorraum_BWM1 = "mqtt.0.Vorraum_BWM1.in"; // {"idx":788,"RSSI":8,"nvalue":0,"svalue":"22.9;16.5;2"} const path = 'javascript.' + instance + '.SENSORBWM1.'; const idsvalue = path + 'Temp'; if(!existsState(idsvalue)) { // Datenpunkte erzeugen createState(idsvalue, 'leer', {type: 'string'}); }; on(idVorraum_BWM1, function(dp) { let obj = JSON.parse(dp.state.val); if(obj.idx == 788) { let arr = obj.svalue.split(';'); setState(idTemp , parseFloat(arr[0]), true); setState(idHum , parseFloat(arr[1]), true); } });