NEWS
Huawei Sun2000 & ioBroker via JS script funktioniert
-
@chris_b Hallo Christian.
Zuerst mal Danke für deine super Arbeit
und ja, eine Schreibfunktion wäre super, denn dann könnte man auch mit Blockly so einiges realisieren,
es ist nicht jeder so verdraut mit JS.
Vielleicht findest ja mal Zeit und Muse um das umzusetzen, würde mich und viele Andere freuen.
Gruß aus Tirol, Jürgen -
@r4 steht auf 0 obwohl im von FusionSolar Dashboard "Fed to grid 1,84 kWh" stehen.
Ok hab es mit Blockly geschafft.
<xml xmlns="https://developers.google.com/blockly/xml"> <variables> <variable id="LJQ,%@iIai_dsi(6J9Eq">Text</variable> </variables> <block type="procedures_defcustomnoreturn" id=",0HSy+dmiy{AB+zsze1i" x="188" y="38"> <mutation statements="false"></mutation> <field name="NAME">pvstatus</field> <field name="SCRIPT"></field> <comment pinned="false" h="80" w="160">Beschreibe diese Funktion …</comment> </block> <block type="procedures_callcustomnoreturn" id="!y_lf7,S[0uMty;@$C^%" x="188" y="87"> <mutation name="pvstatus"></mutation> <next> <block type="on_ext" id="A}I(v:;Yi.q-TZxLt{GJ"> <mutation xmlns="http://www.w3.org/1999/xhtml" items="1"></mutation> <field name="CONDITION">any</field> <field name="ACK_CONDITION"></field> <value name="OID0"> <shadow type="field_oid" id="0D!$+7X}w,Em_*I2_6${"> <field name="oid">telegram.0.communicate.request</field> </shadow> </value> <statement name="STATEMENT"> <block type="variables_set" id="H.85Ep.=RR;96`B$@tRQ"> <field name="VAR" id="LJQ,%@iIai_dsi(6J9Eq">Text</field> <value name="VALUE"> <block type="text_getSubstring" id="e%v9@Avp[CXS/mKMsA|m" inline="false"> <mutation at1="true" at2="true"></mutation> <field name="WHERE1">FROM_START</field> <field name="WHERE2">FROM_START</field> <value name="STRING"> <block type="on_source" id="nr]0AXRI4@ly1Po{ogts"> <field name="ATTR">state.val</field> </block> </value> <value name="AT1"> <block type="math_number" id="mE*{B{|v/.q!Bdr_swye"> <field name="NUM">10</field> </block> </value> <value name="AT2"> <block type="text_indexOf" id="y2M|/VI|buP~WZMst~Sx"> <field name="END">LAST</field> <value name="VALUE"> <block type="on_source" id="0LUSQ6?Mi#4`Mc/%z8L:"> <field name="ATTR">state.val</field> </block> </value> <value name="FIND"> <shadow type="text" id="mgU46.?A,1gXuVE.)%w3"> <field name="TEXT">abc</field> </shadow> <block type="text" id="3DLQE[,IA6%X:Nj~WP6."> <field name="TEXT"></field> </block> </value> </block> </value> </block> </value> <next> <block type="debug" id="H_sQ8%)Qo3xvNpEo=3$c"> <field name="Severity">log</field> <value name="TEXT"> <shadow type="text" id="_UNidu@Lz/~o,l8_O|d!"> <field name="TEXT">test</field> </shadow> <block type="variables_get" id="qAq39wVy-N4?LzT.vUL;"> <field name="VAR" id="LJQ,%@iIai_dsi(6J9Eq">Text</field> </block> </value> <next> <block type="controls_if" id="d!d5kP`uz,iM|_Lvd9O)"> <value name="IF0"> <block type="logic_compare" id="VvAY3u8G^sI3yffZ0#iK"> <field name="OP">EQ</field> <value name="A"> <block type="variables_get" id="%^*B6K=IP2PrOXVD|rhk"> <field name="VAR" id="LJQ,%@iIai_dsi(6J9Eq">Text</field> </block> </value> <value name="B"> <block type="text" id="]?)ij.4#7u/h-#4Vs.$."> <field name="TEXT">pv</field> </block> </value> </block> </value> <statement name="DO0"> <block type="procedures_callnoreturn" id="97Y9~}y3`_P?_:]Y7usI"> <mutation name="pv"></mutation> </block> </statement> </block> </next> </block> </next> </block> </statement> </block> </next> </block> <block type="procedures_defnoreturn" id="ytGlwGFA:7ux]aJ:h)bX" x="188" y="537"> <field name="NAME">pv</field> <comment pinned="false" h="80" w="160">Beschreibe diese Funktion …</comment> </block> <block type="telegram" id="rFF^yU[tK;z8agA5HJ;t" x="187" y="613"> <field name="INSTANCE"></field> <field name="LOG"></field> <field name="SILENT">FALSE</field> <field name="PARSEMODE">default</field> <field name="ESCAPING">FALSE</field> <field name="DISABLE_WEB_PAGE_PREVIEW">FALSE</field> <value name="MESSAGE"> <shadow type="text" id="+NS:?N+v?F?X[sTTbNzr"> <field name="TEXT">Der Trockner ist fertig</field> </shadow> <block type="text_join" id="w==;)w0}uK[]4^$3ECh8"> <mutation items="19"></mutation> <value name="ADD0"> <block type="text" id="#^ya6yc#7PR[!s%UC4eT"> <field name="TEXT">☀️ Erzeugte kW: </field> </block> </value> <value name="ADD1"> <block type="get_value" id="{.Y7C$*/_-vBzHoS-!e("> <field name="ATTR">val</field> <field name="OID">javascript.0.Solarpower.Derived.YieldToday</field> </block> </value> <value name="ADD2"> <block type="text" id="7XWp;_ifO_,6w[h)T8m#"> <field name="TEXT">kW</field> </block> </value> <value name="ADD3"> <block type="text_newline" id="GD)kZE9,.j:fz1#,*VoY"> <field name="Type">\n</field> </block> </value> <value name="ADD4"> <block type="text" id="C3.z38(@7;bLUH+]FEa/"> <field name="TEXT">🌅 Verbrauchte kW: </field> </block> </value> <value name="ADD5"> <block type="get_value" id="v+E{d~%C]4MDBf8u`n:+"> <field name="ATTR">val</field> <field name="OID">0_userdata.0.energy.electricity.kWhPrice</field> </block> </value> <value name="ADD6"> <block type="text" id="u$W|[S^a/UO(Me+jYQWC"> <field name="TEXT">kW</field> </block> </value> <value name="ADD7"> <block type="text_newline" id="mhWg3?3RN4(TScFrlc1x"> <field name="Type">\n</field> </block> </value> <value name="ADD8"> <block type="text" id="N03?2e^)y|9W?@s[~Lx3"> <field name="TEXT">💸 Gekaufte kW: </field> </block> </value> <value name="ADD9"> <block type="get_value" id="bd@e@AVzHj}:yvYCF!VO"> <field name="ATTR">val</field> <field name="OID">0_userdata.0.energy.electricity.kWhPrice</field> </block> </value> <value name="ADD10"> <block type="text" id="O1tva[j5m9*a:je*8K}N"> <field name="TEXT">Wh</field> </block> </value> <value name="ADD11"> <block type="text_newline" id="J0!Md?l_$yeK%ZNT#/dp"> <field name="Type">\n</field> </block> </value> <value name="ADD12"> <block type="text" id="P17{8h/|#5L@{.sM*u+E"> <field name="TEXT">💶 Vekaufte kW: </field> </block> </value> <value name="ADD13"> <block type="text" id="x*b_qh+q$vy|?)1O3c%g"> <field name="TEXT">Wh</field> </block> </value> <value name="ADD14"> <block type="get_value" id="9SSpTQEk1H;2:dMiDjzp"> <field name="ATTR">val</field> <field name="OID">0_userdata.0.energy.electricity.kWhPrice</field> </block> </value> <value name="ADD15"> <block type="text_newline" id="):=!%*$05D]#39H@aHc="> <field name="Type">\n</field> </block> </value> <value name="ADD16"> <block type="text" id="CR#k,91R!s)T?Yp~Df^R"> <field name="TEXT">🔋Speicherstand: </field> </block> </value> <value name="ADD17"> <block type="get_value" id="A*(X.[rn5.[3lZPO.s}K"> <field name="ATTR">val</field> <field name="OID">javascript.0.Solarpower.Huawei.Inverter.1.Batterystack.1.BatterySOC</field> </block> </value> <value name="ADD18"> <block type="text" id="7)w_,xD@RSlv@hDR`^!H"> <field name="TEXT">%</field> </block> </value> </block> </value> </block> </xml>
Die Einheiten muss ich noch anpassen.Theorie:
Verbrauchte kWh pro Tag = Export to grid today - Import from grid today / 1000.
Bei mir wäre das 1075980 - 135910 = 940070 / 1000 = 940,07. Im Dashboard von FusionSolar steht "Verbraucht: 9,24 kWh"gekauft = 'javascript.0.Solarpower.Huawei.Meter.ReverseActiveEnergy'
verkauft = 'javascript.0.Solarpower.Huawei.Meter.PositiveActiveEnergy'
allerdings ja gesamt. Wie kann ich das auf den Zeitpunkt der Abfrage ausrechnen?
Ich habe auch den Zählerstand in ioBroker zur Verfügung. -
@hussi
Bitte nicht ständig den gleichen Beitrag ändern.
Man weiß dann nie was du neu geschrieben hast. -
@chris_b said in Huawei Sun2000 & ioBroker via JS script funktioniert:
@leonundjulie und @R4
Ich möchte kurz die möglichen Spekulationen dämpfen: Ja ich werde weiter am Skript arbeiten, es gibt sicher irgendwann eine V4 (vielleicht benenne ich sie dann anders, besser).
Aus meiner Sicht läuft das Skript, und es gibt wenig Grund Dinge zu entfernen. Unter Umständen liest das Skript zu viele Modbus Register, aber das stört eigentlich nicht weiter, es liest sowieso viel zu viele Werte. Aber die braucht man dann eben nicht. Der Overhead ist nicht sehr gross.
Ich werde als nächstes versuchen, das Skript via Github zur Verfügung zu stellen. Die verschiedenen Versionen hier im Forum sind sehr verwirrend. Dann gibt es nur noch einen Link auf Github, und dort ist immer die aktuelle Version zu finden.
Ob ich einen Adapter mache steht in den Sternen. Ich habe momentan andere Projekte, die (leider) eine höhere Priorität haben. Ich denke, ich käme erst gegen Ende Jahr dazu, sorry. Einen Schreibfunktion einzufügen wäre spannend, muss ich mir einmal überlegen, wie man das verallgemeinert und sinnvoll machen könnte.
Sorry, wenn es einigen von euch zu langsam geht, aber ich kann für das Projekt nur eine bestimmte Zeit aufwenden.
GrussHallo @Chris_B . Danke für den Blick in die Sterne.
Wie ich in einem meiner Fragen und Aussagen schrieb, kann ich bei mir eine Topologie erzeugen, in der das Skript läuft .
Aber ich möchte eigentlich in meiner weiter oben bebilderten Topologie bleiben. Und in der würden sowohl die EnPAL-Box als auch mein PI mit dem IOBroker auf den SUN2000 parallel zugreifen.
Diesbezüglich habe ich jetzt sehr viel gelesen …. die eine Variante sagt, dass der SDongle das nicht kann (ich habe diesbezüglich eine Anfrage bei HUAWEI platziert und warte auf Antwort.
Die andere Variante sagt, dass es geht, wenn man die Master-Slave Regeln beachtet. Auf Grund der Ausführungen unter https://npm.io/package/modbus-serial
frage ich mich, ob Du in Deinem Skript erst einen Slave (genau das ist ja Dein Skript) mit einer eigenen ID definieren musst. Der SDongle ist der Master; sowohl die EnPAL Box als auch der PI mit dem IOPBroker sind Slaves respektive Clients. Mag sein, dass ich mich mit diesen Überlegungen völlig verrannt habe, aber vielleicht findest Du ein paar Minuten Zeit und Lust auf den Gedanken einzugehen.Gruss
Andreas -
Nur mal kurz gefragt, man kann die Daten ja über NodeRed aus einem anderen Thread lesen oder über dieses Script hier aus dem Thread. Mich würde interessieren ob beides im Prinzip gleichwertig ist, oder ob die eine Variante evtl. Vor- oder Nachteile im Vergleich zu der anderen hat?
-
@atifan
Ein Argument: Wenn du bisschen Ahnung von Programmierung hast bist du hier flexibler (wobei ich nie was mit Javascript zu tun hatte...) -
@r4 said in Huawei Sun2000 & ioBroker via JS script funktioniert:
@atifan
Ein Argument: Wenn du bisschen Ahnung von Programmierung hast bist du hier flexibler (wobei ich nie was mit Javascript zu tun hatte...)… dem schließ ich mich sofort an. Ich habe vor ewigen Zeiten Regel-Algorithmen programmiert …. JS ist für mich neu, aber das Grundsätzliche erkenne auch ich ….. und NODE-RED ist zumindest für mich ein Blindflug mit copy&paste -> ganz klar pro JS
-
Wo finde ich die aktuellen Tageswert zu:
Verbrauchte kWh
Gekaufte kWh
Verkaufte kWhKann meinetwegen auch mit Blockly errechnet werden um sie dann in 0_userdata zu schreiben. Aber ich finde leider nichts in dem Objekten
-
-
@hussi Wie @R4 sagt: Ist weiter oben beschrieben.
Zudem werden genau diese Daten jeden Tag geloggt:var log1 = getState(JavaInst + "Solarpower.Derived.YieldToday").val; var log2 = getState(JavaInst + "Solarpower.Derived.GridExportToday").val; var log3 = getState(JavaInst + "Solarpower.Derived.GridImportToday").val; var log4 = getState(JavaInst + "Solarpower.Derived.ConsumptionToday").val;
Die Namen der States sagen ja eigentlich schon alles.
-
Ich bin natürlich zu doof. Dadurch, dass ich das Skript "verschlimmbessern" wollte, habe ich den Teil "Housekeeping" rausgenommen. Somit erklärt sich auch, warum die Werte nie zurückgesetzt worden sind.
Mein Fehler. Tut mir Leid und danke für die Hilfe!
Ich kann nun alle Daten wie gewollt über Telegram abfragen, und habe somit eine App weniger auf dem Handy.
-
@R4 und alle, die es interessiert.
Ich habe jetzt mein Skript (früher Version 3 genannt) nun auf Github.
Das macht es einfacher die aktuelle Version zu finden. Das ist der erste Versuch.
Das Readme File ist noch minimal, das werde ich stark ausbauen um Fragen zum Skript zu klären. Im Wesentlichen Dinge, die hier irgendwann einmal erwähnt wurden.
Hier der Link: -
@Chris_B Vielen Dank für dieses sehr nützliche Skript!
-
@chris_b
Zeile 3 "modidfied"
... ne, ich schau es mir mal kritisch an -
Wenn ich die aktuelle Leistung der PV in den Objekten sehen möchte, kann es dann sein, dass man dazu javascript.0.Solarpower.Huawei.Inverter.1.InputPower auslesen sollte?
Wenn der Akku nicht voll ist, zeigt mir Solarpower.Huawei.Inverter.1.ActivePower nicht den korrekten Wert an. Sobald der Akku voll ist stimmt der Wert.
Oder man rechnet Solarpower.Huawei.Inverter.1.ActivePower und javascript.0.Solarpower.Huawei.Inverter.1.Batterystack.1.ChargeAndDischargePower zusammen?
-
@r4 Danke - ein stottttter typo.
Habe ich korrigiert und das Readme auf Github schnell mit Information befüllt. Muss aber noch verbessert werden. -
@hussi Am besten einmal meine (neue) Github Seite anschauen. Dort sollte alles im Readme File erklärt sein. Ich werde das Readme in nächster Zeit noch erweitern und verbessern. Gruss
-
Hallo,
erst mal Danke für das Script, ich verwende die Version 3 und läuft auch soweit ohne Probleme.
Das einzige Problem ist, dass es alle paar Tage Morgens nicht mehr richtig läuft. Es hat dann Probleme die Werte zu lesen, siehe unten.
Wenn ich es dann noch einmal neu starte läuft es wieder ohne Probleme.Eine Idee woran das leigen könnte?
-
@heinzie
Du könntest versuchen in der Funktion readRegisterSpace() den auskommentierten Bereich zu aktivieren, also:function readRegisterSpace(id, address, length) //--------------------------------------------- { client.setID(ModBusIDs[id-1]); client.readHoldingRegisters(address, length, function(err, data) { if (err) { if (err.modbusCode == null) { console.warn("Lost connection to client. Trying to reconnect..."); ConnectModbus(); } else console.warn("Error received reading address " + address + " from id: " + ModBusIDs[id-1] + " with error: " + modbusErrorMessages[err.modbusCode]); } else { //console.debug("Read data from id/address " + ModBusIDs[id-1] + "/" + address + "\nData is: " + data.data); for (var i = 0; i < length; i++) Buffer[id-1][address + i - BufOffset] = data.data[i]; } }); }
-
@r4 sagte in Huawei Sun2000 & ioBroker via JS script funktioniert:
@heinzie
Du könntest versuchen in der Funktion readRegisterSpace() den auskommentierten Bereich zu aktivieren, also:function readRegisterSpace(id, address, length) //--------------------------------------------- { client.setID(ModBusIDs[id-1]); client.readHoldingRegisters(address, length, function(err, data) { if (err) { if (err.modbusCode == null) { console.warn("Lost connection to client. Trying to reconnect..."); ConnectModbus(); } else console.warn("Error received reading address " + address + " from id: " + ModBusIDs[id-1] + " with error: " + modbusErrorMessages[err.modbusCode]); } else { //console.debug("Read data from id/address " + ModBusIDs[id-1] + "/" + address + "\nData is: " + data.data); for (var i = 0; i < length; i++) Buffer[id-1][address + i - BufOffset] = data.data[i]; } }); }
Danke, habe ich einmal geändert, werde mich dann nach Test wieder melden.