Skip to content
  • Home
  • Aktuell
  • Tags
  • 0 Ungelesen 0
  • Kategorien
  • Unreplied
  • Beliebt
  • GitHub
  • Docu
  • Hilfe
Skins
  • Light
  • Brite
  • Cerulean
  • Cosmo
  • Flatly
  • Journal
  • Litera
  • Lumen
  • Lux
  • Materia
  • Minty
  • Morph
  • Pulse
  • Sandstone
  • Simplex
  • Sketchy
  • Spacelab
  • United
  • Yeti
  • Zephyr
  • Dark
  • Cyborg
  • Darkly
  • Quartz
  • Slate
  • Solar
  • Superhero
  • Vapor

  • Standard: (Kein Skin)
  • Kein Skin
Einklappen
ioBroker Logo

Community Forum

donate donate
  1. ioBroker Community Home
  2. Deutsch
  3. Skripten / Logik
  4. Node-Red
  5. Datum und Zeitverarbeitung mit NodeRed

NEWS

  • Neuer Blogbeitrag: Monatsrückblick - Dezember 2025 🎄
    BluefoxB
    Bluefox
    11
    1
    332

  • Weihnachtsangebot 2025! 🎄
    BluefoxB
    Bluefox
    24
    1
    1.5k

  • UPDATE 31.10.: Amazon Alexa - ioBroker Skill läuft aus ?
    apollon77A
    apollon77
    48
    3
    9.5k

Datum und Zeitverarbeitung mit NodeRed

Geplant Angeheftet Gesperrt Verschoben Node-Red
node-red
33 Beiträge 5 Kommentatoren 32.7k Aufrufe 5 Watching
  • Älteste zuerst
  • Neuste zuerst
  • Meiste Stimmen
Antworten
  • In einem neuen Thema antworten
