Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Skripten / Logik
    4. JavaScript
    5. [gelöst] Elegantere Programmierung?

    NEWS

    • ioBroker@Smart Living Forum Solingen, 14.06. - Agenda added

    • ioBroker goes Matter ... Matter Adapter in Stable

    • Monatsrückblick - April 2025

    [gelöst] Elegantere Programmierung?

    This topic has been deleted. Only users with topic management privileges can see it.
    • S
      skorpil @paul53 last edited by

      @paul53 waaaas!? Du ohne IT Ausbildung? Ich glaube es nicht 😂 Ich hätte meinen Kopf darauf verwettet, Du hättest Informatik studiert.

      Nun gut, dann probiere ich weiter eine Lösung für Tibber zu finden. Und wenn ich was habe, melde ich mich hier.

      Homoran OliverIO 2 Replies Last reply Reply Quote 0
      • Homoran
        Homoran Global Moderator Administrators @skorpil last edited by

        @skorpil sagte in [gelöst] Elegantere Programmierung?:

        Ich hätte meinen Kopf darauf verwettet,

        ab mit dem Kopf!
        (die Herz-Königin in Alice im Wunderland)

        S 1 Reply Last reply Reply Quote 0
        • S
          skorpil @Homoran last edited by

          @homoran ich schrieb „hätte“. Daher bleibt er dran. Liebe Grüße

          1 Reply Last reply Reply Quote 0
          • OliverIO
            OliverIO @skorpil last edited by OliverIO

            @skorpil sagte in [gelöst] Elegantere Programmierung?:

            @paul53 waaaas!? Du ohne IT Ausbildung? Ich glaube es nicht 😂 Ich hätte meinen Kopf darauf verwettet, Du hättest Informatik studiert.

            Nun gut, dann probiere ich weiter eine Lösung für Tibber zu finden. Und wenn ich was habe, melde ich mich hier.

            sorry, hab leider auch keine IT-Ausbildung (und so wie ich manche kennen ist das auch nicht immer eine Auszeichnung 🙂 )

            Wie so oft muss man hier mehrere Technologien kennen.
            1.) Javascript
            2) Eine Bibliothek mit der man http sprechen kann
            3) Eine Bibliothek, die OAuth kann
            4) und zum Schluss muss man graphQL können bzw. die Basiskonzepte verstehen.
            Du hast einen Link zum graphQL Explorer geschickt, der natürlich auch ein Token benötigt. Durch Druck auf den Knopf Demo-Token kann man damit erst mal grundsätzlich arbeiten. Wie das mit Load personal Token application token abgefragt wird und im Hintergrund dann ein AccessToken abgefragt wird, keine Ahnung. Das reagiert nicht so wie ich erwartet hätte

            Also zur groben Lösung

            1. Muss ich voraussetzen, sonst wird es zu schwierig
            2. Hier würde ich Axios empfehlen https://www.npmjs.com/package/axios
            3. Hier würde ich ein Aufsatz auf axios empfehlen https://www.npmjs.com/package/axios-oauth-client
              In deiner tibber oberfläche müsstest du dir irgendwo ein application token/client-secret erzeugen können. Das wird per request an den server übertragen und du erhälst ein oauth-token zurück, welches du dann bei jeder nachfolgenden Abfrage als http-header mit dem vorsatz BEARER mitschicken musst
              Hier unter Authorization ist es erklärt, wohl auch mit Link zur Seite wo du dir einen Token holen kannst.
              https://developer.tibber.com/docs/guides/calling-api
            4. Wie man dann Abfragen ausführt steht dann hier:
              https://developer.tibber.com/docs/guides/calling-api unter "Performing" Requests
              Als curl Befehl ist ein Beispiel aufgeführt, was du an Daten mitgeben musst (Authorization, ContentType und die eigentlichen Daten), das kannst du beim lesen der Axios-Doku fast direkt umsetzen, da sind auch Beispiele für javascript mit dabei.
              Als Daten musst du dann ein graphQL-Ausdruck mitschicken(in einem JSON eingepackt als Attribut query, siehe Beispiel), den du in dem Explorer vorher ausprobieren kannst (halt mit Demo-Daten oder du trägst da ein OAuth-token ein, den du mit 3 geholt hast, dann auch mit deinen Daten)
              Mit dem Explorer müsstest du sehr Detailliert angeben können welche Daten du zurückbekommen möchtest, leider funktioniert die Autovervollständigung und das laden des Schemas (rechte Seite) nicht richtig. Daher könnte es schwierig werden, die richtigen Angaben zu machen. Da musst mal dort nachfragen. Evtl muss man eingeloggt sein.
            Codierknecht S 2 Replies Last reply Reply Quote 1
            • Codierknecht
              Codierknecht Developer Most Active @OliverIO last edited by

              @oliverio sagte in [gelöst] Elegantere Programmierung?:

              sorry, hab leider auch keine IT-Ausbildung

              Willkommen im Club 😁

              und so wie ich manche kennen ist das auch nicht immer eine Auszeichnung

              Manchmal (oft) sogar kontraproduktiv.
              Ich habe jedenfalls die Erfahrung gemacht, dass es (in unserem Team) einfacher ist, einem IT-affinen Fachmann das Programmieren beizubringen als einem (studierten) Programmierer unsere Prozesse zu erklären.
              Der junge Kollege beißt sich daran seit 3 Jahren die Zähne aus 😵

              OliverIO 1 Reply Last reply Reply Quote 2
              • OliverIO
                OliverIO @Codierknecht last edited by OliverIO

                @codierknecht sagte in [gelöst] Elegantere Programmierung?:

                Prozesse zu erklären.
                Der junge Kollege beißt sich daran seit 3 Jahren die Zähne aus

                Da ich mich schon sehr umfangreich auch mit Prozessmanagement auskenne und schon viele Schulungen durchgeführt habe, kann ich sagen, das es Leute gibt, die es nie lernen werden, da sie nicht in der Lage sind komplexe Problemstellungen in kleinere aufzuteilen, zu abstrahieren und dann noch das richtige Maß der Granularität zu finden.

                Dann ist er aber auch kein guter Programmierer? Das ist für mich nämlich ein sehr verwandter Skill.
                Am besten mal 4 Wochen in den Fachbereich schicken und ihn dort mitarbeiten lassen

                Codierknecht Homoran 2 Replies Last reply Reply Quote 1
                • Codierknecht
                  Codierknecht Developer Most Active @OliverIO last edited by

                  [Smalltalk]

                  @oliverio sagte in [gelöst] Elegantere Programmierung?:

                  Dann ist er aber auch kein guter Programmierer?

                  Sagen wir's mal so: Es hält sich in Grenzen 😉
                  Ich maße mir aber auch nicht an, das beurteilen zu können.
                  DCOM / COM+ ist inzwischen aber auch reichlich exotisch und Delphi als Sprache leider auch etwas aus der Mode gekommen.

                  Aber wie im SAP ein Materialstamm aufgebaut ist (A-, B-, C- und D-Segment) und wie daraus anhand von Primärbedarfen über die Stücklistenauflösung Fertigungsaufträge entstehen, sollte man nach 3 Jahren im Schlaf runterbeten können.

                  Aber lassen wir das.
                  Mühsam ernährt sich das Eichhörnchen - bis zu meinem Renteneintritt hat er's dann hoffentlich drauf.

                  S 1 Reply Last reply Reply Quote 1
                  • S
                    skorpil @OliverIO last edited by

                    @oliverio toll Dankeschön. Ich werde mich daran versuchen.

                    Im übrigen, unglaublich, ich hätte gedacht, dass ihr alle hier IT studiert habt

                    1 Reply Last reply Reply Quote 0
                    • Homoran
                      Homoran Global Moderator Administrators @OliverIO last edited by

                      @oliverio das ist aber auch hier im Forum oft der echte Haken!
                      Es ist sehr oft nicht wirklich "Ich kann kein Java", meist hapert's schon am strukturierten Denken / Herangehensweise

                      1 Reply Last reply Reply Quote 1
                      • S
                        skorpil @Codierknecht last edited by

                        @codierknecht @paul53 @OliverIO @Homoran @haus-automatisierung [small talk] ihr seid klasse hier. Es macht auch abseits von konkreten Fragen große Freude, Euch zuzuhören. Danke für Eure Geduld mit mir. Ich lerne viel. So, Lobhudelei Ende!

                        1 Reply Last reply Reply Quote 1
                        • S
                          skorpil last edited by

                          Ich habe mir dieses Javascript auf dem Umweg über Blockly (weil ich nicht das auch noch lernen will) zusammengestrickt:

                          on({id: "javascript.0.BenutzerVariablen.tibber_DP"/*tibber_DP*/, change: "ne"}, async function (obj) {
                            var value = obj.state.val;
                            var oldValue = obj.oldState.val;
                            setState("javascript.0.BenutzerVariablen.tibber_power"/*tibber_power*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'power'), true);
                            setState("javascript.0.BenutzerVariablen.tibber_lastMeterConsumption"/*tibber_lastMeterConsumption*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'lastMeterConsumption'), true);
                            setState("javascript.0.BenutzerVariablen.tibber_accumulatedConsumption"/*tibber_accumulatedConsumption*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'accumulatedConsumption'), true);
                            setState("javascript.0.BenutzerVariablen.tibber_minPower"/*tibber_minPower*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'minPower'), true);
                            setState("javascript.0.BenutzerVariablen.tibber_averagePower"/*tibber_averagePower*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'averagePower'), true);
                            setState("javascript.0.BenutzerVariablen.tibber_maxPower"/*tibber_maxPower*/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'maxPower'), true);
                          });
                          

                          Stichwort "elegante Programmierung": geht das nicht auch viel kürzer? Denn große Teile des Scripts mit Try und catch etc. wiederholen sich ja immer. Das müßte doch auch über eine Funktion gehen? Oder lieber so lassen?

                          paul53 1 Reply Last reply Reply Quote 0
                          • paul53
                            paul53 @skorpil last edited by paul53

                            @skorpil sagte: Das müßte doch auch über eine Funktion gehen?

                            Über eine Zwischenvariable object:

                            Blockly_temp.JPG

                            1 Reply Last reply Reply Quote 0
                            • S
                              skorpil last edited by skorpil

                              @paul53 ich bin schon wieder lost. Zwei Fragen:

                              1. wie setzte ich in Blockly den "WERT"? Wo versteckt sich dieses Blockly. Ich finde es nicht.
                                Unbenannt-2.jpg

                              2. Ich habe mir in der CCU eine SV "tibber-power" angelegt. Mit dem Script

                              setState('hm-rega.0.72161'/**/, jsonataExpression((function () { try {return JSON.parse(getState("javascript.0.BenutzerVariablen.tibber_DP").val);} catch(e) {return {};}})(),'power'), true);
                              

                              wird die Variable auch ständig aktualisiert. In den Objekten im iobroker sieht man das gut. Das Script funktioniert also!
                              Unbenannt-1.jpg

                              Nur, in der CCU wird die SV nicht aktualisiert.
                              Unbenannt-0jpg.jpg

                              Wo ist mein Denkfehler?

                              paul53 1 Reply Last reply Reply Quote 0
                              • paul53
                                paul53 @skorpil last edited by

                                @skorpil sagte: "WERT"? Wo versteckt sich dieses Blockly.

                                Unter "Trigger"

                                Blockly_DP_Props.JPG

                                @skorpil sagte in [gelöst] Elegantere Programmierung?:

                                Nur, in der CCU wird die SV nicht aktualisiert.

                                Für eine SV muss "steuere" anstelle von "aktualisiere" verwendet werden, sonst wird nicht gesendet.

                                S 1 Reply Last reply Reply Quote 1
                                • S
                                  skorpil @paul53 last edited by

                                  @paul53 danke, jetzt klappts

                                  1 Reply Last reply Reply Quote 0
                                  • S
                                    skorpil last edited by

                                    @paul53
                                    ich bin gerade dabei, mir ein Script zu bauen, in dem ich immer wieder eine bestimmte Funktion benötige. Und zwar diese hier:

                                    function getDeviceID(expression) {
                                      const parts = expression.split('.'); // Aufteilen des Ausdrucks in Teile
                                      if (parts.length >= 3) {
                                        return parts[2]; // Rückgabe des dritten Teils (Geräte-ID)
                                        /* 1. Teil = Name Adapterinstanz
                                           2. Teil = Nummer Adapterinstanz
                                           3. Teil = Geraete ID
                                           4. Teil = Geraete Kanal
                                           5. Teil = Zustandsvariable
                                        */   
                                      } else {
                                        return null; // Wenn der Ausdruck nicht die erwartete Form hat, wird null zurückgegeben
                                      }
                                    }
                                    

                                    Um es übersichtlicher zu gestalten, habe ich diese Function unter global als eigenes Script mit dem Namen "Funktion GeraeteID" abgelegt und rufe die Funktion aus einem anderen Script wiefolgt auf

                                    const thermKueche = 'hm-rpc.0.JEQ0553018.2.SETPOINT'/*Thermostat Kueche 1 SETPOINT*/;
                                    const deviceID = getDeviceID(thermKueche);
                                    log(deviceID); // Ausgabe: "JEQ0553018"
                                    

                                    Wenn ich unter global die Funktion "Funktion GeraeteID" modifiziere, meldet das Protokoll "Instanz ist deaktiviert" und es dauert einige Zeit. Danach funktioniert alles.

                                    Warum ist das so? Oder sollte ich die wiederholt benötigte Funktion doch besser im aufrufenden Script selber unterbringen? Was ist richtig und sinnvoll?

                                    paul53 2 Replies Last reply Reply Quote 0
                                    • paul53
                                      paul53 @skorpil last edited by paul53

                                      @skorpil sagte: "Funktion GeraeteID" modifiziere, meldet das Protokoll "Instanz ist deaktiviert" und es dauert einige Zeit. Danach funktioniert alles.

                                      Wenn globale Skripte modifiziert werden, müssen anschließend alle anderen Skripte kompiliert werden, da globale Skripte in alle anderen Skripte eingefügt werden. Das dauert etwas.

                                      @skorpil sagte in [gelöst] Elegantere Programmierung?:

                                      Was ist richtig und sinnvoll?

                                      Genau für Deine Situation, dass eigene Funktionen häufig verwendet werden, sind globale Skripte gedacht.

                                      Anmerkung:

                                      else return null
                                      

                                      ist unnötig, da andernfalls undefined zurück gegeben wird, was genauso wie null geprüft wird.

                                      S 1 Reply Last reply Reply Quote 1
                                      • S
                                        skorpil @paul53 last edited by skorpil

                                        @paul53 super. Danke. Nun noch eine Fragezur Verwendung der ON Anweisung:

                                        on({id: urlaub, change: 'ne', val: 0} && {id: sommer, change: 'ne', val: 0} , function () {
                                        

                                        Der weitere Teil des Scripts soll nur ausgeführt werden, wenn die (vorher definierten Variablen "sommer" und "urlaub" beide den Wert 0 haben. Ist das dann so richtig?

                                        paul53 1 Reply Last reply Reply Quote 0
                                        • paul53
                                          paul53 @skorpil last edited by

                                          @skorpil sagte: beide den Wert 0 haben. Ist das dann so richtig?

                                          Nein, man kann im Trigger keine Werte unterschiedlicher Datenpunkte per UND verknüpfen.

                                          on([urlaub, sommer], function () { // triggert bei Wertänderung eines DP
                                             if(getState(urlaub).val == 0 && getstate(sommer).val == 0) {
                                             }
                                          });
                                          

                                          Die Variablen sommer und urlaub müssen Datenpunkt-IDs enthalten.

                                          S 1 Reply Last reply Reply Quote 1
                                          • S
                                            skorpil @paul53 last edited by

                                            @paul53 sagte in [gelöst] Elegantere Programmierung?:

                                            on([urlaub, sommer], function () { // triggert bei Wertänderung eines DP if(getState(urlaub).val == 0 && getstate(sommer).val == 0) {

                                            Danke. Ich tappe dummerweise immer wieder in die gleichen Fallen.

                                            1 Reply Last reply Reply Quote 0
                                            • First post
                                              Last post

                                            Support us

                                            ioBroker
                                            Community Adapters
                                            Donate

                                            542
                                            Online

                                            31.7k
                                            Users

                                            79.7k
                                            Topics

                                            1.3m
                                            Posts

                                            10
                                            325
                                            36540
                                            Loading More Posts
                                            • Oldest to Newest
                                            • Newest to Oldest
                                            • Most Votes
                                            Reply
                                            • Reply as topic
                                            Log in to reply
                                            Community
                                            Impressum | Datenschutz-Bestimmungen | Nutzungsbedingungen
                                            The ioBroker Community 2014-2023
                                            logo