Anmelden zum Antworten
Dieses Thema wurde gelöscht. Nur Nutzer mit entsprechenden Rechten können es sehen.
  • mickymM mickym

    5. Abschluss

    So zum Abschluss habe ich mal meine neuen Kenntnisse dazu genutzt, um mit den einfachen Boardmitteln einen Timer zu erstellen.

    Einen der nicht geschwätzig ist. Wie gesagt, dass schöne an der Bibliothek ist dass man Objekte mit beliebigen Einheiten dazu addieren kann.

    Hier der Flow:

    429df954-1b1c-4dd4-9c83-d1be0831cf9b-image.png

    Hier mal der 9 Minuten und 28 Sekunden Timer Ausgabe in lesbarer aber nicht geschwätziger Form:
    Die Einheiten werden auch richtig auf und abgerundet. Wenn also in 7 Minuten ausgegeben wird, dann befindet sich er Timer zwischen 7:29 und 6:30 Minuten.

    3d38a592-8c0d-4b83-b447-64a30bb930ee-image.png

    und hier die geschwätzige Ausgabe mittels Angabe von verbleibenden Minuten und Sekunden.

    78f50370-b24a-4122-8ef0-37be86d1ac32-image.png

    Das Ende der Ausgabe ist noch unschön und kann man sicher anders lösen - aber das kann ja jeder für sich mal machen.

    Hier noch den Timerflow zum Import:

    [
       {
           "id": "1ebefb6.e95e305",
           "type": "inject",
           "z": "6e170384.60c96c",
           "name": "",
           "props": [
               {
                   "p": "payload"
               },
               {
                   "p": "topic",
                   "vt": "str"
               }
           ],
           "repeat": "",
           "crontab": "",
           "once": false,
           "onceDelay": 0.1,
           "topic": "timer",
           "payload": "{\"minutes\":9,\"seconds\":28}",
           "payloadType": "json",
           "x": 1030,
           "y": 1720,
           "wires": [
               [
                   "f057bad5.22c0c8"
               ]
           ]
       },
       {
           "id": "f057bad5.22c0c8",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "Timer läuft ab",
           "rules": [
               {
                   "t": "set",
                   "p": "payload",
                   "pt": "msg",
                   "to": "(\t    $x := payload;\t    $moment().add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                   "tot": "jsonata"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1360,
           "y": 1700,
           "wires": [
               [
                   "d539534f.ccfa8"
               ]
           ],
           "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
       },
       {
           "id": "d539534f.ccfa8",
           "type": "trigger",
           "z": "6e170384.60c96c",
           "name": "",
           "op1": "",
           "op2": "0",
           "op1type": "pay",
           "op2type": "str",
           "duration": "-1",
           "extend": false,
           "overrideDelay": false,
           "units": "s",
           "reset": "",
           "bytopic": "all",
           "topic": "topic",
           "outputs": 1,
           "x": 1580,
           "y": 1700,
           "wires": [
               [
                   "173f530c.7cf97d",
                   "ffe4b703.5c7ac8",
                   "5ad21dfe.76dc14"
               ]
           ]
       },
       {
           "id": "ba662ea7.2a6e3",
           "type": "inject",
           "z": "6e170384.60c96c",
           "name": "reset",
           "props": [
               {
                   "p": "reset",
                   "v": "true",
                   "vt": "bool"
               }
           ],
           "repeat": "",
           "crontab": "",
           "once": false,
           "onceDelay": 0.1,
           "topic": "",
           "x": 1350,
           "y": 1760,
           "wires": [
               [
                   "d539534f.ccfa8"
               ]
           ]
       },
       {
           "id": "173f530c.7cf97d",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "",
           "rules": [
               {
                   "t": "set",
                   "p": "payload",
                   "pt": "msg",
                   "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').fromNow()",
                   "tot": "jsonata"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1810,
           "y": 1700,
           "wires": [
               [
                   "16b62a7a.c7e9c6"
               ]
           ]
       },
       {
           "id": "16b62a7a.c7e9c6",
           "type": "rbe",
           "z": "6e170384.60c96c",
           "name": "",
           "func": "rbe",
           "gap": "",
           "start": "",
           "inout": "out",
           "septopics": true,
           "property": "payload",
           "x": 1570,
           "y": 1620,
           "wires": [
               [
                   "3aa094bf.4cd0bc"
               ]
           ]
       },
       {
           "id": "5414e8d9.006ec8",
           "type": "debug",
           "z": "6e170384.60c96c",
           "name": "",
           "active": false,
           "tosidebar": true,
           "console": false,
           "tostatus": false,
           "complete": "false",
           "statusVal": "",
           "statusType": "auto",
           "x": 1970,
           "y": 1560,
           "wires": []
       },
       {
           "id": "9e313ea9.6c67e",
           "type": "comment",
           "z": "6e170384.60c96c",
           "name": "Timer ausgabe",
           "info": "",
           "x": 800,
           "y": 1500,
           "wires": []
       },
       {
           "id": "ffe4b703.5c7ac8",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "",
           "rules": [
               {
                   "t": "set",
                   "p": "payload",
                   "pt": "msg",
                   "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').isSameOrAfter($moment())",
                   "tot": "jsonata"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1810,
           "y": 1760,
           "wires": [
               [
                   "9cb1684c.9731a8"
               ]
           ]
       },
       {
           "id": "9cb1684c.9731a8",
           "type": "switch",
           "z": "6e170384.60c96c",
           "name": "Timer abgelaufen?",
           "property": "payload",
           "propertyType": "msg",
           "rules": [
               {
                   "t": "false"
               }
           ],
           "checkall": "true",
           "repair": false,
           "outputs": 1,
           "x": 1810,
           "y": 1820,
           "wires": [
               [
                   "2a83c3b3.e36c4c"
               ]
           ]
       },
       {
           "id": "2a83c3b3.e36c4c",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "Timer abgelaufen, dann reset",
           "rules": [
               {
                   "t": "set",
                   "p": "reset",
                   "pt": "msg",
                   "to": "true",
                   "tot": "bool"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1520,
           "y": 1880,
           "wires": [
               [
                   "d539534f.ccfa8"
               ]
           ]
       },
       {
           "id": "3aa094bf.4cd0bc",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "",
           "rules": [
               {
                   "t": "change",
                   "p": "payload",
                   "pt": "msg",
                   "from": "vor ein paar Sekunden",
                   "fromt": "str",
                   "to": "Timer wurde beendet",
                   "tot": "str"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1800,
           "y": 1620,
           "wires": [
               [
                   "5414e8d9.006ec8"
               ]
           ]
       },
       {
           "id": "5ad21dfe.76dc14",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
           "rules": [
               {
                   "t": "set",
                   "p": "timeA",
                   "pt": "msg",
                   "to": "$moment().format('DD.MM.YYYY HH:mm:ss')",
                   "tot": "jsonata"
               },
               {
                   "t": "set",
                   "p": "timeB",
                   "pt": "msg",
                   "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                   "tot": "jsonata"
               },
               {
                   "t": "set",
                   "p": "payload",
                   "pt": "msg",
                   "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t        /* Kein Zugriff auf duration  - also manuell berechnen*/\t        \t    $days := $b.diff($a, 'days');\t    $hours := $b.diff($a, 'hours') - 24 * $b.diff($a, 'days');\t    $minutes := $b.diff($a, 'minutes') - 60 * $b.diff($a, 'hours');\t    $seconds := $b.diff($a, 'seconds') - 60 * $b.diff($a, 'minutes');\t    \t    $sec := $b.diff($a)/1000;\t    \t\t    \t    $difference := {\t        \"days\": $days,\t        \"hours\":$hours,\t        \"minutes\": $minutes,\t        \"seconds\": $seconds\t    }; \t    \t\t)",
                   "tot": "jsonata"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1420,
           "y": 1500,
           "wires": [
               [
                   "6c4aa2f5.a318ac"
               ]
           ],
           "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
       },
       {
           "id": "21f3fe07.9102e2",
           "type": "debug",
           "z": "6e170384.60c96c",
           "name": "",
           "active": true,
           "tosidebar": true,
           "console": false,
           "tostatus": false,
           "complete": "false",
           "statusVal": "",
           "statusType": "auto",
           "x": 1950,
           "y": 1500,
           "wires": []
       },
       {
           "id": "6c4aa2f5.a318ac",
           "type": "change",
           "z": "6e170384.60c96c",
           "name": "",
           "rules": [
               {
                   "t": "set",
                   "p": "payload",
                   "pt": "msg",
                   "to": "(\t    $str := payload.days > 0 ? payload.days & ' Tage(n) ' : '';\t    $str := payload.hours > 0 ? $str & payload.hours & ' Stunde(n) ' : '';\t    $str := payload.minutes > 0 ? $str & payload.minutes & ' Minute(n) ' : '';\t    $str := $str & payload.seconds & ' Sekunde(n)';\t    $str := ($str = '0 Sekunde(n)' or str = '-1 Sekunde(n)') ? 'Timer beendet' : $str;\t\t\t)",
                   "tot": "jsonata"
               }
           ],
           "action": "",
           "property": "",
           "from": "",
           "to": "",
           "reg": false,
           "x": 1750,
           "y": 1500,
           "wires": [
               [
                   "21f3fe07.9102e2"
               ]
           ]
       },
       {
           "id": "7acd75d2.b229ac",
           "type": "inject",
           "z": "6e170384.60c96c",
           "name": "",
           "props": [
               {
                   "p": "payload"
               },
               {
                   "p": "topic",
                   "vt": "str"
               }
           ],
           "repeat": "",
           "crontab": "",
           "once": false,
           "onceDelay": 0.1,
           "topic": "timer",
           "payload": "{\"seconds\":5}",
           "payloadType": "json",
           "x": 1070,
           "y": 1760,
           "wires": [
               [
                   "f057bad5.22c0c8"
               ]
           ]
       },
       {
           "id": "6ef0f333.4dd95c",
           "type": "inject",
           "z": "6e170384.60c96c",
           "name": "timer: {\"hours\":2,\"minutes\":47,\"seconds\":28}",
           "props": [
               {
                   "p": "payload"
               },
               {
                   "p": "topic",
                   "vt": "str"
               }
           ],
           "repeat": "",
           "crontab": "",
           "once": false,
           "onceDelay": 0.1,
           "topic": "timer",
           "payload": "{\"hours\":2,\"minutes\":47,\"seconds\":28}",
           "payloadType": "json",
           "x": 990,
           "y": 1680,
           "wires": [
               [
                   "f057bad5.22c0c8"
               ]
           ]
       },
       {
           "id": "6c51e401.1c82bc",
           "type": "inject",
           "z": "6e170384.60c96c",
           "name": "timer: {\"days\":3,\"hours\":1,\"minutes\":47,\"seconds\":28}",
           "props": [
               {
                   "p": "payload"
               },
               {
                   "p": "topic",
                   "vt": "str"
               }
           ],
           "repeat": "",
           "crontab": "",
           "once": false,
           "onceDelay": 0.1,
           "topic": "timer",
           "payload": "{\"days\":3,\"hours\":1,\"minutes\":47,\"seconds\":28}",
           "payloadType": "json",
           "x": 970,
           "y": 1640,
           "wires": [
               [
                   "f057bad5.22c0c8"
               ]
           ]
       }
    ]
    

    FAZIT und ENDE: ;)
    Ich finde die moment.js Bibliothek und wie man sie in NodeRed mit Standardmitteln nutzen kann, wirklich sehr mächtig und es erleichtert einem das Hantieren mit Datums - und Uhrzeitformaten. Wie gesagt was JSONATA Programmierung oder manchmal diese Abkürzungs Syntax bedeutet, das überfordert mich leicht.

    Deshalb mal ganz allgemein: Wenn jemand Fragen hat gerne. Wenn mir jemand erklären kann, wie ich unter JSONATA an das Duration Objekt der moment.js Library komme - gerne.

    Ich hoffe es ist OK, wenn ich hier mal das was ich herausgefunden habe, aufschreibe und hoffe auch, dass es dem einen oder anderen bei seinen eigenen Projekten hilft.

    R Offline
    R Offline
    rewenode
    schrieb am zuletzt editiert von rewenode
    #10

    Hab den Post weiter unten für iobroker und node-red aktualisiert und den alten gelöscht.

    mickymM 1 Antwort Letzte Antwort
    0
    • R rewenode

      Hab den Post weiter unten für iobroker und node-red aktualisiert und den alten gelöscht.

      mickymM Online
      mickymM Online
      mickym
      Most Active
      schrieb am zuletzt editiert von mickym
      #11

      @rewenode Ja das wollte ich eigentlich verhindern - mit der externen Einbindung - da musst Du wieder an der settings.js rumbasteln, dass Du externe Bibliotheken verfügbar machst.

      Konnte Deinen Flow auch nicht einbinden - da irgendein parse Error - aber Dank Deiner Erklärung konnte ich es auch nachvollziehen - allerdings auch nur in der Standalone Version:

      bb64d421-f136-4521-a8f8-d82c43f35da3-image.png

      In der function Node muss man halt die Bibliothek noch bekannt machen:

      af2b6a1e-fe05-4b31-b7c5-4cad429ed9e8-image.png

      Interessant ist dass der duration strings nur bis zu Stunden auflöst - also 195 Stunden.

      Habs mal mit 5 Jahre + 30 Tagen + 3 Stunden + 47 Minuten +28 Sekunden - ausprobiert. :grin: Dann kommen auch nur 44547 Stunden raus. ;) - Damit ist der Mehrwert nun nicht so hoch - als dass sich der Streß mit Einbindung externer Bibliotheken lohnt.

      16220424-1800-4717-9574-1a03385bf557-image.png

      Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

      1 Antwort Letzte Antwort
      0
      • R Offline
        R Offline
        rewenode
        schrieb am zuletzt editiert von rewenode
        #12

        @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

        @rewenode Ja das wollte ich eigentlich verhindern - mit der externen Einbindung - da musst Du wieder an der settings.js rumbasteln, dass Du externe Bibliotheken verfügbar machst.
        Konnte Deinen Flow auch nicht einbinden - da irgendein parse Error - aber Dank Deiner Erklärung konnte ich es auch nachvollziehen - allerdings auch nur in der Standalone Version:

        Ja, sorry. hatte es in einer stand alone version getestet.
        Hier nochmal für die ioBroker Version.
        Da gehts dann ganz einfach, wenn man einmal die moment.js eingebunden hat. Ein function-node ist dann nicht mehr nötig.

        1. binde die moment.js über die settings.js ein.
          Die befindet sich in
        /opt/iobroker/node_modules/iobroker.node-red/settings.js
        

        Dort muss di function functionGlobalContext um den Eintrag

        moment:require("moment")
        

        erweitert werden.

        Beispiel aus meiner settings.ja

        functionGlobalContext: {
                moment:require("moment")
                //'%%functionGlobalContext%%'
                // os:require('os'),
                // jfive:require("johnny-five"),
                // j5board:require("johnny-five").Board({repl:false})
            },
        

        Dann die node-red instanz neu starten.
        Das wars.

        moment.duration kann in JSONata immer mit

        $globalContext('moment.duration')
        

        verwendet werden.

        2021-12-09_21-37-51.png

        moment steht dann natürlich auch in function-nodes zur Verfügung.

        [
            {
                "id": "5d6fc95c.e73438",
                "type": "tab",
                "label": "Flow 3",
                "disabled": false,
                "info": ""
            },
            {
                "id": "c38cc5bf.7f7f38",
                "type": "inject",
                "z": "5d6fc95c.e73438",
                "name": "jetzt + 8 Tagen + 3 Stunden + 47 Minuten +28 Sekunden",
                "props": [
                    {
                        "p": "payload"
                    },
                    {
                        "p": "x",
                        "v": "{\"days\":8,\"hours\":3,\"minutes\":47,\"seconds\":28}",
                        "vt": "json"
                    }
                ],
                "repeat": "",
                "crontab": "",
                "once": false,
                "onceDelay": 0.1,
                "topic": "",
                "payload": "",
                "payloadType": "date",
                "x": 250,
                "y": 140,
                "wires": [
                    [
                        "f6e8d65f.929de8",
                        "c535b75f1d3d6bb6"
                    ]
                ]
            },
            {
                "id": "f6e8d65f.929de8",
                "type": "change",
                "z": "5d6fc95c.e73438",
                "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                "rules": [
                    {
                        "t": "set",
                        "p": "timeA",
                        "pt": "msg",
                        "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                        "tot": "jsonata"
                    },
                    {
                        "t": "set",
                        "p": "timeB",
                        "pt": "msg",
                        "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                        "tot": "jsonata"
                    },
                    {
                        "t": "delete",
                        "p": "x",
                        "pt": "msg"
                    },
                    {
                        "t": "set",
                        "p": "payload",
                        "pt": "msg",
                        "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t        /* Kein Zugriff auf duration  - also manuell berechnen*/\t        \t    $days := $b.diff($a, 'days');\t    $hours := $b.diff($a, 'hours') - 24 * $b.diff($a, 'days');\t    $minutes := $b.diff($a, 'minutes') - 60 * $b.diff($a, 'hours');\t    $seconds := $b.diff($a, 'seconds') - 60 * $b.diff($a, 'minutes');\t    \t    $sec := $b.diff($a)/1000;\t    \t\t    \t    $difference := {\t        \"days\": $days,\t        \"hours\":$hours,\t        \"minutes\": $minutes,\t        \"seconds\": $seconds\t    }; \t    \t\t)",
                        "tot": "jsonata"
                    }
                ],
                "action": "",
                "property": "",
                "from": "",
                "to": "",
                "reg": false,
                "x": 720,
                "y": 140,
                "wires": [
                    [
                        "fb984b77.f03568"
                    ]
                ],
                "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
            },
            {
                "id": "fb984b77.f03568",
                "type": "debug",
                "z": "5d6fc95c.e73438",
                "name": "",
                "active": true,
                "tosidebar": true,
                "console": false,
                "tostatus": false,
                "complete": "true",
                "targetType": "full",
                "statusVal": "",
                "statusType": "auto",
                "x": 1050,
                "y": 140,
                "wires": []
            },
            {
                "id": "9958c57.83b6e38",
                "type": "comment",
                "z": "5d6fc95c.e73438",
                "name": "Zeitvergleiche",
                "info": "",
                "x": 110,
                "y": 100,
                "wires": []
            },
            {
                "id": "c535b75f1d3d6bb6",
                "type": "change",
                "z": "5d6fc95c.e73438",
                "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                "rules": [
                    {
                        "t": "set",
                        "p": "timeA",
                        "pt": "msg",
                        "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                        "tot": "jsonata"
                    },
                    {
                        "t": "set",
                        "p": "timeB",
                        "pt": "msg",
                        "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                        "tot": "jsonata"
                    },
                    {
                        "t": "delete",
                        "p": "x",
                        "pt": "msg"
                    },
                    {
                        "t": "set",
                        "p": "payload",
                        "pt": "msg",
                        "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t    $globalContext('moment.duration')($b.diff($a)).humanize()\t\t)",
                        "tot": "jsonata"
                    }
                ],
                "action": "",
                "property": "",
                "from": "",
                "to": "",
                "reg": false,
                "x": 720,
                "y": 200,
                "wires": [
                    [
                        "cf428b715b7fa6a5"
                    ]
                ],
                "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
            },
            {
                "id": "cf428b715b7fa6a5",
                "type": "debug",
                "z": "5d6fc95c.e73438",
                "name": "",
                "active": true,
                "tosidebar": true,
                "console": false,
                "tostatus": false,
                "complete": "true",
                "targetType": "full",
                "statusVal": "",
                "statusType": "auto",
                "x": 1050,
                "y": 200,
                "wires": []
            }
        ]
        

        Gruß
        Reiner

        mickymM 2 Antworten Letzte Antwort
        0
        • R rewenode

          @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

          @rewenode Ja das wollte ich eigentlich verhindern - mit der externen Einbindung - da musst Du wieder an der settings.js rumbasteln, dass Du externe Bibliotheken verfügbar machst.
          Konnte Deinen Flow auch nicht einbinden - da irgendein parse Error - aber Dank Deiner Erklärung konnte ich es auch nachvollziehen - allerdings auch nur in der Standalone Version:

          Ja, sorry. hatte es in einer stand alone version getestet.
          Hier nochmal für die ioBroker Version.
          Da gehts dann ganz einfach, wenn man einmal die moment.js eingebunden hat. Ein function-node ist dann nicht mehr nötig.

          1. binde die moment.js über die settings.js ein.
            Die befindet sich in
          /opt/iobroker/node_modules/iobroker.node-red/settings.js
          

          Dort muss di function functionGlobalContext um den Eintrag

          moment:require("moment")
          

          erweitert werden.

          Beispiel aus meiner settings.ja

          functionGlobalContext: {
                  moment:require("moment")
                  //'%%functionGlobalContext%%'
                  // os:require('os'),
                  // jfive:require("johnny-five"),
                  // j5board:require("johnny-five").Board({repl:false})
              },
          

          Dann die node-red instanz neu starten.
          Das wars.

          moment.duration kann in JSONata immer mit

          $globalContext('moment.duration')
          

          verwendet werden.

          2021-12-09_21-37-51.png

          moment steht dann natürlich auch in function-nodes zur Verfügung.

          [
              {
                  "id": "5d6fc95c.e73438",
                  "type": "tab",
                  "label": "Flow 3",
                  "disabled": false,
                  "info": ""
              },
              {
                  "id": "c38cc5bf.7f7f38",
                  "type": "inject",
                  "z": "5d6fc95c.e73438",
                  "name": "jetzt + 8 Tagen + 3 Stunden + 47 Minuten +28 Sekunden",
                  "props": [
                      {
                          "p": "payload"
                      },
                      {
                          "p": "x",
                          "v": "{\"days\":8,\"hours\":3,\"minutes\":47,\"seconds\":28}",
                          "vt": "json"
                      }
                  ],
                  "repeat": "",
                  "crontab": "",
                  "once": false,
                  "onceDelay": 0.1,
                  "topic": "",
                  "payload": "",
                  "payloadType": "date",
                  "x": 250,
                  "y": 140,
                  "wires": [
                      [
                          "f6e8d65f.929de8",
                          "c535b75f1d3d6bb6"
                      ]
                  ]
              },
              {
                  "id": "f6e8d65f.929de8",
                  "type": "change",
                  "z": "5d6fc95c.e73438",
                  "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                  "rules": [
                      {
                          "t": "set",
                          "p": "timeA",
                          "pt": "msg",
                          "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                          "tot": "jsonata"
                      },
                      {
                          "t": "set",
                          "p": "timeB",
                          "pt": "msg",
                          "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                          "tot": "jsonata"
                      },
                      {
                          "t": "delete",
                          "p": "x",
                          "pt": "msg"
                      },
                      {
                          "t": "set",
                          "p": "payload",
                          "pt": "msg",
                          "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t        /* Kein Zugriff auf duration  - also manuell berechnen*/\t        \t    $days := $b.diff($a, 'days');\t    $hours := $b.diff($a, 'hours') - 24 * $b.diff($a, 'days');\t    $minutes := $b.diff($a, 'minutes') - 60 * $b.diff($a, 'hours');\t    $seconds := $b.diff($a, 'seconds') - 60 * $b.diff($a, 'minutes');\t    \t    $sec := $b.diff($a)/1000;\t    \t\t    \t    $difference := {\t        \"days\": $days,\t        \"hours\":$hours,\t        \"minutes\": $minutes,\t        \"seconds\": $seconds\t    }; \t    \t\t)",
                          "tot": "jsonata"
                      }
                  ],
                  "action": "",
                  "property": "",
                  "from": "",
                  "to": "",
                  "reg": false,
                  "x": 720,
                  "y": 140,
                  "wires": [
                      [
                          "fb984b77.f03568"
                      ]
                  ],
                  "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
              },
              {
                  "id": "fb984b77.f03568",
                  "type": "debug",
                  "z": "5d6fc95c.e73438",
                  "name": "",
                  "active": true,
                  "tosidebar": true,
                  "console": false,
                  "tostatus": false,
                  "complete": "true",
                  "targetType": "full",
                  "statusVal": "",
                  "statusType": "auto",
                  "x": 1050,
                  "y": 140,
                  "wires": []
              },
              {
                  "id": "9958c57.83b6e38",
                  "type": "comment",
                  "z": "5d6fc95c.e73438",
                  "name": "Zeitvergleiche",
                  "info": "",
                  "x": 110,
                  "y": 100,
                  "wires": []
              },
              {
                  "id": "c535b75f1d3d6bb6",
                  "type": "change",
                  "z": "5d6fc95c.e73438",
                  "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                  "rules": [
                      {
                          "t": "set",
                          "p": "timeA",
                          "pt": "msg",
                          "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                          "tot": "jsonata"
                      },
                      {
                          "t": "set",
                          "p": "timeB",
                          "pt": "msg",
                          "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                          "tot": "jsonata"
                      },
                      {
                          "t": "delete",
                          "p": "x",
                          "pt": "msg"
                      },
                      {
                          "t": "set",
                          "p": "payload",
                          "pt": "msg",
                          "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t    $globalContext('moment.duration')($b.diff($a)).humanize()\t\t)",
                          "tot": "jsonata"
                      }
                  ],
                  "action": "",
                  "property": "",
                  "from": "",
                  "to": "",
                  "reg": false,
                  "x": 720,
                  "y": 200,
                  "wires": [
                      [
                          "cf428b715b7fa6a5"
                      ]
                  ],
                  "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
              },
              {
                  "id": "cf428b715b7fa6a5",
                  "type": "debug",
                  "z": "5d6fc95c.e73438",
                  "name": "",
                  "active": true,
                  "tosidebar": true,
                  "console": false,
                  "tostatus": false,
                  "complete": "true",
                  "targetType": "full",
                  "statusVal": "",
                  "statusType": "auto",
                  "x": 1050,
                  "y": 200,
                  "wires": []
              }
          ]
          

          Gruß
          Reiner

          mickymM Online
          mickymM Online
          mickym
          Most Active
          schrieb am zuletzt editiert von mickym
          #13

          @rewenode Vielen Dank - die Anleitung ist sicher nützlich - wobei ich deswegen nun keine externe Bibliotheken einbinden würde. Aber in meiner Standalone Konfig habe ich das Duration Teil auch mal getestet, wie Du es wieder aufgelöst bekommst und es funktioniert !!!! - wenn man keine Wochen benutzt. ;)

          0d75ad0f-18fe-4a39-9ce0-ffdc0d045d99-image.png

          aber dann ist das Duration doch ganz nützlich - hier mal der Code:

          (
              $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');
              $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');
              
              /* $b.diff($a) */
              $d := $globalContext('md')($b.diff($a));
          
              $difference := {
                  "years" : $d.years(),
                  "months":$d.months(),
                  "weeks": $d.weeks(),
                  "days": $d.days(),
                  "hours":$d.hours(),
                  "minutes":$d.minutes(),
                  "seconds": $d.seconds()
              }; 
          )
          

          Das gute ist ja dass das richtig umgerechnet wird und wirklich Schaltjahre etc. alles berücksichtig wird.
          Ausgangspunkt: 9.12.2021, 21:51:50
          Als zu addierenden Zeitraum habe ich bewußt mal einen etwas redundanten Zeitraum genommen (2 Wochen und 30 Tage)

          {
              "years": 5,
              "months": 1,
              "weeks": 2,
              "days": 30,
              "hours": 3,
              "minutes": 47,
              "seconds": 28
          }
          

          Also dieser Zeitraum wird addiert und das Ergebnis ist verifiziert:

          9.12.2021 + 5 Jahre = 9.12.2026
          9.12.2026 + 1 Monat = 9.1.2027
          9.1.2027 + 2 Wochen = 23.1.2027
          23.1.2027 + 30 Tage = 22.2.2027 21:51:50
          22.2.2027 21:51:50 + 3 Std. = 23.2.2027 0:51:50
          23.2.2027 0:51:50 + 47 Minuten = 23.2.2027 1:38:50
          23.2.2027 1:38:50 + 28 Sekunden = 23.2.2027 1:39:18

          Ausgangspunkt: 9.12.2021, 21:51:50
          Das Duration Objekt liefert (wie gesagt - Wochen muss man rausnehmen)

          {
              "years": 5,
              "months": 2,
              "days": 13,
              "hours": 3,
              "minutes": 47,
              "seconds": 28
          }
          

          9.12.2021 + 5 Jahre = 9.12.2026
          9.12.2026 + 2 Monate = 9.2.2027
          9.2.2027 + 1 Woche = 16.2.2027
          16.2.2027 + 13 Tage = 1.3.2027 ?????????
          9.2.2027 + 13 Tage = 22.2.2027 21:51:50
          22.2.2027 21:51:50 + 3 Std. = 23.2.2027 0:51:50
          23.2.2027 0:51:50 + 47 Minuten = 23.2.2027 1:38:50
          23.2.2027 1:38:50 + 28 Sekunden = 23.2.2027 1:39:18

          Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

          1 Antwort Letzte Antwort
          0
          • R rewenode

            @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

            @rewenode Ja das wollte ich eigentlich verhindern - mit der externen Einbindung - da musst Du wieder an der settings.js rumbasteln, dass Du externe Bibliotheken verfügbar machst.
            Konnte Deinen Flow auch nicht einbinden - da irgendein parse Error - aber Dank Deiner Erklärung konnte ich es auch nachvollziehen - allerdings auch nur in der Standalone Version:

            Ja, sorry. hatte es in einer stand alone version getestet.
            Hier nochmal für die ioBroker Version.
            Da gehts dann ganz einfach, wenn man einmal die moment.js eingebunden hat. Ein function-node ist dann nicht mehr nötig.

            1. binde die moment.js über die settings.js ein.
              Die befindet sich in
            /opt/iobroker/node_modules/iobroker.node-red/settings.js
            

            Dort muss di function functionGlobalContext um den Eintrag

            moment:require("moment")
            

            erweitert werden.

            Beispiel aus meiner settings.ja

            functionGlobalContext: {
                    moment:require("moment")
                    //'%%functionGlobalContext%%'
                    // os:require('os'),
                    // jfive:require("johnny-five"),
                    // j5board:require("johnny-five").Board({repl:false})
                },
            

            Dann die node-red instanz neu starten.
            Das wars.

            moment.duration kann in JSONata immer mit

            $globalContext('moment.duration')
            

            verwendet werden.

            2021-12-09_21-37-51.png

            moment steht dann natürlich auch in function-nodes zur Verfügung.

            [
                {
                    "id": "5d6fc95c.e73438",
                    "type": "tab",
                    "label": "Flow 3",
                    "disabled": false,
                    "info": ""
                },
                {
                    "id": "c38cc5bf.7f7f38",
                    "type": "inject",
                    "z": "5d6fc95c.e73438",
                    "name": "jetzt + 8 Tagen + 3 Stunden + 47 Minuten +28 Sekunden",
                    "props": [
                        {
                            "p": "payload"
                        },
                        {
                            "p": "x",
                            "v": "{\"days\":8,\"hours\":3,\"minutes\":47,\"seconds\":28}",
                            "vt": "json"
                        }
                    ],
                    "repeat": "",
                    "crontab": "",
                    "once": false,
                    "onceDelay": 0.1,
                    "topic": "",
                    "payload": "",
                    "payloadType": "date",
                    "x": 250,
                    "y": 140,
                    "wires": [
                        [
                            "f6e8d65f.929de8",
                            "c535b75f1d3d6bb6"
                        ]
                    ]
                },
                {
                    "id": "f6e8d65f.929de8",
                    "type": "change",
                    "z": "5d6fc95c.e73438",
                    "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                    "rules": [
                        {
                            "t": "set",
                            "p": "timeA",
                            "pt": "msg",
                            "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                            "tot": "jsonata"
                        },
                        {
                            "t": "set",
                            "p": "timeB",
                            "pt": "msg",
                            "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                            "tot": "jsonata"
                        },
                        {
                            "t": "delete",
                            "p": "x",
                            "pt": "msg"
                        },
                        {
                            "t": "set",
                            "p": "payload",
                            "pt": "msg",
                            "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t        /* Kein Zugriff auf duration  - also manuell berechnen*/\t        \t    $days := $b.diff($a, 'days');\t    $hours := $b.diff($a, 'hours') - 24 * $b.diff($a, 'days');\t    $minutes := $b.diff($a, 'minutes') - 60 * $b.diff($a, 'hours');\t    $seconds := $b.diff($a, 'seconds') - 60 * $b.diff($a, 'minutes');\t    \t    $sec := $b.diff($a)/1000;\t    \t\t    \t    $difference := {\t        \"days\": $days,\t        \"hours\":$hours,\t        \"minutes\": $minutes,\t        \"seconds\": $seconds\t    }; \t    \t\t)",
                            "tot": "jsonata"
                        }
                    ],
                    "action": "",
                    "property": "",
                    "from": "",
                    "to": "",
                    "reg": false,
                    "x": 720,
                    "y": 140,
                    "wires": [
                        [
                            "fb984b77.f03568"
                        ]
                    ],
                    "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
                },
                {
                    "id": "fb984b77.f03568",
                    "type": "debug",
                    "z": "5d6fc95c.e73438",
                    "name": "",
                    "active": true,
                    "tosidebar": true,
                    "console": false,
                    "tostatus": false,
                    "complete": "true",
                    "targetType": "full",
                    "statusVal": "",
                    "statusType": "auto",
                    "x": 1050,
                    "y": 140,
                    "wires": []
                },
                {
                    "id": "9958c57.83b6e38",
                    "type": "comment",
                    "z": "5d6fc95c.e73438",
                    "name": "Zeitvergleiche",
                    "info": "",
                    "x": 110,
                    "y": 100,
                    "wires": []
                },
                {
                    "id": "c535b75f1d3d6bb6",
                    "type": "change",
                    "z": "5d6fc95c.e73438",
                    "name": "timeA = payload; timeB =payload +x; diff timeB-timeA",
                    "rules": [
                        {
                            "t": "set",
                            "p": "timeA",
                            "pt": "msg",
                            "to": "$moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').format('DD.MM.YYYY HH:mm:ss')",
                            "tot": "jsonata"
                        },
                        {
                            "t": "set",
                            "p": "timeB",
                            "pt": "msg",
                            "to": "(\t    $x := x;\t    $moment(payload,['DD.MM.YYYY HH:mm:ss','x'],'de').add($x).format('DD.MM.YYYY HH:mm:ss');\t    )",
                            "tot": "jsonata"
                        },
                        {
                            "t": "delete",
                            "p": "x",
                            "pt": "msg"
                        },
                        {
                            "t": "set",
                            "p": "payload",
                            "pt": "msg",
                            "to": "(\t    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');\t    \t    /* $b.diff($a) */\t    /* $moment.duration($b.diff($a)); */\t    \t    $globalContext('moment.duration')($b.diff($a)).humanize()\t\t)",
                            "tot": "jsonata"
                        }
                    ],
                    "action": "",
                    "property": "",
                    "from": "",
                    "to": "",
                    "reg": false,
                    "x": 720,
                    "y": 200,
                    "wires": [
                        [
                            "cf428b715b7fa6a5"
                        ]
                    ],
                    "info": "akzeptiert Datumsformate ('DD.MM.YYYY HH:mm:ss') oder timestamps('x')"
                },
                {
                    "id": "cf428b715b7fa6a5",
                    "type": "debug",
                    "z": "5d6fc95c.e73438",
                    "name": "",
                    "active": true,
                    "tosidebar": true,
                    "console": false,
                    "tostatus": false,
                    "complete": "true",
                    "targetType": "full",
                    "statusVal": "",
                    "statusType": "auto",
                    "x": 1050,
                    "y": 200,
                    "wires": []
                }
            ]
            

            Gruß
            Reiner

            mickymM Online
            mickymM Online
            mickym
            Most Active
            schrieb am zuletzt editiert von
            #14

            @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.

            Der Import geht aber im Moment trotzdem nicht - manche Flows mögen irgendwie nicht. :(

            798dec83-bddd-42bb-a3be-813f7838d1c6-image.png

            Aber egal - ich hab das ja erst mal miteinbinden können und auch testen können mit dem Durations Objekt - also vielen Dank Dir dafür.

            Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

            R 1 Antwort Letzte Antwort
            0
            • mickymM mickym

              @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.

              Der Import geht aber im Moment trotzdem nicht - manche Flows mögen irgendwie nicht. :(

              798dec83-bddd-42bb-a3be-813f7838d1c6-image.png

              Aber egal - ich hab das ja erst mal miteinbinden können und auch testen können mit dem Durations Objekt - also vielen Dank Dir dafür.

              R Offline
              R Offline
              rewenode
              schrieb am zuletzt editiert von
              #15

              @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

              @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.

              Ah, sorry. Bin halt zu selten hier. Hab's grad korrigiert und einen Reimport getestet, also bei mir gehts.

              Gruß
              Reiner

              mickymM 1 Antwort Letzte Antwort
              0
              • R rewenode

                @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.

                Ah, sorry. Bin halt zu selten hier. Hab's grad korrigiert und einen Reimport getestet, also bei mir gehts.

                Gruß
                Reiner

                mickymM Online
                mickymM Online
                mickym
                Most Active
                schrieb am zuletzt editiert von mickym
                #16

                @rewenode sagte in Datum und Zeitverarbeitung mit NodeRed:

                @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.
                

                Ah, sorry. Bin halt zu selten hier. Hab's grad korrigiert und einen Reimport getestet, also bei mir gehts.

                Gruß
                Reiner

                Ja mit den CodeTags geht es :+1: :+1: :+1: - da funkt der Browser nicht dazwischen. Das ist ja die Version ohne function Node aber global verfügbarer moment Bibliothek ;) - aber wie gesagt ich habe es ja nun ausprobiert - und mit Deiner Anleitung geht es ja direkt.

                (
                    $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');
                    $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');
                    
                    /* $b.diff($a) */
                    /* $moment.duration($b.diff($a)); */
                    
                    $globalContext('moment.duration')($b.diff($a)).humanize()
                
                )
                

                Aber wie gesagt - nur dafür würde ich moments nicht extra einbinden. Im Prinzip ist es halt nur die exakte Umrechnung von Zeitperioden erforderlich - ansonsten war ja gerade der Sinn des Postings - was alles mit Boardmitteln geht. ;)

                Für die eigentliche Umrechnung - für die aus meiner Sicht das Duration Objekt nützlich ist, habe ich ja unten gepostet.

                Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                R 1 Antwort Letzte Antwort
                0
                • mickymM mickym

                  @rewenode sagte in Datum und Zeitverarbeitung mit NodeRed:

                  @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                  @rewenode In Zukunft wäre es hilfreich Spoiler und CodeTags zu nutzen - sonst muss man immer scrollen um den Code zu kopieren.
                  

                  Ah, sorry. Bin halt zu selten hier. Hab's grad korrigiert und einen Reimport getestet, also bei mir gehts.

                  Gruß
                  Reiner

                  Ja mit den CodeTags geht es :+1: :+1: :+1: - da funkt der Browser nicht dazwischen. Das ist ja die Version ohne function Node aber global verfügbarer moment Bibliothek ;) - aber wie gesagt ich habe es ja nun ausprobiert - und mit Deiner Anleitung geht es ja direkt.

                  (
                      $b := $moment(timeB,['DD.MM.YYYY HH:mm:ss','x'],'de');
                      $a := $moment(timeA,['DD.MM.YYYY HH:mm:ss','x'],'de');
                      
                      /* $b.diff($a) */
                      /* $moment.duration($b.diff($a)); */
                      
                      $globalContext('moment.duration')($b.diff($a)).humanize()
                  
                  )
                  

                  Aber wie gesagt - nur dafür würde ich moments nicht extra einbinden. Im Prinzip ist es halt nur die exakte Umrechnung von Zeitperioden erforderlich - ansonsten war ja gerade der Sinn des Postings - was alles mit Boardmitteln geht. ;)

                  Für die eigentliche Umrechnung - für die aus meiner Sicht das Duration Objekt nützlich ist, habe ich ja unten gepostet.

                  R Offline
                  R Offline
                  rewenode
                  schrieb am zuletzt editiert von rewenode
                  #17

                  @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                  Aber wie gesagt - nur dafür würde ich moments nicht extra einbinden. Im Prinzip ist es halt nur die exakte Umrechnung von Zeitperioden erforderlich - ansonsten war ja gerade der Sinn des Postings - was alles mit Boardmitteln geht.

                  Ja, wollte nur eine Möglichkeit zeigen, moment.duration in node-red JSONata zugänglich zu machen.
                  Kann ja mal hilfreich sein, zumal man sich über diesen Trick nahezu beliebige Bibliotheken in JSONata zugänglich machen kann.

                  Ja, und was Bordmittel angeht, sind bei mir immer die die ich zur Lösung des Problems unbedingt brauche und die in node-red prinzipiell möglich sind;-)
                  Aber generell stimme ich zu, auch wenn man Vieles nicht so eng sehen darf.
                  Ich glaube nicht, dass eine komplexe JSONata-Funktion im change-node weniger undurchsichtig ist, wie die gleiche Funktionalität per Javascript im function-node.

                  Gruß
                  Reiner

                  mickymM 1 Antwort Letzte Antwort
                  0
                  • R rewenode

                    @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                    Aber wie gesagt - nur dafür würde ich moments nicht extra einbinden. Im Prinzip ist es halt nur die exakte Umrechnung von Zeitperioden erforderlich - ansonsten war ja gerade der Sinn des Postings - was alles mit Boardmitteln geht.

                    Ja, wollte nur eine Möglichkeit zeigen, moment.duration in node-red JSONata zugänglich zu machen.
                    Kann ja mal hilfreich sein, zumal man sich über diesen Trick nahezu beliebige Bibliotheken in JSONata zugänglich machen kann.

                    Ja, und was Bordmittel angeht, sind bei mir immer die die ich zur Lösung des Problems unbedingt brauche und die in node-red prinzipiell möglich sind;-)
                    Aber generell stimme ich zu, auch wenn man Vieles nicht so eng sehen darf.
                    Ich glaube nicht, dass eine komplexe JSONata-Funktion im change-node weniger undurchsichtig ist, wie die gleiche Funktionalität per Javascript im function-node.

                    Gruß
                    Reiner

                    mickymM Online
                    mickymM Online
                    mickym
                    Most Active
                    schrieb am zuletzt editiert von
                    #18

                    @rewenode sagte in Datum und Zeitverarbeitung mit NodeRed:

                    Ja, und was Bordmittel angeht, sind bei mir immer die die ich zur Lösung des Problems unbedingt brauche und die in node-red prinzipiell möglich sind;-)
                    Aber generell stimme ich zu, auch wenn man Vieles nicht so eng sehen darf.
                    Ich glaube nicht, dass eine komplexe JSONata-Funktion im change-node weniger undurchsichtig ist, wie die gleiche Funktionalität per Javascript im function-node.

                    Grundsätzlich stimme ich Dir zu - wobei ich function Nodes wirklich nur nutze, wenn alles andere sehr umständlich ist - weil ich eben denke, dass man sich die ganze Genialität von Node Red zunichte machen kann, wenn man zuviel in function Nodes packt. Sehe ich leider sehr häufig auch in Beispielen. ;) Letzlich kannst jeden Flow in eine function Node packen - aber dann brauche ich Node Red nicht - sondern kann ja - zumindest im iobroker - Javascript original schreiben. - OK als Standaloneversion hat man mit Node Red halt noch die Hardwareanbindung zur Verfügung. Aber die Klärung philosophischer Fragen sind dann wohl eher OT. ;)

                    Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                    1 Antwort Letzte Antwort
                    1
                    • mickymM mickym

                      Da ich mich nun mal 1-2 Tage intensiv mit der "eingebauten" Datums- und Uhrzeitverarbeitung in NodeRed beschäftigt habe, würde ich meine Erkenntnisse gerne teilen und vielleicht springt ja für den einen oder anderen was nützliches heraus.

                      Es geht in diesem Thread nicht um die Installation von zusätzlichen Nodes, die noch weitere Steuerungsmöglichkeiten bieten, sondern das was im Grundbaukasten bereits vorhanden ist.

                      Als wer noch mehr zu den Themen - scheduling oder Zeit filtern sucht - der soll sich mal folgende Nodes anschauen:

                      1. cronplus: https://flows.nodered.org/node/node-red-contrib-cron-plus - alles und noch mehr was iobroker mit dem cron als Scheduler bietet.
                      2. lightscheduler: https://flows.nodered.org/node/node-red-contrib-light-scheduler triggert - aber vor allem filtert nach Zeitereignissen
                      3. bigtimer - wobei ich zwar am Anfang begeistert war, aber ich inzwischen den Nutzen nicht mehr sehr hoch einschätze:
                        https://flows.nodered.org/node/node-red-contrib-bigtimer

                      Doch mir geht es ja nicht darum, mit welchen zusätzlichen Nodes man noch was machen kann, so wie man mit dem Grundbaukasten Zeit- und Datumsereignisse verarbeitet. Zum Schluss stelle ich dann noch einen kleinen Flow vor, den man als Timer nutzen kann. ;) - und zwar flexibler als bei den fertigen Nodes. ;)

                      Die Javascript Bibliothek, die in Node Red für die Datums- und Uhrzeit verarbeitung verfügbar ist, ist wohl die moment.js - und die Funktionen dieser Bibliothek werden ich nun im Verlauf weiter vorstellen - allerdings nicht über Javascript, sondern im Prinzip kann man diese ganze Bibliothek über JSONATA nutzen. Erst seit dem ich mich damit nun etwas näher beschäftigt habe, ergeben sich neuen Möglichkeiten, wobei ich auch noch hin und wieder ??? habe und deswegen vielleicht manches hier umständlich aussieht.

                      Dennoch kommt bei diesem Thread quasi kein natives Javascript zur Anwendung (also keine Function Nodes). ;) ;)

                      Ich denke, das im Blockly wohl auch die moment.js eingebunden ist - insofern kann man das vergleichen und gegenüberstellen.

                      Was im Standard nicht verfügbar ist, sind die Astrozeiten - dafür bieten sich aber alle 3 oben genannten Nodes an - die hier eine Vielzahl an Funktionen bieten - wer da mehr wissen will, meldet sich einfach.

                      Wie gesagt - sowohl im Blockly als auch in Node Red wurde für die Datums und Uhrzeitverarbeitung die moment.js Bibliothek verfügbar gemacht. Dabei geht in NodeRed in meinen Augen die Möglichkeiten der Nutzung über das hinaus, was üblicherweise in den Puzzleteilchen angeboten wird. ;)

                      Die volle Funktionalität der moment.js Bibliothek sollte man sich als Nachschlagewerk als Seite abspeichern:
                      https://momentjs.com/docs/#/parsing/

                      Anscheinend wird die Bibliothek nicht mehr weiter entwickelt - wird aber noch gewartet.

                      Nodes die vollkommen überflüssig sind, sind die moment.js Nodes - zumindest seitdem die moment.js Bibliothek unter NodeRed im JSONATA verfügbar gemacht wurde. Diese Nodes : https://flows.nodered.org/node/node-red-contrib-moment können also komplett ersetzt werden. Ich hatte die anfangs auch, werden nun aber rausgeschmissen.

                      Dreh- und Angelpunkte bei der Nutzung der moment.js Bibliothek sind natürlich die moment-Objekte. Diese sind keine date-Objekte von Javascript. Wer also die moment.js Bibliothek im Javascript nutzen will, muss sie nach Installation halt wie üblich mit require in den Code einbinden.

                      https://momentjs.com/docs/#/use-it/

                      var moment = require('moment'); // require
                      moment().format()
                      

                      In NodeRed ist es wie gesagt nicht erforderlich und man benötigt weder eine moment.js Insallation noch muss man Bibliohteken global verfügbar machen - außer man wollte die moment.js in function Nodes nutzen, was hier aber nicht der Fall ist und deswegen auch nicht weiter beschrieben wird. Hier geht es deshalb ausschließlich um die moment.js Funktionen, die JSONATA in NodeRed zur Verfügung stellt und die deshalb in verschiedenen Nodes wie der Change-Node, der Inject-Node, der Switch-Node, Join-Node etc. zur Verfügung steht und angewandt wird.

                      Im folgenden werde ich mich etwas an Blockly orientieren - da in meinen Augen die gleiche Bibliothek dahinter liegt und erklären, wie man das im JSONATA und in NodeRed umsetzt.

                      mickymM Online
                      mickymM Online
                      mickym
                      Most Active
                      schrieb am zuletzt editiert von mickym
                      #19

                      So noch ein paar Ergänzungen bzw. neue Erkenntnisse - wenn man es braucht.

                      Um Switch Nodes mit Zeitfenstern zu definieren - kann man die moments Bibliothek wie folgt nutzen:

                      [
                         {
                             "id": "9ec63930.7a41d8",
                             "type": "switch",
                             "z": "6e170384.60c96c",
                             "name": "Is es Zwischen 10:48 and 17:05?",
                             "property": "$moment().isBetween($moment(\"10:48\", \"HH:mm\"),$moment(\"17:05\", \"HH:mm\"),'minute')",
                             "propertyType": "jsonata",
                             "rules": [
                                 {
                                     "t": "true"
                                 },
                                 {
                                     "t": "false"
                                 }
                             ],
                             "checkall": "true",
                             "repair": false,
                             "outputs": 2,
                             "x": 1560,
                             "y": 2140,
                             "wires": [
                                 [
                                     "63e091c.217c47"
                                 ],
                                 [
                                     "57759b84.262984"
                                 ]
                             ]
                         },
                         {
                             "id": "76ac8c2.21f6f74",
                             "type": "inject",
                             "z": "6e170384.60c96c",
                             "name": "",
                             "props": [
                                 {
                                     "p": "payload"
                                 }
                             ],
                             "repeat": "",
                             "crontab": "",
                             "once": false,
                             "onceDelay": 0.1,
                             "topic": "",
                             "payload": "payload ist unwichtig, wo ich rauskomme ist wichtig ;)",
                             "payloadType": "str",
                             "x": 1330,
                             "y": 2140,
                             "wires": [
                                 [
                                     "9ec63930.7a41d8"
                                 ]
                             ]
                         },
                         {
                             "id": "63e091c.217c47",
                             "type": "debug",
                             "z": "6e170384.60c96c",
                             "name": "true",
                             "active": true,
                             "tosidebar": true,
                             "console": false,
                             "tostatus": false,
                             "complete": "payload",
                             "targetType": "msg",
                             "statusVal": "",
                             "statusType": "auto",
                             "x": 1810,
                             "y": 2120,
                             "wires": []
                         },
                         {
                             "id": "57759b84.262984",
                             "type": "debug",
                             "z": "6e170384.60c96c",
                             "name": "false",
                             "active": true,
                             "tosidebar": true,
                             "console": false,
                             "tostatus": false,
                             "complete": "payload",
                             "targetType": "msg",
                             "statusVal": "",
                             "statusType": "auto",
                             "x": 1810,
                             "y": 2160,
                             "wires": []
                         }
                      ]
                      

                      Ohne 3. Parameter bei dem Moments Funktion isBetween() - kann man dann über eckige bzw. runde Klammern noch angeben, ob die Anfangs- bzw. Endzeiten aus bzw. eingeschlossen sind in dem definierten Zeitraum.
                      Ohne Angabe ist die Anfangszeit eingeschlossen, die Endzeit ausgeschlossen!!!

                      5fcdf127-b7db-49bb-ab91-8d96e9db1c15-image.png

                      Braucht man wieder über Nachtzeiträume negiert man das entweder oder nutzt den false Ausgang. ;)

                      Am sichersten ist man, wenn man 2 moments Objekte vergleicht und dann auch noch die Genauigkeit spezifiziert:

                      $moment().isBetween($moment("10:48", "HH:mm"),$moment("17:05", "HH:mm"),'minute')
                      

                      Das 1. moments Objekt ist Anfangszeit mit Format zu richtigen Interpretation, das 2. moments Objekt die Endzeit mit Format zur richtigen Interpretation und die Genauigkeit ist auf Minute gesetzt.

                      Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                      1 Antwort Letzte Antwort
                      0
                      • mickymM mickym

                        Da ich mich nun mal 1-2 Tage intensiv mit der "eingebauten" Datums- und Uhrzeitverarbeitung in NodeRed beschäftigt habe, würde ich meine Erkenntnisse gerne teilen und vielleicht springt ja für den einen oder anderen was nützliches heraus.

                        Es geht in diesem Thread nicht um die Installation von zusätzlichen Nodes, die noch weitere Steuerungsmöglichkeiten bieten, sondern das was im Grundbaukasten bereits vorhanden ist.

                        Als wer noch mehr zu den Themen - scheduling oder Zeit filtern sucht - der soll sich mal folgende Nodes anschauen:

                        1. cronplus: https://flows.nodered.org/node/node-red-contrib-cron-plus - alles und noch mehr was iobroker mit dem cron als Scheduler bietet.
                        2. lightscheduler: https://flows.nodered.org/node/node-red-contrib-light-scheduler triggert - aber vor allem filtert nach Zeitereignissen
                        3. bigtimer - wobei ich zwar am Anfang begeistert war, aber ich inzwischen den Nutzen nicht mehr sehr hoch einschätze:
                          https://flows.nodered.org/node/node-red-contrib-bigtimer

                        Doch mir geht es ja nicht darum, mit welchen zusätzlichen Nodes man noch was machen kann, so wie man mit dem Grundbaukasten Zeit- und Datumsereignisse verarbeitet. Zum Schluss stelle ich dann noch einen kleinen Flow vor, den man als Timer nutzen kann. ;) - und zwar flexibler als bei den fertigen Nodes. ;)

                        Die Javascript Bibliothek, die in Node Red für die Datums- und Uhrzeit verarbeitung verfügbar ist, ist wohl die moment.js - und die Funktionen dieser Bibliothek werden ich nun im Verlauf weiter vorstellen - allerdings nicht über Javascript, sondern im Prinzip kann man diese ganze Bibliothek über JSONATA nutzen. Erst seit dem ich mich damit nun etwas näher beschäftigt habe, ergeben sich neuen Möglichkeiten, wobei ich auch noch hin und wieder ??? habe und deswegen vielleicht manches hier umständlich aussieht.

                        Dennoch kommt bei diesem Thread quasi kein natives Javascript zur Anwendung (also keine Function Nodes). ;) ;)

                        Ich denke, das im Blockly wohl auch die moment.js eingebunden ist - insofern kann man das vergleichen und gegenüberstellen.

                        Was im Standard nicht verfügbar ist, sind die Astrozeiten - dafür bieten sich aber alle 3 oben genannten Nodes an - die hier eine Vielzahl an Funktionen bieten - wer da mehr wissen will, meldet sich einfach.

                        Wie gesagt - sowohl im Blockly als auch in Node Red wurde für die Datums und Uhrzeitverarbeitung die moment.js Bibliothek verfügbar gemacht. Dabei geht in NodeRed in meinen Augen die Möglichkeiten der Nutzung über das hinaus, was üblicherweise in den Puzzleteilchen angeboten wird. ;)

                        Die volle Funktionalität der moment.js Bibliothek sollte man sich als Nachschlagewerk als Seite abspeichern:
                        https://momentjs.com/docs/#/parsing/

                        Anscheinend wird die Bibliothek nicht mehr weiter entwickelt - wird aber noch gewartet.

                        Nodes die vollkommen überflüssig sind, sind die moment.js Nodes - zumindest seitdem die moment.js Bibliothek unter NodeRed im JSONATA verfügbar gemacht wurde. Diese Nodes : https://flows.nodered.org/node/node-red-contrib-moment können also komplett ersetzt werden. Ich hatte die anfangs auch, werden nun aber rausgeschmissen.

                        Dreh- und Angelpunkte bei der Nutzung der moment.js Bibliothek sind natürlich die moment-Objekte. Diese sind keine date-Objekte von Javascript. Wer also die moment.js Bibliothek im Javascript nutzen will, muss sie nach Installation halt wie üblich mit require in den Code einbinden.

                        https://momentjs.com/docs/#/use-it/

                        var moment = require('moment'); // require
                        moment().format()
                        

                        In NodeRed ist es wie gesagt nicht erforderlich und man benötigt weder eine moment.js Insallation noch muss man Bibliohteken global verfügbar machen - außer man wollte die moment.js in function Nodes nutzen, was hier aber nicht der Fall ist und deswegen auch nicht weiter beschrieben wird. Hier geht es deshalb ausschließlich um die moment.js Funktionen, die JSONATA in NodeRed zur Verfügung stellt und die deshalb in verschiedenen Nodes wie der Change-Node, der Inject-Node, der Switch-Node, Join-Node etc. zur Verfügung steht und angewandt wird.

                        Im folgenden werde ich mich etwas an Blockly orientieren - da in meinen Augen die gleiche Bibliothek dahinter liegt und erklären, wie man das im JSONATA und in NodeRed umsetzt.

                        mickymM Online
                        mickymM Online
                        mickym
                        Most Active
                        schrieb am zuletzt editiert von mickym
                        #20

                        @mickym Wieder ein Test der positiv verlief mit der Moment Bibliothek.

                        Auch die Verarbeitung von Monatsnamen um moment-Objekte zu erstellen, funktioniert einwandfrei:

                        a576556e-3318-415c-8229-b78815c61d81-image.png

                        $moment(payload,'DD.MMMM YYYY','de').format('DD.MM.YY - HH:mm')
                        

                        Auch Umlaute wie März werden korrekt verarbeitet.

                        Bei abgekürzten Monatsnamen muss man im Deutschen darauf achten, dass die Strings immer mit 4 Zeichen abgekürzt werden.

                        $moment(payload,'DD.MMM. YYYY','de').format('DD.MM.YY - HH:mm')
                        

                        Sind die Monatsnamen 4 Buchstaben oder kürzer werden sie voll ausgeschrieben (wichtig also März ausschreiben), ansonsten die ersten 3 Buchstaben + Punkt. Eine ähnliche Absonderlichkeit ergab sich bei 3stelligen Wochentagsnamen die immer 2 Buchstaben + Punkt enthalten:

                        d0ce53a1-6316-477f-890e-c3577c35ec75-image.png

                        Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                        1 Antwort Letzte Antwort
                        0
                        • mickymM Online
                          mickymM Online
                          mickym
                          Most Active
                          schrieb am zuletzt editiert von mickym
                          #21

                          Einen weitere Anwendung bzw. Flow - um Zeitspannen der moments- Bibliothek zu nutzen, um sich in seiner Visualisierung nicht den Zeitstempel der letzten Meldung, sondern die Zeitspanne seit der letzten Meldung ausgeben zu lassen.

                          42b10732-55ce-45e1-a69e-1d0af478a00e-image.png

                          etwas später:

                          d8fe7d91-8981-4c3e-9c9e-dd5baefc7c3f-image.png

                          Die Texte werden automatisch durch die Humanize Funktion ausgegeben und sind also nicht änderbar und folgen folgendem Schema:

                          c370e5ad-5625-4e31-b0c6-a4fe253f4223-image.png

                          Ich gebe es hier direkt im NR-Dashboard aus, man könnte es aber auch natürlich in eigene Datenpunkte schreiben lassen und diese dann im VIS oder einer anderen Visualisierung ausgeben lassen. Es werden ja Strings zurückgegeben.

                          Alle Geräte der letzte Meldungstatus man anzeigen lassen will, leiten man in einen zentralen Flow:

                          0c88002f-1f6e-4e6f-8649-91941eb4763f-image.png

                          Die Daten werden alle im Flowtext gesammelt und momentan nicht mit einem festen zeitlichen Trigger ausgegeben, sondern sobald sich ein Gerät meldet, werden alle anderen mit aktualisiert. Das belastet das System weniger. Man könne aber einen zeitlichen Trigger leicht realisieren, indem man die Inject-Node triggern lässt. Ansonsten triggert für Node-Red Dashboard Nutzer natürlich auch ein Wechsel der Seiten.

                          Hiermal der Kern des Flows - für Ein- und Ausgang muss man selbst sorgen. Für die einzelnen Geräte wird immer das topic zur Identifikation genutzt (ggf. muss man der Nachricht halt selbst ein Topic verpassen).

                          [
                             {
                                 "id": "753c1b7eb6df5805",
                                 "type": "function",
                                 "z": "c297780ec96fcd08",
                                 "name": "Zeitstempel in fState",
                                 "func": "flow.set ('fState.' + msg.topic, msg.payload);\nreturn msg;",
                                 "outputs": 1,
                                 "noerr": 0,
                                 "initialize": "",
                                 "finalize": "",
                                 "libs": [],
                                 "x": 700,
                                 "y": 520,
                                 "wires": [
                                     [
                                         "2a3f9c0bbc0ea013"
                                     ]
                                 ]
                             },
                             {
                                 "id": "abee507e5c1eb5bf",
                                 "type": "change",
                                 "z": "c297780ec96fcd08",
                                 "name": "",
                                 "rules": [
                                     {
                                         "t": "set",
                                         "p": "payload",
                                         "pt": "msg",
                                         "to": "$moment()\t",
                                         "tot": "jsonata"
                                     }
                                 ],
                                 "action": "",
                                 "property": "",
                                 "from": "",
                                 "to": "",
                                 "reg": false,
                                 "x": 470,
                                 "y": 520,
                                 "wires": [
                                     [
                                         "753c1b7eb6df5805"
                                     ]
                                 ]
                             },
                             {
                                 "id": "2a3f9c0bbc0ea013",
                                 "type": "change",
                                 "z": "c297780ec96fcd08",
                                 "name": "",
                                 "rules": [
                                     {
                                         "t": "set",
                                         "p": "payload",
                                         "pt": "msg",
                                         "to": "fState",
                                         "tot": "flow"
                                     }
                                 ],
                                 "action": "",
                                 "property": "",
                                 "from": "",
                                 "to": "",
                                 "reg": false,
                                 "x": 930,
                                 "y": 520,
                                 "wires": [
                                     [
                                         "c1c76a5e7c06c5d8"
                                     ]
                                 ]
                             },
                             {
                                 "id": "c1c76a5e7c06c5d8",
                                 "type": "split",
                                 "z": "c297780ec96fcd08",
                                 "name": "",
                                 "splt": "\\n",
                                 "spltType": "str",
                                 "arraySplt": 1,
                                 "arraySpltType": "len",
                                 "stream": false,
                                 "addname": "topic",
                                 "x": 1110,
                                 "y": 520,
                                 "wires": [
                                     [
                                         "cf60e787307a4d6f"
                                     ]
                                 ]
                             },
                             {
                                 "id": "cf60e787307a4d6f",
                                 "type": "change",
                                 "z": "c297780ec96fcd08",
                                 "name": "",
                                 "rules": [
                                     {
                                         "t": "set",
                                         "p": "payload",
                                         "pt": "msg",
                                         "to": "$moment(payload).locale('de').fromNow()",
                                         "tot": "jsonata"
                                     }
                                 ],
                                 "action": "",
                                 "property": "",
                                 "from": "",
                                 "to": "",
                                 "reg": false,
                                 "x": 1280,
                                 "y": 520,
                                 "wires": [
                                     [
                                         "8d22e1665c558f9b"
                                     ]
                                 ]
                             },
                             {
                                 "id": "8215e211b17961ce",
                                 "type": "ui_ui_control",
                                 "z": "c297780ec96fcd08",
                                 "name": "",
                                 "events": "all",
                                 "x": 740,
                                 "y": 480,
                                 "wires": [
                                     [
                                         "2a3f9c0bbc0ea013"
                                     ]
                                 ]
                             },
                             {
                                 "id": "74b5b9f8b6250713",
                                 "type": "inject",
                                 "z": "c297780ec96fcd08",
                                 "name": "Trigger",
                                 "props": [
                                     {
                                         "p": "payload"
                                     }
                                 ],
                                 "repeat": "",
                                 "crontab": "",
                                 "once": false,
                                 "onceDelay": 0.1,
                                 "topic": "",
                                 "payload": "true",
                                 "payloadType": "bool",
                                 "x": 750,
                                 "y": 560,
                                 "wires": [
                                     [
                                         "2a3f9c0bbc0ea013"
                                     ]
                                 ]
                             }
                          ]
                          

                          Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                          1 Antwort Letzte Antwort
                          2
                          • mickymM Online
                            mickymM Online
                            mickym
                            Most Active
                            schrieb am zuletzt editiert von
                            #22

                            Hier mal wieder eine genial, einfache Methode, wie man verschiedene Bestandteile eines Datums mit Hilfe der moments library analysieren und als Objekt zur Verfügung stellen kann - um sie ggf. auch dann wieder in einzelne Datenpunkte zu zerlegen.

                            Als erstes nutzt man halt den Formatstring, um verschiedene Bestandteile des Datums zu ermitteln und trennt diese mit Kommas.

                            Zum Beispiel:

                            $moment().format('YYYY,Q,M,DDD,W,e,H')	
                            

                            Das Ganze soll auch noch in Zahlenwerte konvertiert werden - deshalb habe ich alles vermieden, was Wörter ausspuckt.

                            YYYY = gibt die 4 stellige Jahreszahl an
                            Q = das Quartal (1-4)
                            M = der Monat (1-12)
                            DDD = Tag des Jahres (1-365(bzw. 366))
                            W = Kalenderwoche europäisch
                            e = Wochentag beginnend mit 0 = So.
                            H = Stunde (0-23)

                            Der Trick ist nun, dass man einfach eine CSV Node zur Konvertierung nutzt.

                            d2d05e41-8e0f-4492-b795-ada92c289e2f-image.png

                            Mit Zahlenwerte ermitteln wird automatisch die Konvertierung durchgeführt und mit den Namen als Spalte werden die Objekteigenschaften gleich richtig benannt.

                            Somit ergibt dieser Flow - ein Objekt mit den entsprechenden Eigenschaften:

                            c1bbd2a2-8fde-45f7-8c59-8c3e884d4af1-image.png

                            Hier der einfache Flow:

                            [
                               {
                                   "id": "90e855fff6300508",
                                   "type": "debug",
                                   "z": "a3b6dd9f9b565227",
                                   "name": "Datum analysiert",
                                   "active": true,
                                   "tosidebar": true,
                                   "console": false,
                                   "tostatus": false,
                                   "complete": "payload",
                                   "targetType": "msg",
                                   "statusVal": "",
                                   "statusType": "auto",
                                   "x": 890,
                                   "y": 4460,
                                   "wires": []
                               },
                               {
                                   "id": "81785bd9ea91f30d",
                                   "type": "inject",
                                   "z": "a3b6dd9f9b565227",
                                   "name": "",
                                   "props": [
                                       {
                                           "p": "payload"
                                       },
                                       {
                                           "p": "topic",
                                           "vt": "str"
                                       }
                                   ],
                                   "repeat": "",
                                   "crontab": "",
                                   "once": false,
                                   "onceDelay": 0.1,
                                   "topic": "",
                                   "payload": "true",
                                   "payloadType": "bool",
                                   "x": 330,
                                   "y": 4460,
                                   "wires": [
                                       [
                                           "37cb9f15de0e0171"
                                       ]
                                   ]
                               },
                               {
                                   "id": "37cb9f15de0e0171",
                                   "type": "change",
                                   "z": "a3b6dd9f9b565227",
                                   "name": "",
                                   "rules": [
                                       {
                                           "t": "set",
                                           "p": "payload",
                                           "pt": "msg",
                                           "to": "$moment().format('YYYY,Q,M,DDD,W,e,H')\t",
                                           "tot": "jsonata"
                                       }
                                   ],
                                   "action": "",
                                   "property": "",
                                   "from": "",
                                   "to": "",
                                   "reg": false,
                                   "x": 510,
                                   "y": 4460,
                                   "wires": [
                                       [
                                           "16121dc11c791032"
                                       ]
                                   ]
                               },
                               {
                                   "id": "16121dc11c791032",
                                   "type": "csv",
                                   "z": "a3b6dd9f9b565227",
                                   "name": "",
                                   "sep": ",",
                                   "hdrin": "",
                                   "hdrout": "none",
                                   "multi": "one",
                                   "ret": "\\n",
                                   "temp": "year,quarter,month,day of year,week of year,day of week,hour",
                                   "skip": "0",
                                   "strings": true,
                                   "include_empty_strings": "",
                                   "include_null_values": "",
                                   "x": 710,
                                   "y": 4460,
                                   "wires": [
                                       [
                                           "90e855fff6300508"
                                       ]
                                   ]
                               }
                            ]
                            

                            Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                            1 Antwort Letzte Antwort
                            1
                            • mickymM Online
                              mickymM Online
                              mickym
                              Most Active
                              schrieb am zuletzt editiert von mickym
                              #23

                              Hier mal wieder eine neue Anwendung mit der $moments Bibliothek um zu überprüfen ob der aktuelle Zeitpunkt in einem von mehreren definierten Zeitfenstern liegt.

                              Praktische Anwendung könnte zum Beispiel sein, dass ein Gerät oder Heizung zu bestimmten Zeit innerhalb eines Tages eingeschaltet werden soll oder nicht. Dies habe ich gerade als Lösungsvorschlag für einem Beispielflow erstellt.

                              [
                                 {
                                     "id": "811d7797475ab2a4",
                                     "type": "change",
                                     "z": "6e170384.60c96c",
                                     "name": "Check mehrer Zeitfenster",
                                     "rules": [
                                         {
                                             "t": "set",
                                             "p": "payload",
                                             "pt": "msg",
                                             "to": "[]",
                                             "tot": "json"
                                         },
                                         {
                                             "t": "set",
                                             "p": "payload[0]",
                                             "pt": "msg",
                                             "to": "$moment().isBetween($moment(\"06:00\", \"HH:mm\"),$moment(\"08:00\", \"HH:mm\"),'minute','[)')",
                                             "tot": "jsonata"
                                         },
                                         {
                                             "t": "set",
                                             "p": "payload[1]",
                                             "pt": "msg",
                                             "to": "$moment().isBetween($moment(\"11:00\", \"HH:mm\"),$moment(\"13:00\", \"HH:mm\"),'minute','[)')",
                                             "tot": "jsonata"
                                         },
                                         {
                                             "t": "set",
                                             "p": "payload[2]",
                                             "pt": "msg",
                                             "to": "$moment().isBetween($moment(\"19:00\", \"HH:mm\"),$moment(\"23:00\", \"HH:mm\"),'minute','[)')",
                                             "tot": "jsonata"
                                         },
                                         {
                                             "t": "set",
                                             "p": "payload",
                                             "pt": "msg",
                                             "to": "payload ~> $reduce (function($A, $i){$A or $i})",
                                             "tot": "jsonata"
                                         }
                                     ],
                                     "action": "",
                                     "property": "",
                                     "from": "",
                                     "to": "",
                                     "reg": false,
                                     "x": 1610,
                                     "y": 2060,
                                     "wires": [
                                         [
                                             "a49f55322d881c9c"
                                         ]
                                     ]
                                 }
                              ]
                              

                              Die Prüfung der Zeitfenster speichert man in einem Array:

                              594f25d1-3ec6-4876-9e57-69bb4609b0b3-image.png

                              Zum Schluss verknüpft man alle Arraywerte mit ODER, um zu überprüfen, ob aktuell eines der definierten Zeitfenster WAHR ist und gibt das Ergebnis als ein Boolean aus.

                              Außerdem wird hier die Anwendung des Ein- und Ausschlusses der Zeitgrenzen gezeigt. Normalerweise sind die angegebenen Zeitgrenzen ausgeschlossen. Mit eckigen Klammern schließt man diese ein, mit runden Klammern schließt man sie aus.

                              Das Beispiel [) schließt also den Anfang ein, das Ende aus.

                              Ergänzung: Man kann das ganze natürlich auch in einen Ausdruck verfrachten, dann schaut das noch schöner aus:

                              a52dc24e-2884-4c57-9ac7-78558307514f-image.png

                              [
                                 {
                                     "id": "218060d1fbd89410",
                                     "type": "change",
                                     "z": "578345f4956c7a57",
                                     "name": "Check Zeitfenster",
                                     "rules": [
                                         {
                                             "t": "set",
                                             "p": "payload",
                                             "pt": "msg",
                                             "to": "[\t   $moment().isBetween(\t       $moment(\"06:00\", \"HH:mm\"),\t       $moment(\"08:00\", \"HH:mm\"),\t       'minute',\t       '[)'\t   ),\t   $moment().isBetween(\t       $moment(\"11:00\", \"HH:mm\"),\t       $moment(\"13:00\", \"HH:mm\"),\t       'minute',\t       '[)'\t   ),\t   $moment().isBetween(\t       $moment(\"18:00\", \"HH:mm\"),\t       $moment(\"23:00\", \"HH:mm\"),\t       'minute',\t       '[)'\t   )\t]~> $reduce (function($A, $i){$A or $i})",
                                             "tot": "jsonata"
                                         }
                                     ],
                                     "action": "",
                                     "property": "",
                                     "from": "",
                                     "to": "",
                                     "reg": false,
                                     "x": 450,
                                     "y": 2640,
                                     "wires": [
                                         [
                                             "85ad88e10464701c"
                                         ]
                                     ]
                                 }
                              ]
                              

                              Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                              1 Antwort Letzte Antwort
                              1
                              • B Offline
                                B Offline
                                Bernhard59
                                schrieb am zuletzt editiert von
                                #24

                                Hallo zusammen, ich habe auch ein Problem mit der Zeit... wer hat das nicht :wink:

                                Ich möchte in der Regelung meiner Solaranlage den Wechselrichter abschalten, wenn er eine bestimmte Zeit keine Werte mehr über meinen DTU Adapter zu meinem Flow gesendet hat. Das kommt manchmal vor und um zu verhindern, daß mein Solarakku zu tief entladen wird, soll der Wechselrichter abgeschaltet werden. Dazu möchte ich den " timestamp of last inverter statistics udpate" mit der aktuellen Uhrzeit vergleichen und wenn die Differenz der beiden z.B. größer als 10 min ist, soll der Befehl zum Abschalten erfolgen. Ich weiß, wie ich den " timestamp of last inverter statistics udpate" mit einer iobroker Node auslese und wie ich abschalte, aber wie ich die beiden Zeiten verwenden und daraus eine Differenzüberschreitung ermitteln kann ist mir noch unklar.

                                Vielleicht kann mir ja hier jemand helfen... besten Dank im Voraus...

                                mickymM 1 Antwort Letzte Antwort
                                0
                                • B Bernhard59

                                  Hallo zusammen, ich habe auch ein Problem mit der Zeit... wer hat das nicht :wink:

                                  Ich möchte in der Regelung meiner Solaranlage den Wechselrichter abschalten, wenn er eine bestimmte Zeit keine Werte mehr über meinen DTU Adapter zu meinem Flow gesendet hat. Das kommt manchmal vor und um zu verhindern, daß mein Solarakku zu tief entladen wird, soll der Wechselrichter abgeschaltet werden. Dazu möchte ich den " timestamp of last inverter statistics udpate" mit der aktuellen Uhrzeit vergleichen und wenn die Differenz der beiden z.B. größer als 10 min ist, soll der Befehl zum Abschalten erfolgen. Ich weiß, wie ich den " timestamp of last inverter statistics udpate" mit einer iobroker Node auslese und wie ich abschalte, aber wie ich die beiden Zeiten verwenden und daraus eine Differenzüberschreitung ermitteln kann ist mir noch unklar.

                                  Vielleicht kann mir ja hier jemand helfen... besten Dank im Voraus...

                                  mickymM Online
                                  mickymM Online
                                  mickym
                                  Most Active
                                  schrieb am zuletzt editiert von
                                  #25

                                  @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                  Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                                  B 1 Antwort Letzte Antwort
                                  0
                                  • mickymM mickym

                                    @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                    B Offline
                                    B Offline
                                    Bernhard59
                                    schrieb am zuletzt editiert von
                                    #26

                                    @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                                    @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                    Danke für die schnelle Antwort... und wie funktioniert das ?

                                    mickymM 1 Antwort Letzte Antwort
                                    0
                                    • B Bernhard59

                                      @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                                      @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                      Danke für die schnelle Antwort... und wie funktioniert das ?

                                      mickymM Online
                                      mickymM Online
                                      mickym
                                      Most Active
                                      schrieb am zuletzt editiert von mickym
                                      #27

                                      @bernhard59 sagte in Datum und Zeitverarbeitung mit NodeRed:

                                      @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                                      @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                      Danke für die schnelle Antwort... und wie funktioniert das ?

                                      6ad223ac-e357-48f4-9fe8-47b2ef5649e0-image.png

                                      und halt vorne dran die Node oder die Nachrichten die Du sonst empfängst, also parallel schalten. Sind die 10 Minuten abgelaufen, wird der trigger automatisch wieder scharf-

                                      Jeder Flow bzw. jedes Script, das ich hier poste implementiert jeder auf eigene Gefahr. Flows und Scripts können Fehler aufweisen und weder der Seitenbetreiber noch ich persönlich können hierfür haftbar gemacht werden. Das gleiche gilt für Empfehlungen aller Art.

                                      B 1 Antwort Letzte Antwort
                                      0
                                      • mickymM mickym

                                        @bernhard59 sagte in Datum und Zeitverarbeitung mit NodeRed:

                                        @mickym sagte in Datum und Zeitverarbeitung mit NodeRed:

                                        @bernhard59 Für so was nutzt man eine trigger Node und führt keine Berechnungen durch.

                                        Danke für die schnelle Antwort... und wie funktioniert das ?

                                        6ad223ac-e357-48f4-9fe8-47b2ef5649e0-image.png

                                        und halt vorne dran die Node oder die Nachrichten die Du sonst empfängst, also parallel schalten. Sind die 10 Minuten abgelaufen, wird der trigger automatisch wieder scharf-

                                        B Offline
                                        B Offline
                                        Bernhard59
                                        schrieb am zuletzt editiert von
                                        #28

                                        @mickym

                                        Danke... dann versuche ich das mal...

                                        B 1 Antwort Letzte Antwort
                                        0
                                        • B Bernhard59

                                          @mickym

                                          Danke... dann versuche ich das mal...

                                          B Offline
                                          B Offline
                                          Bernhard59
                                          schrieb am zuletzt editiert von
                                          #29

                                          Perfekt, es funktioniert und soooo einfach... DANKE

                                          mickymM 1 Antwort Letzte Antwort
                                          0
                                          Antworten
                                          • In einem neuen Thema antworten
                                          Anmelden zum Antworten
                                          • Älteste zuerst
                                          • Neuste zuerst
                                          • Meiste Stimmen


                                          Support us

                                          ioBroker
                                          Community Adapters
                                          Donate

                                          390

                                          Online

                                          32.5k

                                          Benutzer

                                          81.7k

                                          Themen

                                          1.3m

                                          Beiträge
                                          Community
                                          Impressum | Datenschutz-Bestimmungen | Nutzungsbedingungen | Einwilligungseinstellungen
                                          ioBroker Community 2014-2025
                                          logo
                                          • Anmelden

                                          • Du hast noch kein Konto? Registrieren

                                          • Anmelden oder registrieren, um zu suchen
                                          • Erster Beitrag
                                            Letzter Beitrag
                                          0
                                          • Home
                                          • Aktuell
                                          • Tags
                                          • Ungelesen 0
                                          • Kategorien
                                          • Unreplied
                                          • Beliebt
                                          • GitHub
                                          • Docu
                                          • Hilfe