Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Visualisierung
    4. [gelöst] - VIS Intance Id - automatisiert ermitteln

    NEWS

    • Neuer Blog: Fotos und Eindrücke aus Solingen

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

    • ioBroker goes Matter ... Matter Adapter in Stable

    [gelöst] - VIS Intance Id - automatisiert ermitteln

    This topic has been deleted. Only users with topic management privileges can see it.
    • M
      mguenther @BananaJoe last edited by mguenther

      @bananajoe
      habe es mit Hilfe von @liv-in-sky und 2 Web-Instanzen hinbekommen. Funktioniert so wie es soll.
      Ging bei mir um unterschiedliche Darstellung in 2 Tablets.

      Danke für alle Anregungen 🙂

      Marcus

      1 Reply Last reply Reply Quote 0
      • T
        tobrog last edited by

        @bananajoe Vielen Dank für die ausführliche Anleitung. Tolles Projekt 👍
        Ich habe das jetzt auch mal umgesetzt, bin aber anfangs erst verzweifelt, weil es nicht lief. Ich habe zur Abfrage der IP lighttpd genutzt, da der eh schon bei mir lief. Über "meineip.php" lies sich die IP auch im browser abfragen, aber die Einbindung in VIS lieferte kein Ergebnis. Hat etwas gedauert bis ich gemerkt habe, dass die Anfrage vom Server geblockt wird wegen "Access-Control-Allow-Origin" 🙄
        Nachdem ich die cross site Anfragen in der lighttpd.conf mit

        setenv.add-response-header = ( "Access-Control-Allow-Origin" => "*" )
        

        erlaubt habe, klappt es auch in VIS mit der IP Abfrage.

        Evtl hilft das dem ein oder anderen, der hier in die gleichen Probleme läuft.

        1 Reply Last reply Reply Quote 1
        • A
          Andersmacher @BananaJoe last edited by

          @bananajoe Ersteinmal Danke für den Beitrag!

          Ich suche selber schon seit längerer Zeit nach einer Lösung für das "Instanz-ID"-Problem und bin gestern zufällig auf Deinen Beitrag gestoßen. Ich wollte Deine IP-Adressen-Variante ´mal nachvollziehen, und habe daher folgendes gemacht (ioBroker läuft bei mir auf Raspi4 8GB unter Debian Bookworm und die VIS rufe ich zum Testen in FireFox auf):

          • apache-server und php installiert.
          • Die "/etc/apache2/apache2.conf" kann ich nur verändern/speichern, wenn ich nano mit sudo aufrufe. Das hast Du in Deinem Beitrag nicht so angegeben. Ist das bei Dir anders/nicht nötig gewesen? Habe ich mir damit die nachfolgenden Probleme erzeugt? Wie hätte ich das besser machen können?
          • Den Befehl "systemctl restart apache2.service" kann ich auch nur mit sudo ausführen!
            Aber selbst dann ergibt sich folgendes Problem:
          • Mit der Original .conf klappt ein Restart des apache und ich kann via ipAdresse/meineIP.php die von Dir dargestellte Ausgabe sehen. (also der apache läuft und php funzt).
          • Wenn ich in der .conf aber die Zeile "Header set Access-Control-Allow-Origin "*"" hinten angefügt habe, erhalte ich beim Restart des apache die Meldungen (im Terminal komplett in rot):
          pi@Pi4B-8GB:~$ sudo systemctl restart apache2.service
          Job for apache2.service failed because the control process exited with error code.
          See "systemctl status apache2.service" and "journalctl -xeu apache2.service" for details.
          
          

          Diese Hinweise helfen mir mit meinem 1/4-Wissen nicht wirklich weiter, denn aus

          pi@Pi4B-8GB:~$ systemctl status apache2.service
          × apache2.service - The Apache HTTP Server
               Loaded: loaded (/lib/systemd/system/apache2.service; enabled; preset: enabled)
               Active: failed (Result: exit-code) since Sun 2024-09-29 18:38:20 CEST; 9min ago
             Duration: 39min 51.418s
                 Docs: https://httpd.apache.org/docs/2.4/
              Process: 199266 ExecStart=/usr/sbin/apachectl start (code=exited, status=1/FAILURE)
                  CPU: 94ms
          

          und

          pi@Pi4B-8GB:~$ journalctl -xeu apache2.service
          Hint: You are currently not seeing messages from other users and the system.
                Users in groups 'adm', 'systemd-journal' can see all messages.
                Pass -q to turn off this notice.
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          ~
          -- No entries --
          
          

          werd´ ich auch nicht schlauer. (;-(

          Ich habe das "basic - Screen Resolution Widget" in meine Start-View eingefügt und via Script-code

          vis.setValue('0_userdata.0.Vis_Navigation.VIS_Instanz', "Hallo");
          

          kann ich beim Starten von VIS / der View auch in den ioBroker-Datenpunkt schreiben, aber eben nur "statisch".
          Das "dynamische" Schreiben der Instanz-ID mit Deinem Skript klappt jedoch nicht, was vermutlich daran liegt, daß der Apache bei mir nur läuft, wenn die .conf nicht ergänzt ist.

          Hättest Du einen Hinweis für mich, wo/wie ich da zu einer Lösung ansetzen kann?

          Du hast in Deinem Skript ein paar auskommentierte Befehlszeilen:

                  //vis.generateInstance();
          
                  // alert(vis.instance);
          
                      //alert(data.ipAddress.replaceAll('.','-'));
          

          Ist das bewußt so, oder müssen die aktiviert werden?

          BananaJoe 1 Reply Last reply Reply Quote 0
          • BananaJoe
            BananaJoe Most Active @Andersmacher last edited by

            @andersmacher also, ja , richtig, egal was du änderst du musst das immer mit root-Rechten machen.
            Also überall(!) ein sudo davor schreiben (oder mit sudo -i kurzzeitig dauerhaft zum root wechseln),
            ergo auch beim Befehl journalctl -xeu apache2.service

            Vermutlich hast du einen Schreibfehler drin. Kann auch sein das du versehentlich woanders ein Zeichen eingefügt hast,
            das Log sollte dir das schnell erzählen. (Passiert mir auch schon mal)

            Kann auch sein das bei dir das Modul "headers" nicht aktiv ist:

            sudo a2enmod headers
            

            Wenn er dir sagt das er es aktivieren musste, den Apache noch mal neu starten.
            Die Header-Zeile steht bei mir einfach unten drunter
            9de0d9ab-dcd9-4617-9752-3ac24a0dc542-image.png

            A 1 Reply Last reply Reply Quote 0
            • A
              Andersmacher @BananaJoe last edited by

              @bananajoe sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

              Kann auch sein das bei dir das Modul "headers" nicht aktiv ist:

              sudo a2enmod headers
              

              Das war es! Wenn man nur immer und überall so viel, gute und schnelle Hilfe hätte, wie hier.
              Vielen Dank!

              1 Reply Last reply Reply Quote 0
              • A
                Andersmacher last edited by

                @BananaJoe Nochmals Danke für Dein Skript und die Hilfe, es bei mir zum Laufen zu bringen!

                Ich habe daran noch etwas "herumgebastelt", um es für mich anzupassen und hoffe es ist ok, wenn ich das hier beschreibe/"vorstelle", weil es aus meiner Sicht direkt zum Thema paßt und für den Fall, daß das auch für andere interressant wäre. Falls nicht, gern auch in einen anderen / separaten Beitrag verschieben.

                Was war mein Bestreben?:
                Dein Skipt benutzt ja

                            switch (data.ipAddress) {
                                case "192.168.1.80":
                

                also "hart gekodete" Fallunterscheidungen, um die Vis-Instanz-ID in den passenden ioBroker-Datenpunkt zu schreiben.
                Bei mir führte das dazu, daß ich beim "Herumprobieren" (u. a. auch als ich weitere Geräte mit einbeziehen wollte) öfter mal das VIS-Projekt erneut auch auf die bereits berücksichtigten Geräte spielen mußte, was bei mir auf Grund des Alters der Geräte z. T. recht lange dauert. Außerdem mußte ich dann jedesmal im Vis-Editor (das Skript) UND im ioBroker (die Datenpunkte) anpassen, also an mehr als eine Stelle ´ran.
                Da ich zwischenzeitlich mit Blockly einigermaßen umgehen kann, habe ich mir angewöhnt, alles, was potentiell "dynamisch sein könnte", möglichst dort und nur "an einer Stelle" zu machen.

                Was habe ich angepaßt/ergänzt?:
                Ich habe ein Blockly erstellt, daß mir die erforderlichen/gewünschten Datenpunkte erzeugt und soweit möglich "initialisiert". Den Code zum Importieren mußte ich als Datei beifügen, weil er wohl zu lang für´s Forum ist. Daher hier nur der JS-Code des Blockly
                Blockly_Exportcode.xml :

                var id, common, Debug_Info, Liste_Vis_Instanznamen, Liste_IP_Adressen, Vis_Instanznamen, k, IP_Adressen, Datenpunkte_f_C3_BCr_Instanzen, i, j;
                
                // Beschreibe diese Funktion …
                async function Datenpunkt_erzeugen(id, common) {
                    createState(id, JSON.parse(common));
                }
                
                
                // Hier werden für die Vis-Instanzen, die vom
                // ioBroker separat/einzeln angesteuert werden
                // sollen, Datenpunkte definiert.
                //
                // Initialisierungen
                Debug_Info = false;
                //
                // Den Ordner erzeugen, unter dem die Instanzen
                // liegen sollen und 2 Datenpunkte eintragen.
                await Datenpunkt_erzeugen('0_userdata.0.Vis_Navigation.Instanzen.Instanznamen', '{"name":"Instanznamen", "type":"string", "read":true, "write":true, "role":"Merker"}');
                await Datenpunkt_erzeugen('0_userdata.0.Vis_Navigation.Instanzen.IP_Adressen', '{"name":"IP_Adressen", "type":"string", "read":true, "write":true, "role":"Merker"}');
                // Liste mit allen gewünschten/denkbaren Instanznamen
                // erzeugen:
                Liste_Vis_Instanznamen = ['Gerät_1', 'Gerät_2', 'Gerät_3'];
                // Für jeden Instanznamen die IP-Adresse angeben,
                // auf der die Instanz läuft. Das setzt voraus, daß
                // das Gerät, auf dem die VIS-Instanz läuft, immer
                // die gleiche IP-Adresse hat!.
                // Die Zuordnung zu den Instanznamen erfolgt über die
                // angegebene Reihenfolge!
                Liste_IP_Adressen = ['192.168.1.1', '192.168.1.2', '192.168.1.3'];
                // Alle Instanznamen mit ";" getrennt in einen
                // Datenpunkt des Ordners eintragen:
                Vis_Instanznamen = '';
                for (var k_index in Liste_Vis_Instanznamen) {
                  k = Liste_Vis_Instanznamen[k_index];
                  Vis_Instanznamen = [Vis_Instanznamen,k,';'].join('');
                }
                console.info(Vis_Instanznamen);
                // Das letzte ";" wieder entfernen:
                setStateDelayed('0_userdata.0.Vis_Navigation.Instanzen.Instanznamen' /* Instanznamen */, (Vis_Instanznamen.slice(0, Vis_Instanznamen.length - 1)), true, 1000, false);
                // Alle IP-Adressen mit ";" getrennt in einen
                // Datenpunkt des Ordners eintragen:
                IP_Adressen = '';
                for (var k_index2 in Liste_IP_Adressen) {
                  k = Liste_IP_Adressen[k_index2];
                  IP_Adressen = [IP_Adressen,k,';'].join('');
                }
                console.info(IP_Adressen);
                // Das letzte ";" wieder entfernen:
                setStateDelayed('0_userdata.0.Vis_Navigation.Instanzen.IP_Adressen' /* IP_Adressen */, (IP_Adressen.slice(0, IP_Adressen.length - 1)), true, 1000, false);
                // Zusätzlich zum Instanznamen gewünschte Datenpunkte
                // für jede Instanz definieren:
                Datenpunkte_f_C3_BCr_Instanzen = ['Instanz_ID', 'aktuelle_View'];
                // Dafür sorgen, daß unter
                // 0_userdata.0.Vis_Navigation.Instanzen für jede
                // Instanz ein Verzeichnis verfügbar ist.
                var i_end = Liste_Vis_Instanznamen.length;
                var i_inc = 1;
                if (1 > i_end) {
                  i_inc = -i_inc;
                }
                for (i = 1; i_inc >= 0 ? i <= i_end : i >= i_end; i += i_inc) {
                  // Dafür sorgen, daß in den erzeugten Verzeichnissen
                  // jeweils die gewünschten Instanzdatenpunkte
                  // verfügbar sind.
                  for (var j_index in Datenpunkte_f_C3_BCr_Instanzen) {
                    j = Datenpunkte_f_C3_BCr_Instanzen[j_index];
                    // Falls ein Datenpunkt bereits vorhanden ist,
                    // passiert nichts:
                    await Datenpunkt_erzeugen(['0_userdata.0.Vis_Navigation.Instanzen.',Liste_Vis_Instanznamen[(i - 1)],'.',j].join(''), ['{"name":"Vis_Instanz_',Liste_Vis_Instanznamen[(i - 1)],'.',j,'", "type":"string", "read":true, "write":true, "role":"Merker"}'].join(''));
                  }
                }
                if (Debug_Info) {
                  console.info('Datenpunkte "0_userdata.0.Vis_Navigation.Instanzen" sichergestellt.');
                } else {
                  console.debug('Datenpunkte "0_userdata.0.Vis_Navigation.Instanzen" sichergestellt.');
                }
                
                //JTNDeG1sJTIweG1sbnMlM0QlMjJodHRwcyUzQSUyRiUyRmRldmVsb3BlcnMuZ29vZ2xlLmNvbSUyRmJsb2NrbHklMkZ4bWwlMjIlM0UlM0N2YXJpYWJsZXMlM0UlM0N2YXJpYWJsZSUyMGlkJTNEJTIyaXRDdjlJQmIoJTJGJTNBVSolM0QlN0JQMlNqQyUyMiUzRWlkJTNDJTJGdmFyaWFibGUlM0UlM0N2YXJpYWJsZSUyMGlkJTNEJTIycSFZTTEoJTJDOX4pQVclMjRhVmMlNUQlMkZ4dSUyMiUzRWNvbW1vbiUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMmRGaE4lM0FrJTI1ZnJWTVUlMjM0MFklNURhJTNBJTdDJTIyJTNFRGVidWdfSW5mbyUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMlV5dDVkS24lMkIlMjMxaCUzQTJDWSUyNFYlMjVJJTIzJTIyJTNFTGlzdGVfVmlzX0luc3RhbnpuYW1lbiUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMnY5RyU0ME04JTIzQjQlNjB5R2lGcXJNaSU1RSElMjIlM0VMaXN0ZV9JUF9BZHJlc3NlbiUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMk9UN2lOUUpkKGc2JTJGRFc3JTNGTTcpOSUyMiUzRVZpc19JbnN0YW56bmFtZW4lM0MlMkZ2YXJpYWJsZSUzRSUzQ3ZhcmlhYmxlJTIwaWQlM0QlMjJxJTJCWUNNLmMlMkNrOE0lM0FOWDVmKDNpJTI1JTIyJTNFayUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMiU3Qnh+UE1tbiUzQSUyNChRRkolMjVVejJUMiElMjIlM0VJUF9BZHJlc3NlbiUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMmpXOVIlNDB2KVFJd1AlMjQ1c2FTb05JKSUyMiUzRURhdGVucHVua3RlX2YlQzMlQkNyX0luc3RhbnplbiUzQyUyRnZhcmlhYmxlJTNFJTNDdmFyaWFibGUlMjBpZCUzRCUyMnF6SSklNURaJTdEUENUJTdCeDA0Ym1oZyUzRmIlMjIlM0VpJTNDJTJGdmFyaWFibGUlM0UlM0N2YXJpYWJsZSUyMGlkJTNEJTIyKndrOCUzRHV3bCU3QiUyM3VDVkslMkZILTglM0RfJTIyJTNFaiUzQyUyRnZhcmlhYmxlJTNFJTNDJTJGdmFyaWFibGVzJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIycHJvY2VkdXJlc19kZWZjdXN0b21ub3JldHVybiUyMiUyMGlkJTNEJTIyJTJDdXkwJTVCKjh5d0FONFZucyhXRyUzRGElMjIlMjB4JTNEJTIyLTI3NjIlMjIlMjB5JTNEJTIyLTUxMiUyMiUzRSUzQ211dGF0aW9uJTIwc3RhdGVtZW50cyUzRCUyMmZhbHNlJTIyJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmlkJTIyJTIwdmFyaWQlM0QlMjJpdEN2OUlCYiglMkYlM0FVKiUzRCU3QlAyU2pDJTIyJTNFJTNDJTJGYXJnJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmNvbW1vbiUyMiUyMHZhcmlkJTNEJTIycSFZTTEoJTJDOX4pQVclMjRhVmMlNUQlMkZ4dSUyMiUzRSUzQyUyRmFyZyUzRSUzQyUyRm11dGF0aW9uJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyTkFNRSUyMiUzRURhdGVucHVua3RfZXJ6ZXVnZW4lM0MlMkZmaWVsZCUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlNDUklQVCUyMiUzRVkzSmxZWFJsVTNSaGRHVW9hV1FzSUVwVFQwNHVjR0Z5YzJVb1kyOXRiVzl1S1NrN0lBJTNEJTNEJTNDJTJGZmllbGQlM0UlM0Njb21tZW50JTIwcGlubmVkJTNEJTIyZmFsc2UlMjIlMjBoJTNEJTIyODAlMjIlMjB3JTNEJTIyMTYwJTIyJTNFQmVzY2hyZWliZSUyMGRpZXNlJTIwRnVua3Rpb24lMjAlRTIlODAlQTYlM0MlMkZjb21tZW50JTNFJTNDJTJGYmxvY2slM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJ+cFNvQyUyQyU1Qmd2JTNGJTdDJTNCZnVZc1A5cVUlMjIlMjB4JTNEJTIyLTI3NjIlMjIlMjB5JTNEJTIyLTQzNyUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VIaWVyJTIwd2VyZGVuJTIwZiVDMyVCQ3IlMjBkaWUlMjBWaXMtSW5zdGFuemVuJTJDJTIwZGllJTIwdm9tJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyLS5QWGp3JTQwTGRJMiU3QyUyM2goZyU3RCFwLSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0Vpb0Jyb2tlciUyMHNlcGFyYXQlMkZlaW56ZWxuJTIwYW5nZXN0ZXVlcnQlMjB3ZXJkZW4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJmMjFiJTI0cmwlNUIlM0ZsQUMtMl9oOWFFbCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0Vzb2xsZW4lMkMlMjBEYXRlbnB1bmt0ZSUyMGRlZmluaWVydC4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJQKS52a2slM0FReWclMkNINkxFV0IlM0R6NyUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0UlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjI5bCUzRlJfKlpsJTdCa2ltSCk1VEIlNURRWSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VJbml0aWFsaXNpZXJ1bmdlbiUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19zZXQlMjIlMjBpZCUzRCUyMi5KeUxVXyUyNWZmUmcqSEJVKnRNRFUlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMmRGaE4lM0FrJTI1ZnJWTVUlMjM0MFklNURhJTNBJTdDJTIyJTNFRGVidWdfSW5mbyUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyVkFMVUUlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJsb2dpY19ib29sZWFuJTIyJTIwaWQlM0QlMjIlN0NwQkQxRVclMkJ2UFklMjNueVAlM0YlN0RtS0slMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJCT09MJTIyJTNFRkFMU0UlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIySWFDcVZyZzZ0JTJCRGklM0IlN0QlMjNtJTdDWGt+JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMm80QXJQcFhBNSU1RTBNb2FtJTJCSEFkSiUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VEZW4lMjBPcmRuZXIlMjBlcnpldWdlbiUyQyUyMHVudGVyJTIwZGVtJTIwZGllJTIwSW5zdGFuemVuJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIydyUzQXVpd3lkVCUzRkhDU3ZwaXVsNmd3JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRWxpZWdlbiUyMHNvbGxlbiUyMHVuZCUyMDIlMjBEYXRlbnB1bmt0ZSUyMGVpbnRyYWdlbi4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJwcm9jZWR1cmVzX2NhbGxjdXN0b21ub3JldHVybiUyMiUyMGlkJTNEJTIyOWpBKDJiWVVpVmsqeCUyRiU3RHZvLm5HJTIyJTNFJTNDbXV0YXRpb24lMjBuYW1lJTNEJTIyRGF0ZW5wdW5rdF9lcnpldWdlbiUyMiUzRSUzQ2FyZyUyMG5hbWUlM0QlMjJpZCUyMiUzRSUzQyUyRmFyZyUzRSUzQ2FyZyUyMG5hbWUlM0QlMjJjb21tb24lMjIlM0UlM0MlMkZhcmclM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFSRzAlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJGQU0lN0I5cSUzRCU1QiUyQzkhUylhUyU0MGUtNiU0MCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UwX3VzZXJkYXRhLjAuVmlzX05hdmlnYXRpb24uSW5zdGFuemVuLkluc3RhbnpuYW1lbiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFSRzElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJucCUzQiU3RCUzREtkJTVENkFxR3pxJTVFSjlDeDElMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFJTdCJTIybmFtZSUyMiUzQSUyMkluc3RhbnpuYW1lbiUyMiUyQyUyMCUyMnR5cGUlMjIlM0ElMjJzdHJpbmclMjIlMkMlMjAlMjJyZWFkJTIyJTNBdHJ1ZSUyQyUyMCUyMndyaXRlJTIyJTNBdHJ1ZSUyQyUyMCUyMnJvbGUlMjIlM0ElMjJNZXJrZXIlMjIlN0QlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIycHJvY2VkdXJlc19jYWxsY3VzdG9tbm9yZXR1cm4lMjIlMjBpZCUzRCUyMm1wdXclMjRuVSU3QmV3b1p0JTIzNjBoKnMlM0YlMjIlM0UlM0NtdXRhdGlvbiUyMG5hbWUlM0QlMjJEYXRlbnB1bmt0X2VyemV1Z2VuJTIyJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmlkJTIyJTNFJTNDJTJGYXJnJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmNvbW1vbiUyMiUzRSUzQyUyRmFyZyUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQVJHMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMmc0YSUzRjZSeCU1RWVSJTVER3EtNkhVKDMlM0IlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFMF91c2VyZGF0YS4wLlZpc19OYXZpZ2F0aW9uLkluc3Rhbnplbi5JUF9BZHJlc3NlbiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFSRzElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjIuaDA2Qnl6WCU3Q1klNjAlMjMlM0FIJTQwJTVCbCUyNS1qJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSU3QiUyMm5hbWUlMjIlM0ElMjJJUF9BZHJlc3NlbiUyMiUyQyUyMCUyMnR5cGUlMjIlM0ElMjJzdHJpbmclMjIlMkMlMjAlMjJyZWFkJTIyJTNBdHJ1ZSUyQyUyMCUyMndyaXRlJTIyJTNBdHJ1ZSUyQyUyMCUyMnJvbGUlMjIlM0ElMjJNZXJrZXIlMjIlN0QlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyNUJxb3k5JTdEWWs4NWFObUNaNVhhZCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VMaXN0ZSUyMG1pdCUyMGFsbGVuJTIwZ2V3JUMzJUJDbnNjaHRlbiUyRmRlbmtiYXJlbiUyMEluc3RhbnpuYW1lbiUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMm5YSSUzRiUyNG40diUyNFFfIVJpJTQwJTQwN2xGRSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VlcnpldWdlbiUzQSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19zZXQlMjIlMjBpZCUzRCUyMkkucSU1QjJGVlpkfl8yUjVfN1olMkZEJTJDJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJVeXQ1ZEtuJTJCJTIzMWglM0EyQ1klMjRWJTI1SSUyMyUyMiUzRUxpc3RlX1Zpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIybGlzdHNfY3JlYXRlX3dpdGglMjIlMjBpZCUzRCUyMiU1RSU3RHJnSCUyRiUzRiU1RTc2SVk3MHQlM0JUM19JJTIyJTNFJTNDbXV0YXRpb24lMjBpdGVtcyUzRCUyMjMlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDAlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJINllLSCUzRCU3QlkxZTdEOVYlMjRUZWl6JTNEJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRUdlciVDMyVBNHRfMSUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJBSzFlJTdCNVAlMkNGVjR2JTdDVHpSJTJDKFFCJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRUdlciVDMyVBNHRfMiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDIlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJ+JTNBN2dnTS5yVkNkVyU1RWglM0IyTF9oYiUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0VHZXIlQzMlQTR0XzMlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMiEzeHElMkMlNDBRWjQ5YXRfVSU3RCU2MDlmYkclMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFRiVDMyVCQ3IlMjBqZWRlbiUyMEluc3RhbnpuYW1lbiUyMGRpZSUyMElQLUFkcmVzc2UlMjBhbmdlYmVuJTJDJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyKF8lN0NHYSU1RFBpQVI1eCUyRkQ5JTVESkgwJTNEJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRWF1ZiUyMGRlciUyMGRpZSUyMEluc3RhbnolMjBsJUMzJUE0dWZ0LiUyMERhcyUyMHNldHp0JTIwdm9yYXVzJTJDJTIwZGElQzMlOUYlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjIlMjVWayU3RH4lM0ZFeWp+JTJDeCU1RUJfdkUlNURWSSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VkYXMlMjBHZXIlQzMlQTR0JTJDJTIwYXVmJTIwZGVtJTIwZGllJTIwVklTLUluc3RhbnolMjBsJUMzJUE0dWZ0JTJDJTIwaW1tZXIlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJ+UnJQWChwemshUyUyMzdvJTI1RiU3Q1pHJTIzJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRWRpZSUyMGdsZWljaGUlMjBJUC1BZHJlc3NlJTIwaGF0IS4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjIlMjQuJTNBb0YqUkslM0JaY3FEZmZxd3VQJTI0JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRURpZSUyMFp1b3JkbnVuZyUyMHp1JTIwZGVuJTIwSW5zdGFuem5hbWVuJTIwZXJmb2xndCUyMCVDMyVCQ2JlciUyMGRpZSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMjlMRDclM0JrQyU1QnFVV1AlNUJvQjA3NlRqJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRWFuZ2VnZWJlbmUlMjBSZWloZW5mb2xnZSElM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfc2V0JTIyJTIwaWQlM0QlMjJESlV1Y0hXZGVtNENraXRUMSUzQXQ5JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJ2OUclNDBNOCUyM0I0JTYweUdpRnFyTWklNUUhJTIyJTNFTGlzdGVfSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIybGlzdHNfY3JlYXRlX3dpdGglMjIlMjBpZCUzRCUyMkY4U1glMjVjKU5YRVkxdS1ubSlHa3MlMjIlM0UlM0NtdXRhdGlvbiUyMGl0ZW1zJTNEJTIyMyUyMiUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMjVtJTNCJTVFJTIzYlZWRFVQSCU0MDklNjB6dWZRMSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UxOTIuMTY4LjEuMSUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJfLjlZJTdEaCU1RSU3RFpqTXExNSU1Qn5IJTNBNiUyNCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UxOTIuMTY4LjEuMiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDIlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjI1LiUzRiU1QjIlN0N4WiU1REZWQzElM0F5NFpQcG8lMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFMTkyLjE2OC4xLjMlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMmJGS1RKJTJDfldDb3BzIWYlN0N6YSUzRFl1JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRUFsbGUlMjBJbnN0YW56bmFtZW4lMjBtaXQlMjAlMjIlM0IlMjIlMjBnZXRyZW5udCUyMGluJTIwZWluZW4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJQUCU2MFJUTiU3RG8hJTdCJTI1UWM3JTJCdnJaJTNCZSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VEYXRlbnB1bmt0JTIwZGVzJTIwT3JkbmVycyUyMGVpbnRyYWdlbiUzQSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19zZXQlMjIlMjBpZCUzRCUyMiU3QiUyNSlhYVdWJTVCcGpLdEp6KDE0aFolNUUlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMk9UN2lOUUpkKGc2JTJGRFc3JTNGTTcpOSUyMiUzRVZpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dCUyMiUyMGlkJTNEJTIySmlVWCUzQlRuSTZmJTdEJTVEcWVBJTJGMU9LJTNCJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb250cm9sc19mb3JFYWNoJTIyJTIwaWQlM0QlMjIlNUQ2JTIzSSUyQk1RRzllaSU1RCoxJTNEdS0lMkJzJTI0JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJxJTJCWUNNLmMlMkNrOE0lM0FOWDVmKDNpJTI1JTIyJTNFayUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyTElTVCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMjIlMkYxTXp4TFVwJTJDY3FBJTIzbGc2JTdEcSUyNCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIyVXl0NWRLbiUyQiUyMzFoJTNBMkNZJTI0ViUyNUklMjMlMjIlM0VMaXN0ZV9WaXNfSW5zdGFuem5hbWVuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDc3RhdGVtZW50JTIwbmFtZSUzRCUyMkRPJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX3NldCUyMiUyMGlkJTNEJTIyUi1VT1cxLiU0MCU1QnQhaH5WSFY3d2YlNjAlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMk9UN2lOUUpkKGc2JTJGRFc3JTNGTTcpOSUyMiUzRVZpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dF9qb2luJTIyJTIwaWQlM0QlMjJYYyEoaThWUnJyeSU3QyUzQTFOREpGajElMjIlM0UlM0NtdXRhdGlvbiUyMGl0ZW1zJTNEJTIyMyUyMiUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMiUyRlp1OColNUJ+JTNBWnlrRnhHJTdEMjVONkElMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMk9UN2lOUUpkKGc2JTJGRFc3JTNGTTcpOSUyMiUzRVZpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBREQxJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyaHZHJTNBJTI1TmtmYyU2MDd2b3UqJTNBJTdCNCUzRiU1RSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIycSUyQllDTS5jJTJDazhNJTNBTlg1ZigzaSUyNSUyMiUzRWslM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBREQyJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dCUyMiUyMGlkJTNEJTIyJTJGaVJCRnQyY0d5JTVCQnoyMCU1QiU1QklaJTJDJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnN0YXRlbWVudCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJkZWJ1ZyUyMiUyMGlkJTNEJTIyViU3Q3J+dSUyNUlRRjgwcik3T2MlM0JHJTdDaiUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlNldmVyaXR5JTIyJTNFaW5mbyUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJSN1I1ZHRBJTVEKUglNUJwJTIzVkdxUSUyNTFGJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRXRlc3QlM0MlMkZmaWVsZCUzRSUzQyUyRnNoYWRvdyUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMk94ZSl1cWEuOU5HVGUlMkJrdzd5bmslMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMk9UN2lOUUpkKGc2JTJGRFc3JTNGTTcpOSUyMiUzRVZpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyJTJDdmNyUFUxSVNPOCU3Q19qY1lIIVJ2JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRURhcyUyMGxldHp0ZSUyMCUyMiUzQiUyMiUyMHdpZWRlciUyMGVudGZlcm5lbiUzQSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnVwZGF0ZSUyMiUyMGlkJTNEJTIyWHIlMjR4ZSFzayglNUJRKSUzRiElNURPSy0qQSUyMiUyMGlubGluZSUzRCUyMmZhbHNlJTIyJTNFJTNDbXV0YXRpb24lMjB4bWxucyUzRCUyMmh0dHAlM0ElMkYlMkZ3d3cudzMub3JnJTJGMTk5OSUyRnhodG1sJTIyJTIwZGVsYXlfaW5wdXQlM0QlMjJ0cnVlJTIyJTNFJTNDJTJGbXV0YXRpb24lM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJPSUQlMjIlM0UwX3VzZXJkYXRhLjAuVmlzX05hdmlnYXRpb24uSW5zdGFuemVuLkluc3RhbnpuYW1lbiUzQyUyRmZpZWxkJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyV0lUSF9ERUxBWSUyMiUzRVRSVUUlM0MlMkZmaWVsZCUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkRFTEFZX01TJTIyJTNFMTAwMCUzQyUyRmZpZWxkJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVU5JVCUyMiUzRW1zJTNDJTJGZmllbGQlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDTEVBUl9SVU5OSU5HJTIyJTNFRkFMU0UlM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dF9nZXRTdWJzdHJpbmclMjIlMjBpZCUzRCUyMmUoOTNuYll5bSUyQil2TkclNUUlMkIlNUQlNURvRSUyMiUyMGlubGluZSUzRCUyMmZhbHNlJTIyJTNFJTNDbXV0YXRpb24lMjBhdDElM0QlMjJmYWxzZSUyMiUyMGF0MiUzRCUyMnRydWUlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMldIRVJFMSUyMiUzRUZJUlNUJTNDJTJGZmllbGQlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJXSEVSRTIlMjIlM0VGUk9NX1NUQVJUJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJTVFJJTkclMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfZ2V0JTIyJTIwaWQlM0QlMjJIOE1XJTNCbCUyNU5GWGtXUmQlMjRheiU2MEgwJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJPVDdpTlFKZChnNiUyRkRXNyUzRk03KTklMjIlM0VWaXNfSW5zdGFuem5hbWVuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQVQyJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIybWF0aF9hcml0aG1ldGljJTIyJTIwaWQlM0QlMjJOXyUzQSUyQlZGR3NaWSh6dGhxa3olMkMlNUJXJTIyJTIwaW5saW5lJTNEJTIyZmFsc2UlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJPUCUyMiUzRU1JTlVTJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBJTIyJTNFJTNDc2hhZG93JTIwdHlwZSUzRCUyMm1hdGhfbnVtYmVyJTIyJTIwaWQlM0QlMjIlM0RBaS1LeSU2ME80YmtlMFclNDBkJTdCJTI0KS0lMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJOVU0lMjIlM0UwJTNDJTJGZmllbGQlM0UlM0MlMkZzaGFkb3clM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0X2xlbmd0aCUyMiUyMGlkJTNEJTIyan5YTm8yVEFBdG8lMkMlMkNhU3ElNUI0TnMlMjIlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJWQUxVRSUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJoeEpDT2Nzc1IhJTI1ajNkMURYTmw4JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyJTdDciUyMyU0MFoyelgpSUglN0MwWHkqMjAlNDBIJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJPVDdpTlFKZChnNiUyRkRXNyUzRk03KTklMjIlM0VWaXNfSW5zdGFuem5hbWVuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkIlMjIlM0UlM0NzaGFkb3clMjB0eXBlJTNEJTIybWF0aF9udW1iZXIlMjIlMjBpZCUzRCUyMl8weW5oRVJKMiUzQjh6Q041TUwpeGclMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJOVU0lMjIlM0UxJTNDJTJGZmllbGQlM0UlM0MlMkZzaGFkb3clM0UlM0MlMkZ2YWx1ZSUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMmMlM0JWZnQ2JTdDS3Q0SEc3SmUlNUQlM0YoR3clMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFQWxsZSUyMElQLUFkcmVzc2VuJTIwbWl0JTIwJTIyJTNCJTIyJTIwZ2V0cmVubnQlMjBpbiUyMGVpbmVuJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyJTNBJTQwJTJGZ2hVa1NaOX4xT05BcThtJTYwbCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VEYXRlbnB1bmt0JTIwZGVzJTIwT3JkbmVycyUyMGVpbnRyYWdlbiUzQSUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19zZXQlMjIlMjBpZCUzRCUyMlFBV1daWSU3QiUyNWVCJTdDJTQwIW0lMkNLSWolNDBVJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dCUyMiUyMGlkJTNEJTIyKUsucDR6ajhxN1pjVUMlNUJEOCUyNW14JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb250cm9sc19mb3JFYWNoJTIyJTIwaWQlM0QlMjJCdkFEaElKSG1seVolMjRXZiUzRE1UJTJDayUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIycSUyQllDTS5jJTJDazhNJTNBTlg1ZigzaSUyNSUyMiUzRWslM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkxJU1QlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfZ2V0JTIyJTIwaWQlM0QlMjJ1MGglN0JuYTNYODh2UWoyRCElMjNMaTclMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMnY5RyU0ME04JTIzQjQlNjB5R2lGcXJNaSU1RSElMjIlM0VMaXN0ZV9JUF9BZHJlc3NlbiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3N0YXRlbWVudCUyMG5hbWUlM0QlMjJETyUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19zZXQlMjIlMjBpZCUzRCUyMjN6TiUyQmwlMjMuS19LYmMqKFUlMjNFQlhtJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dF9qb2luJTIyJTIwaWQlM0QlMjJkSkM0enBuS2ElNDAlNUJGLSU1QmpXX1ZvOSUyMiUzRSUzQ211dGF0aW9uJTIwaXRlbXMlM0QlMjIzJTIyJTNFJTNDJTJGbXV0YXRpb24lM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBREQwJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIybSUyRiUyQ0dqJTNGfiU3QlhSVyUzRG5FQiU1Qm9SJTNEJTNGJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBREQxJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyTl8lMkZkJTI1aCU3Q2Q4JTNBRzVRLVQxd25qJTI1JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJxJTJCWUNNLmMlMkNrOE0lM0FOWDVmKDNpJTI1JTIyJTNFayUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDIlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJVdSUyNEE2XyUzRGF5czZDRUpZRzUoJTJDeSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UlM0IlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZzdGF0ZW1lbnQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyZGVidWclMjIlMjBpZCUzRCUyMnclN0JaTiUyNENtflklMjN6MFolN0IlMjRISTU3JTdDJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyU2V2ZXJpdHklMjIlM0VpbmZvJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJURVhUJTIyJTNFJTNDc2hhZG93JTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMlI3UjVkdEElNUQpSCU1QnAlMjNWR3FRJTI1MUYlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFdGVzdCUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyMDhMYkZPaCU1RVdaa0dTYVUqSXYpJTJDJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIyaGp5MDUxSyU1QjYwd0VmbkwlMjQlM0QuZy0lMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFRGFzJTIwbGV0enRlJTIwJTIyJTNCJTIyJTIwd2llZGVyJTIwZW50ZmVybmVuJTNBJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydXBkYXRlJTIyJTIwaWQlM0QlMjIqcl9HMUIlMkIxc09KNi1YZjQlN0N5bDclMjIlMjBpbmxpbmUlM0QlMjJmYWxzZSUyMiUzRSUzQ211dGF0aW9uJTIweG1sbnMlM0QlMjJodHRwJTNBJTJGJTJGd3d3LnczLm9yZyUyRjE5OTklMkZ4aHRtbCUyMiUyMGRlbGF5X2lucHV0JTNEJTIydHJ1ZSUyMiUzRSUzQyUyRm11dGF0aW9uJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyT0lEJTIyJTNFMF91c2VyZGF0YS4wLlZpc19OYXZpZ2F0aW9uLkluc3Rhbnplbi5JUF9BZHJlc3NlbiUzQyUyRmZpZWxkJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyV0lUSF9ERUxBWSUyMiUzRVRSVUUlM0MlMkZmaWVsZCUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkRFTEFZX01TJTIyJTNFMTAwMCUzQyUyRmZpZWxkJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVU5JVCUyMiUzRW1zJTNDJTJGZmllbGQlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDTEVBUl9SVU5OSU5HJTIyJTNFRkFMU0UlM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dF9nZXRTdWJzdHJpbmclMjIlMjBpZCUzRCUyMllIZE9JWFB1JTJDSEdRQTBZaHMlMjVJMyUyMiUyMGlubGluZSUzRCUyMmZhbHNlJTIyJTNFJTNDbXV0YXRpb24lMjBhdDElM0QlMjJmYWxzZSUyMiUyMGF0MiUzRCUyMnRydWUlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMldIRVJFMSUyMiUzRUZJUlNUJTNDJTJGZmllbGQlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJXSEVSRTIlMjIlM0VGUk9NX1NUQVJUJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJTVFJJTkclMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfZ2V0JTIyJTIwaWQlM0QlMjJtRGc2dCUyQ2kwcHIlMkNuJTYwczlJSlclM0FnJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBVDIlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJtYXRoX2FyaXRobWV0aWMlMjIlMjBpZCUzRCUyMnYzJTdDLjZwbTBWY2dJUnc0UFUyRiU1RCUyMiUyMGlubGluZSUzRCUyMmZhbHNlJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyT1AlMjIlM0VNSU5VUyUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQSUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJtYXRoX251bWJlciUyMiUyMGlkJTNEJTIyJTNEQWktS3klNjBPNGJrZTBXJTQwZCU3QiUyNCktJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyTlVNJTIyJTNFMCUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dF9sZW5ndGglMjIlMjBpZCUzRCUyMnAwJTNCJTYwRy5venMlM0ZZcypsJTYwNENGM20lMjIlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJWQUxVRSUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJoeEpDT2Nzc1IhJTI1ajNkMURYTmw4JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyMk8lM0RqJTVEZWN2JTNGdlVFei5TWCUyRiUyRlJxJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIlN0J4flBNbW4lM0ElMjQoUUZKJTI1VXoyVDIhJTIyJTNFSVBfQWRyZXNzZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQiUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJtYXRoX251bWJlciUyMiUyMGlkJTNEJTIyeiElNUUlM0FqMSUzRGw2VSU3QlEyZkxjNiU1RDN5JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyTlVNJTIyJTNFMSUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJ3TSUzRCU1RE9HR2YlNUQqJTNGZElmbExEaSU1QmslMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFWnVzJUMzJUE0dHpsaWNoJTIwenVtJTIwSW5zdGFuem5hbWVuJTIwZ2V3JUMzJUJDbnNjaHRlJTIwRGF0ZW5wdW5rdGUlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjIlMkJ2WlkhMGRQM2NuLjBubnRLRVMlN0MlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFZiVDMyVCQ3IlMjBqZWRlJTIwSW5zdGFueiUyMGRlZmluaWVyZW4lM0ElM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfc2V0JTIyJTIwaWQlM0QlMjJmM09sWkZVJTNBc3V1R1F4MVMyJTJCJTYwcCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIyalc5UiU0MHYpUUl3UCUyNDVzYVNvTkkpJTIyJTNFRGF0ZW5wdW5rdGVfZiVDMyVCQ3JfSW5zdGFuemVuJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJWQUxVRSUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmxpc3RzX2NyZWF0ZV93aXRoJTIyJTIwaWQlM0QlMjIlN0MhJTNEJTdEYkR4JTdDJTdEJTIzYSUzRDliN1JOLTYuJTIyJTNFJTNDbXV0YXRpb24lMjBpdGVtcyUzRCUyMjIlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDAlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJQc18lM0FtLSUzQSU1RWdvJTVCZnIlMkNlLlNOWVElMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFSW5zdGFuel9JRCUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJUVnd3ZUd0VkxDTyU2MFdtRjRMSXZSJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRWFrdHVlbGxlX1ZpZXclM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMmJqRCUyRkl+WW5jJTYwJTI1JTNGbHdEdl8lM0QxWCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VEYWYlQzMlQkNyJTIwc29yZ2VuJTJDJTIwZGElQzMlOUYlMjB1bnRlciUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMjVvV2ZnJTNBJTdCJTVFKiU0MFRnS2clM0RtNWd5JTIzJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRTBfdXNlcmRhdGEuMC5WaXNfTmF2aWdhdGlvbi5JbnN0YW56ZW4lMjBmJUMzJUJDciUyMGplZGUlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjJNamclNDBYLW4lM0ZpfnQtTHI4UiU2MHRJaSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0VJbnN0YW56JTIwZWluJTIwVmVyemVpY2huaXMlMjB2ZXJmJUMzJUJDZ2JhciUyMGlzdC4lM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb250cm9sc19mb3IlMjIlMjBpZCUzRCUyMnMlNjBvJTVFSiU3QkElNDBWZyU1QiU2ME9OZE5jeUJRJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJxekkpJTVEWiU3RFBDVCU3QngwNGJtaGclM0ZiJTIyJTNFaSUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyRlJPTSUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJtYXRoX251bWJlciUyMiUyMGlkJTNEJTIyXyU3Q21WNVR6JTdDaXJfMS5CIWU4cllVJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyTlVNJTIyJTNFMSUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJUTyUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJtYXRoX251bWJlciUyMiUyMGlkJTNEJTIySXJhdjglMjNBJTNGUjdOTiolM0ZxWnY3ZzMlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJOVU0lMjIlM0UxMCUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIybGlzdHNfbGVuZ3RoJTIyJTIwaWQlM0QlMjIlNUVXT0haemxNSmNzeCUyM2g0LShKMiUyQyUyMiUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyOUIyIU1RSyUzQW4teFIlNUU0Q2NudCUyM2slMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMlV5dDVkS24lMkIlMjMxaCUzQTJDWSUyNFYlMjVJJTIzJTIyJTNFTGlzdGVfVmlzX0luc3RhbnpuYW1lbiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJCWSUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJtYXRoX251bWJlciUyMiUyMGlkJTNEJTIyJTdEKjEwM1BuSypHVCUyMyUyQjRwNlFRZTQlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJOVU0lMjIlM0UxJTNDJTJGZmllbGQlM0UlM0MlMkZzaGFkb3clM0UlM0MlMkZ2YWx1ZSUzRSUzQ3N0YXRlbWVudCUyMG5hbWUlM0QlMjJETyUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMihIVDFmRHM4KiglMkZaayUyQkUlMkZORlQlMkMlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFRGFmJUMzJUJDciUyMHNvcmdlbiUyQyUyMGRhJUMzJTlGJTIwaW4lMjBkZW4lMjBlcnpldWd0ZW4lMjBWZXJ6ZWljaG5pc3NlbiUzQyUyRmZpZWxkJTNFJTNDbmV4dCUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmNvbW1lbnQlMjIlMjBpZCUzRCUyMiUzRjlqXyktOHlHNShRekY4ZWhyVG0lMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFamV3ZWlscyUyMGRpZSUyMGdldyVDMyVCQ25zY2h0ZW4lMjBJbnN0YW56ZGF0ZW5wdW5rdGUlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjI3em8lM0YhWEElMkYlNUUlM0QlMkIlNjAlMjVac1QtJTJCKiU1RSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMkNPTU1FTlQlMjIlM0V2ZXJmJUMzJUJDZ2JhciUyMHNpbmQuJTNDJTJGZmllbGQlM0UlM0NuZXh0JTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29udHJvbHNfZm9yRWFjaCUyMiUyMGlkJTNEJTIydyUyNXdyRiUyQjZGOVIyd3M1RDZjM19qJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIqd2s4JTNEdXdsJTdCJTIzdUNWSyUyRkgtOCUzRF8lMjIlM0VqJTNDJTJGZmllbGQlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJMSVNUJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyekolN0ROJTJCZCU0MGxlLiUyRiUyNCUyNXglM0RaR1VNdiUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIyalc5UiU0MHYpUUl3UCUyNDVzYVNvTkkpJTIyJTNFRGF0ZW5wdW5rdGVfZiVDMyVCQ3JfSW5zdGFuemVuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDc3RhdGVtZW50JTIwbmFtZSUzRCUyMkRPJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIyY29tbWVudCUyMiUyMGlkJTNEJTIySUFJMndxWWkpJTQwRiU3Qk9EdCU3RCUyQnFVJTQwJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyQ09NTUVOVCUyMiUzRUZhbGxzJTIwZWluJTIwRGF0ZW5wdW5rdCUyMGJlcmVpdHMlMjB2b3JoYW5kZW4lMjBpc3QlMkMlM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb21tZW50JTIyJTIwaWQlM0QlMjIlM0YxNyUzRGglM0RZMXlDJTJDNyU0MHUlMjUlMkJwJTJCZGUlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJDT01NRU5UJTIyJTNFcGFzc2llcnQlMjBuaWNodHMlM0ElM0MlMkZmaWVsZCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJwcm9jZWR1cmVzX2NhbGxjdXN0b21ub3JldHVybiUyMiUyMGlkJTNEJTIyeFV4Rk8lM0RiTkIlN0MlNjBsJTdETlpDJTNBViUzQW8lMjIlM0UlM0NtdXRhdGlvbiUyMG5hbWUlM0QlMjJEYXRlbnB1bmt0X2VyemV1Z2VuJTIyJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmlkJTIyJTNFJTNDJTJGYXJnJTNFJTNDYXJnJTIwbmFtZSUzRCUyMmNvbW1vbiUyMiUzRSUzQyUyRmFyZyUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQVJHMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHRfam9pbiUyMiUyMGlkJTNEJTIyJTdCISU3Qm01amltaWVfNzUlMjMlMkNlNUwlMjUlMkMlMjIlM0UlM0NtdXRhdGlvbiUyMGl0ZW1zJTNEJTIyNCUyMiUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMnBXJTNBeXclNDBLbCU1QlNUbCUyNC4lMkYqSUk3UCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UwX3VzZXJkYXRhLjAuVmlzX05hdmlnYXRpb24uSW5zdGFuemVuLiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJsaXN0c19nZXRJbmRleCUyMiUyMGlkJTNEJTIyJTIzJTI0X2M1TyU1QnFrJTI1Myl1eHUhSSUyQyU3QlclMjIlMjBpbmxpbmUlM0QlMjJmYWxzZSUyMiUzRSUzQ211dGF0aW9uJTIwc3RhdGVtZW50JTNEJTIyZmFsc2UlMjIlMjBhdCUzRCUyMnRydWUlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMk1PREUlMjIlM0VHRVQlM0MlMkZmaWVsZCUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMldIRVJFJTIyJTNFRlJPTV9TVEFSVCUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyVkFMVUUlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ2YXJpYWJsZXNfZ2V0JTIyJTIwaWQlM0QlMjI5QWElMkZqUFI1JTI0N3hfVCU2MFUtRmMlN0J6JTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJVeXQ1ZEtuJTJCJTIzMWglM0EyQ1klMjRWJTI1SSUyMyUyMiUzRUxpc3RlX1Zpc19JbnN0YW56bmFtZW4lM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBVCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMjFlenpMNilIZHMlNjBNLU1FcCU1RV9PeCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIycXpJKSU1RFolN0RQQ1QlN0J4MDRibWhnJTNGYiUyMiUzRWklM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMiUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMkZLMUIhayUzRiUyNGUlN0NrZlIpX0ElNURxZyUzQSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMyUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMlElMkIqS1YlMjV3SmpBUykhKiU2MGFNJTIzKWElMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMip3azglM0R1d2wlN0IlMjN1Q1ZLJTJGSC04JTNEXyUyMiUzRWolM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQVJHMSUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHRfam9pbiUyMiUyMGlkJTNEJTIyOFFsMSU3RFNud0klM0JTbyowa0ZJTGRvJTIyJTNFJTNDbXV0YXRpb24lMjBpdGVtcyUzRCUyMjUlMjIlM0UlM0MlMkZtdXRhdGlvbiUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDAlMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJObFdrQU1oJTdCayUzRCUyQnUlNURfZl8xZyUyRlIlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJURVhUJTIyJTNFJTdCJTIybmFtZSUyMiUzQSUyMlZpc19JbnN0YW56XyUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFERDElMjIlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJsaXN0c19nZXRJbmRleCUyMiUyMGlkJTNEJTIyJTI0VSUzQmZSSHlreCFtaTltbUpCaUlOJTIyJTIwaW5saW5lJTNEJTIyZmFsc2UlMjIlM0UlM0NtdXRhdGlvbiUyMHN0YXRlbWVudCUzRCUyMmZhbHNlJTIyJTIwYXQlM0QlMjJ0cnVlJTIyJTNFJTNDJTJGbXV0YXRpb24lM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJNT0RFJTIyJTNFR0VUJTNDJTJGZmllbGQlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJXSEVSRSUyMiUzRUZST01fU1RBUlQlM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlZBTFVFJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyJTIzaFRTM2MlNjAlN0RSMmRfSyU3QzQlM0QlMjR+T0QlMjIlM0UlM0NmaWVsZCUyMG5hbWUlM0QlMjJWQVIlMjIlMjBpZCUzRCUyMlV5dDVkS24lMkIlMjMxaCUzQTJDWSUyNFYlMjVJJTIzJTIyJTNFTGlzdGVfVmlzX0luc3RhbnpuYW1lbiUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMkFUJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyU01TaWIpdXhFJTI1JTJGVTYlNUUlM0YxJTJGWWZNJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjJxekkpJTVEWiU3RFBDVCU3QngwNGJtaGclM0ZiJTIyJTNFaSUzQyUyRmZpZWxkJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZ2YWx1ZSUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0N2YWx1ZSUyMG5hbWUlM0QlMjJBREQyJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydGV4dCUyMiUyMGlkJTNEJTIyNHpZJTdDMU8lMjN5IWYlNURnWHM2JTNEbVpwRyUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UuJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQUREMyUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnZhcmlhYmxlc19nZXQlMjIlMjBpZCUzRCUyMkxxJTdCUiUzQSU2MDNNZE8lNDAlMjQlM0ZLSFBnVnRPJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVkFSJTIyJTIwaWQlM0QlMjIqd2s4JTNEdXdsJTdCJTIzdUNWSyUyRkgtOCUzRF8lMjIlM0VqJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyQURENCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMnRleHQlMjIlMjBpZCUzRCUyMmdmcVlCdFMzJTNBbjAyMmdTJTNGTDklNjBRJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUyMiUyQyUyMCUyMnR5cGUlMjIlM0ElMjJzdHJpbmclMjIlMkMlMjAlMjJyZWFkJTIyJTNBdHJ1ZSUyQyUyMCUyMndyaXRlJTIyJTNBdHJ1ZSUyQyUyMCUyMnJvbGUlMjIlM0ElMjJNZXJrZXIlMjIlN0QlM0MlMkZmaWVsZCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZuZXh0JTNFJTNDJTJGYmxvY2slM0UlM0MlMkZuZXh0JTNFJTNDJTJGYmxvY2slM0UlM0MlMkZzdGF0ZW1lbnQlM0UlM0MlMkZibG9jayUzRSUzQyUyRm5leHQlM0UlM0MlMkZibG9jayUzRSUzQyUyRm5leHQlM0UlM0MlMkZibG9jayUzRSUzQyUyRm5leHQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnN0YXRlbWVudCUzRSUzQ25leHQlM0UlM0NibG9jayUyMHR5cGUlM0QlMjJjb250cm9sc19pZiUyMiUyMGlkJTNEJTIyUk9zKC1sTSU3RGRRMkVjJTJDVGZzfm14JTIyJTNFJTNDbXV0YXRpb24lMjBlbHNlJTNEJTIyMSUyMiUzRSUzQyUyRm11dGF0aW9uJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIySUYwJTIyJTNFJTNDYmxvY2slMjB0eXBlJTNEJTIydmFyaWFibGVzX2dldCUyMiUyMGlkJTNEJTIyUzklNUUydy5NJTYwSEZJKTAlNjBESkcqJTVEVSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlZBUiUyMiUyMGlkJTNEJTIyZEZoTiUzQWslMjVmclZNVSUyMzQwWSU1RGElM0ElN0MlMjIlM0VEZWJ1Z19JbmZvJTNDJTJGZmllbGQlM0UlM0MlMkZibG9jayUzRSUzQyUyRnZhbHVlJTNFJTNDc3RhdGVtZW50JTIwbmFtZSUzRCUyMkRPMCUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmRlYnVnJTIyJTIwaWQlM0QlMjIhKjVTQWFQJTJCdVFJeXBGcyUzRDclNUUtdSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlNldmVyaXR5JTIyJTNFaW5mbyUzQyUyRmZpZWxkJTNFJTNDdmFsdWUlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRSUzQ3NoYWRvdyUyMHR5cGUlM0QlMjJ0ZXh0JTIyJTIwaWQlM0QlMjJyJTNCJTNBJTNCRDhMZzIpJTVEJTYwaTYlM0RPQ2VabSUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0VEYXRlbnB1bmt0ZSUyMCUyMjBfdXNlcmRhdGEuMC5WaXNfTmF2aWdhdGlvbi5JbnN0YW56ZW4lMjIlMjBzaWNoZXJnZXN0ZWxsdC4lM0MlMkZmaWVsZCUzRSUzQyUyRnNoYWRvdyUzRSUzQyUyRnZhbHVlJTNFJTNDJTJGYmxvY2slM0UlM0MlMkZzdGF0ZW1lbnQlM0UlM0NzdGF0ZW1lbnQlMjBuYW1lJTNEJTIyRUxTRSUyMiUzRSUzQ2Jsb2NrJTIwdHlwZSUzRCUyMmRlYnVnJTIyJTIwaWQlM0QlMjJJWmUyVyUzRmZjbW12SXFDKiUyM3lWJTVEcCUyMiUzRSUzQ2ZpZWxkJTIwbmFtZSUzRCUyMlNldmVyaXR5JTIyJTNFZGVidWclM0MlMkZmaWVsZCUzRSUzQ3ZhbHVlJTIwbmFtZSUzRCUyMlRFWFQlMjIlM0UlM0NzaGFkb3clMjB0eXBlJTNEJTIydGV4dCUyMiUyMGlkJTNEJTIyJTNEcVhNQ0RydGtoJTdDJTVEVyU1RXclM0YpbFNaJTIyJTNFJTNDZmllbGQlMjBuYW1lJTNEJTIyVEVYVCUyMiUzRURhdGVucHVua3RlJTIwJTIyMF91c2VyZGF0YS4wLlZpc19OYXZpZ2F0aW9uLkluc3RhbnplbiUyMiUyMHNpY2hlcmdlc3RlbGx0LiUzQyUyRmZpZWxkJTNFJTNDJTJGc2hhZG93JTNFJTNDJTJGdmFsdWUlM0UlM0MlMkZibG9jayUzRSUzQyUyRnN0YXRlbWVudCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGbmV4dCUzRSUzQyUyRmJsb2NrJTNFJTNDJTJGeG1sJTNF
                


                Dein Skript sieht bei mir jetzt so aus:

                // Funktion, die auf ein Widget wartet.
                // Stammt von: https://forum.iobroker.net/topic/48663/howto-skripte-im-vis-editor-mit-jquery
                // Modifiziert: 60 Durchläufe maximal, 1 Sekunde Pause dazwischen (Warum gerade 60? Reicht das für alle denkbaren Fälle als Wartezeit?)
                function waitForElement(parent, elementPath, wid, widgetName, callBack, counter = 0, debug = false) {
                        if (counter < 60) {
                            setTimeout(function () {
                                if (parent.find(elementPath).length > 0) {
                                    if (debug) console.log(`[${widgetName} ${wid}] it took ${counter}ms to wait for the element '${elementPath}'`);
                                    callBack();
                                } else {
                                    if (debug) console.log(`[${widgetName} ${wid}] wait for element '${elementPath}'`);
                                    counter++
                                    waitForElement(parent, elementPath, wid, widgetName, callBack, counter, debug);
                                }
                            }, 1000);
                        } else {
                           if (debug) console.warn(`[${widgetName} ${wid}] stop waiting after ${counter} retries`);
                            callBack();
                       }
                   }
                
                
                //Selbst angelegte Variable "Vis_Instanzen" und "Vis_IP_Adressen", die jeweils mit einem Semikolon-getrennten-String befüllt werden sollen, der aus den ioBroker-Datenpunkten "0:userdata.0.Vis_Navigation.Instanzen.Instanznamen" und "0:userdata.0.Vis_Navigation.Instanzen.IP_Adressen" gelesen werden muß:
                vis.conn._socket.emit('getState', '0_userdata.0.Vis_Navigation.Instanzen.Instanznamen', function (err, obj) {
                           if (!err && obj) {
                               //Alles OK!
                               Vis_Instanznamen = obj.val;
                console.info("Folgende mögliche Instanzen wurden ermittelt: ",Vis_Instanznamen);
                           } else {
                               //Es ist ein Fehler aufgetreten!
                               alert("Die möglichen VIS-Instanzen konnten nicht ermittelt werden!");
                console.info("Die möglichen VIS-Instanzen konnten nicht ermittelt werden!");
                           }
                       });
                
                vis.conn._socket.emit('getState', '0_userdata.0.Vis_Navigation.Instanzen.IP_Adressen', function (err, obj) {
                           if (!err && obj) {
                               //Alles OK!
                               Vis_IP_Adressen = obj.val;
                console.info("Folgende mögliche IP_Adressen wurden ermittelt: ",Vis_IP_Adressen);
                           } else {
                               //Es ist ein Fehler aufgetreten!
                               alert("Die möglichen IP_Adressen konnten nicht ermittelt werden!");
                console.info("Die möglichen IP_Adressen konnten nicht ermittelt werden!");
                           }
                       });
                
                
                // Warten auf das (basic - Scren Resolution)-Widget:
                //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                waitForElement($('body'),'#w01980', 'dummy', 'dummy', function () {
                  // Widget ist geladen, auf das Widget klicken:
                   setTimeout( () => {
                //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                       $('#w01980>div>span>span').trigger('click');
                       //Die vorhergehende Zeile sorft wohl dafür, daß das aktuelle VIS eine Instanz-ID erhält!?
                   }, 1000);
                   // und in einen Datenpunkt schreiben
                   setTimeout( () => {
                //+++++ Achtung: In der nächsten Zeile die IP-Adresse anpassen!!! +++++
                       $.getJSON('http://192.168.12.72/meineip.php', function(data) {
                
                
                //Aus den Semikolon-getrennten-Strings jeweils eine Liste machen:
                Liste_Instanznamen = Vis_Instanznamen.split(';');
                Liste_IP_Adressen = Vis_IP_Adressen.split(';');
                //Konsolenausgaben für das Austesten des Skripts:
                console.info(Vis_IP_Adressen.length);
                console.info(Liste_IP_Adressen.length);
                //Schleife über alle möglichen/vorgesehenen VIS-Instanzen:
                for (var i_index in Liste_Instanznamen) {
                 i = Liste_Instanznamen[i_index];
                 if (data.ipAddress == Liste_IP_Adressen[i_index]) {
                   vis.setValue((['0_userdata.0.Vis_Navigation.Instanzen.',i,'.Instanz_ID'].join('')), { val: vis.instance, ack: true }, parseInt(((0) || '').toString(), 10), false);
                console.info(i);
                 }
                }
                
                
                       });
                   }, 3000);
                }, 0, true);
                


                In Zeile 23 bis 46 hole ich mir die Namen und IP-Adressen, die das Blockly beim Starten von ioBroker in zwei Datenpunkte geschrieben hat und in Zeile 64 bis 77 werte ich das so aus, daß die jeweilige Vis-Instanz, in der das Skript gerade läuft, ihre Instanz-ID in den passenden ioBroker-Datenpunkt schreibt.

                Soll jetzt z. B. ein weiteres Gerät/VIS an meinen ioBroker geknüpft werden, brauche ich das Vis-Skript nicht mehr anzufassen, sondern ergänze nur noch im Blockly den VIS-/Geräte-Namen und die zugehörige IP-Adresse:

                Unbenannt.PNG

                Nun noch folgende Bemerkung:
                Ich kann zwar zu einem gewissen Grad programmieren, habe aber kaum JS-Kenntnisse. Seht es mir daher bitte nach, wenn der Code vielleicht Macken hat oder "umständlich" ist. Für meine Zwecke funktioniert er. Was nicht (immer) zu klappen scheint, aber offenbar auch nicht stört, sind die Skriptzeilen mit alert- und console.info-Befehlen. Ich vermute hier Timing-Probleme.

                Anregungen/Korrekturen nehme ich jederzeit gern entgegen.

                1 Reply Last reply Reply Quote 1
                • A
                  Andersmacher @BananaJoe last edited by Andersmacher

                  @bananajoe Wenn ich es richtig verstanden habe, weiß der ioBroker also, welche Deiner VIS-Instanzen den Reload angefordert hat, weil jede Deiner Instanzen die Anforderung in "den zur jeweiligen Instanz-ID gehörenden" Datenpunkt Reload-VIS schreibt - richtig?

                  Beim Start einer VIS-Instanz, schreibt diese ja offenbar ihre Instanz-ID nach vis.0.control.instanz. Ist das der einzige Fall, bei dem dieser Datenpunkt nicht durch z. B. ein JS-Skript auf dem Server, sondern von einer vis-Instanz gesetzt wird?

                  Ich hatte gehofft, daß z. B. auch durch ein Widget, das einen Datenpunkt auf dem Server setzt, die Instanz-ID des VIS, in dem sich dieses Widget befindet, nach vis.0.control.instanz geschrieben wird, aber das scheint nicht der Fall zu sein!?

                  A 1 Reply Last reply Reply Quote 0
                  • A
                    Andersmacher @Andersmacher last edited by Andersmacher

                    @BananaJoe Ich glaube, ich muß den Hintergrund meiner Frage noch erklären:
                    So, wie Du Deinen Reload-Button beschreibst, würde er doch in jeder Deiner VIS-Instanzen den gleichen Datenpunkt beschreiben!? Oder wie bringst Du ihm bei, daß er auf jedem Endgerät/vis-Instanz einen anderen Datenpunkt setzt? Hast Du für jedes Endgerät/vis-Instanz ein eigenes Projekt bzw. jeweils immer eine andere View in der dieser Button jeweils immer anders definiert ist?

                    BananaJoe 1 Reply Last reply Reply Quote 0
                    • BananaJoe
                      BananaJoe Most Active @Andersmacher last edited by

                      @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                      Hast Du für jedes Endgerät/vis-Instanz ein eigenes Projekt

                      Jedes Endgerät hat seine eigene View bei mir, ja.
                      Dementsprechend steuert jede View hier seinen eigenen Datenpunkt an der ein Blockly auslöst.
                      Das sucht sich dann die richtige View-Instanz heraus und stößt den reload an.

                      Die eigene Instanz-Id ist ja innerhalb von VIS bekannt - vis.instance
                      Jetzt müsste man schauen wie man diese Info in einem Binding verwenden könnte, bzw. als Information bei einem Widget-Klick mit übergeben könnte.
                      Hatte ich glaube ich schon mal, fällt mir gerade aber nicht ein

                      A 1 Reply Last reply Reply Quote 0
                      • A
                        Andersmacher @BananaJoe last edited by

                        @bananajoe sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                        Jetzt müsste man schauen wie man diese Info in einem Binding verwenden könnte, bzw. als Information bei einem Widget-Klick mit übergeben könnte.

                        Total einfach wäre das, wenn das Special-Binding "instance" funktionieren würde:
                        Unbenannt.PNG
                        Tut es aber leider nicht, siehe auch VIS-Issue #329. Sehr schade, daß das Issue damals unerledigt wieder geschlossen wurde, denn soweit ich das gestern probiert habe, funktioniert es mit der aktuellen VIS nun nicht nur in "basic - HTML"-Widget and as Object ID in "Sichtbarkeit" nicht, sondern in gar keinem Widget mehr.

                        Ich habe es daher einfach mal mit dem Special-Binding "local_*" probiert. Das scheint zu funktionieren. Die gewünschte "local_Variable" mit/in einem VIS-Skript "initialisieren" und dann als Binding in einem VIS-Widget (beim Test ein Basic-String") nutzen:

                        vis.setValue('local_Test', vis.instance);
                        

                        Unbenannt.PNG
                        Ergebnis in der Visu ist meine aktuelle Instanz-ID:
                        Unbenannt.PNG

                        Als "workaround" ist das für mich erst´mal so handhabbar. Aber wenn das Thema für andere auch/doch noch aktuell ist/wird (Das Issue ist ja offenbar wegen Mangel an Interesse/Relevanz geschlossen worden), kann/sollte man vielleicht das Issue noch einmal eröffnen!? Ist ja auch "irgendwie seltsam", daß alle "Special-Bindings" funktionieren, nur "instance" nicht.🤔

                        BananaJoe 1 Reply Last reply Reply Quote 1
                        • BananaJoe
                          BananaJoe Most Active @Andersmacher last edited by

                          @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                          vis.setValue('local_Test', vis.instance);

                          coole idee, funktioniert 100%

                          Ich hab das einfach in mein Script zur Instanzbestimmung eingebaut:
                          8b91ff84-3cb6-4822-8c5e-8b6b22e2dec3-image.png

                          und kann es so z.B. in einem Widget verwenden (hier vis-inventwo - Universal Switch)
                          5d3a61f5-654a-486f-b0f2-d8548714f880-image.png

                          Damit könnte man nun einen zentralen "reload" Button verwenden der dieses schreibt und das Skript das den Datenpunkt auswertet kann entsprechend reagieren. Kann man natürlich auch für alles mögliche andere verwenden.

                          1 Reply Last reply Reply Quote 0
                          • A
                            Andersmacher last edited by

                            @bananajoe sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                            Ich hab das einfach in mein Script zur Instanzbestimmung eingebaut:

                            Da steht es bei mir auch. Schien mir die einfachste Lösung zum Testen und auch irgendwie "sinnig".😀

                            Damit könnte man nun einen zentralen "reload" Button verwenden der dieses schreibt und das Skript das den Datenpunkt auswertet kann entsprechend reagieren. Kann man natürlich auch für alles mögliche andere verwenden.

                            Das war mein Anliegen.😇

                            Freut mich, wenn es Dir/anderen nützt.

                            Wie ich gerade feststellen mußte, nutzt es mir noch nicht so viel, weil ich das Widget (ein "select Value"), für das ich es nutzen wollte, bisher nicht mit konkreten Werten, sondern via Binding "befüllt" habe, um den Inhalt nicht in der VIS, sondern via Datenpunkt pflegen zu können. Nun habe ich das Problem, daß ich quasi ein Binding im Binding bräuchte (was wohl nicht geht), um jedem Auswahlwert die Instanz-Herkunft mitgeben zu können.😕
                            So ginge es:
                            Unbenannt.PNG

                            So leider (noch) nicht:
                            Unbenannt.PNG
                            Da muß ich noch´mal grübeln, ob ich nicht doch in den "sauren Apfel beiße" und wieder auf konkrete Auswahl-Werte zurückgehe.🤔

                            1 Reply Last reply Reply Quote 0
                            • BananaJoe
                              BananaJoe Most Active last edited by BananaJoe

                              Mal als Nachtrag zu meiner Webserver / PHP-Skript Lösung um die eigene IP-Adresse heraus zu bekommen:

                              Es ginge statt eines Apache2 Webservers + PHP nun auch folgendes JavaScript:

                              Prototyp!

                              // Webserver erstellen der die eigene IP zurück gibt, gefunden hier: https://www.digitalocean.com/community/tutorials/how-to-create-a-web-server-in-node-js-with-the-http-module
                              const http = require("http");
                              const host = '0.0.0.0';
                              const port = 16384;
                              
                              // Für ReverseDNS-Abfrage:
                              // Importing the dns module
                              const dns = require('dns');
                              
                              
                              const requestListener = function (req, res) {
                                  res.setHeader('Access-Control-Allow-Origin', '*');
                                  res.setHeader("Content-Type", "application/json");
                                  res.writeHead(200);
                                  dns.reverse(req.socket.remoteAddress, function(err, addresses) {
                                      console.log('Reverse DNS: ' + addresses);
                                      res.end('{"ipAddress":"' + req.socket.remoteAddress + '","reverseDNS":"' + addresses + '"}');
                                  });
                                  //res.end('{"ipAddress":"' + req.socket.remoteAddress + '"}');
                              };
                              
                              var server = http.createServer(requestListener);
                              server.listen(port, host, () => {
                                  console.log(`Server is running on http://${host}:${port}`);
                              });
                              
                              // Socket-Handling, gefunden hier: https://stackoverflow.com/questions/14626636/how-do-i-shut-down-a-node-js-https-server-immediately
                              // Maintain a hash of all connected sockets
                              var sockets = {}, nextSocketId = 0;
                              server.on('connection', function (socket) {
                                // Add a newly connected socket
                                var socketId = nextSocketId++;
                                sockets[socketId] = socket;
                                console.log('socket ' + socketId + ' opened');
                              
                                // Remove the socket when it closes
                                socket.on('close', function () {
                                  console.log('socket ' + socketId + ' closed');
                                  delete sockets[socketId];
                                });
                              
                                // Extend socket lifetime for demo purposes
                                socket.setTimeout(4000);
                              });
                              
                              // Wenn das Skript beendet wird, Webserver auch stoppen und alle noch offenen Sockets beenden
                              // auch von https://stackoverflow.com/questions/14626636/how-do-i-shut-down-a-node-js-https-server-immediately
                              // close Webserver if script stopped
                              onStop(function (callback) {
                                  console.log("Stopping Server ...");
                                  // Close the server
                                  server.close(function () { console.log('Server closed!'); });
                                  // Destroy all open sockets
                                  for (var socketId in sockets) {
                                      console.log('socket ' + socketId + ' destroyed');
                                      sockets[socketId].destroy();
                                  }
                                  callback();
                              }, 10000 /*ms*/);
                              

                              der in diesem Beispiel auf Port 16384 lauschen würde. Es wird die IP-Adresse und der DNS-Name zurück gegeben (ReverseDNS-Abfrage gegen den DNS-Server, im Zweifel kommt da undefined zurück)
                              Beispielausgabe:

                              {
                                  "ipAddress": "192.168.1.58",
                                  "reverseDNS": "TabletWoZi.fritz.box"
                              }
                              

                              Ich muss noch mal etwas darüber nachdenken, würde mir aber gerne eine Version bauen die dann mit 2 Skripten auskommt:

                              • Ein Skript in JavaScript auf dem ioBroker welches den Tablets Ihre IP-Adresse verrät, Datenpunkte anlegt und die Instanze-ID hinterlegt
                              • Jeweils ein JavaScript was man in der VIS hinterlegt - aber bis auf die Widget-Id des basic - Screen Resolution Widget nicht weiter angepasst werden muss.
                              A 1 Reply Last reply Reply Quote 0
                              • A
                                Andersmacher @BananaJoe last edited by

                                @bananajoe sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                (ReverseDNS-Abfrage gegen den DNS-Server, im Zweifel kommt da undefined zurück)

                                Der DNS-Server ist in der Regel (also wenn man keinen extra/speziellen im LAN eingerichtet hat) der Router, z. B. die FritzBox - oder?
                                Wäre dann der Erfolgt der Reverse-Abfrage von der Einstellung in der Fritte abhängig? Ich meine da gab es irgendwo ein Häckchen für "zulassen", wobei ich aber nicht sicher bin, ob das I-Net-mäßig sicher wäre!?

                                @bananajoe sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                Ich muss noch mal etwas darüber nachdenken, würde mir aber gerne eine Version bauen die dann mit 2 Skripten auskommt:

                                Wenn Du das sicher zum Laufen bekommst, fänd ich das super.

                                @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                So leider (noch) nicht:

                                Da muß ich noch´mal grübeln, ob ich nicht doch in den "sauren Apfel beiße" und wieder auf konkrete Auswahl-Werte zurückgehe.

                                Bin momentan auch am Testen einer Lösung gegen "saures Obst". Werde das hier mitteilen, wenn es läuft. Kann aber noch etwas dauern, da momentan im "richtigen Leben" auch noch Aktuelles ansteht.🤗

                                BananaJoe 1 Reply Last reply Reply Quote 0
                                • BananaJoe
                                  BananaJoe Most Active @Andersmacher last edited by

                                  @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                  Der DNS-Server ist in der Regel (also wenn man keinen extra/speziellen im LAN eingerichtet hat) der Router, z. B. die FritzBox - oder?
                                  Wäre dann der Erfolgt der Reverse-Abfrage von der Einstellung in der Fritte abhängig?

                                  Genau, in der Regel sollte das aber bei Standard-Einstellungen funktionieren. Wenn sich das Gerät mit einem Namen beim ersten mal gemeldet hat, dann dieser. Sonst so etwas wie PC-192-168-1-63 oder den Namen den man dem Gerät in der Box gibt.

                                  War jetzt halt meine erste Idee um einen Namen automatisch zu ermitteln ohne den extra hinterlegen zu müssen.

                                  A 1 Reply Last reply Reply Quote 0
                                  • A
                                    Andersmacher @BananaJoe last edited by

                                    @bananajoe Da ich Dein Skript für den "apache-Ersatz" ´mal ausprobieren wollte, fällt mir auf, daß ich als Nicht-Experte noch nicht sicher bin, wie/wo genau Du Dein Skript speicherst und wann/wodurch Du es startest/beendest. Kannst Du dazu bitte noch eine Erläuterung geben?

                                    Update zu meiner "sauren Obst-Vermeidung":
                                    @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                    So leider (noch) nicht:

                                    Ist zwar etwas Aufwand, aber so scheint das dann doch zu klappen (das "simuliert" quasi ein "Binding im Binding" via VIS-Skript-Code):
                                    Anstelle des Bindings

                                    {0_userdata.0.Vis_Navigation.ViewListe.Nummern}
                                    

                                    schreibe ich in mein "Select ValueList-Widget" ins Feld "Werte":

                                    {local_Hauptnavigation_Werte}
                                    

                                    Diese Vis-lokale Variable initialisere ich im VIS-Skript von @BananaJoe so:

                                    ...
                                    //Variable erzeugen, die dann die Nummern des Hauptmenüs als Inhalt bekommt:
                                    vis.conn._socket.emit('getState', '0_userdata.0.Vis_Navigation.ViewListe.Nummern', function (err, obj) {
                                                if (!err && obj) {
                                                    //Alles OK!
                                                    Vis_Hauptmenu_Nummern = obj.val;
                                    console.info("Folgende Vis_Hauptmenu_Nummern wurden ermittelt: ",Vis_Hauptmenu_Nummern);
                                                } else {
                                                    //Es ist ein Fehler aufgetreten!
                                                    alert("Die Vis_Hauptmenu_Nummern konnten nicht ermittelt werden!");
                                    console.info("Die Vis_Hauptmenu_Nummern konnten nicht ermittelt werden!");
                                                }
                                            });
                                    
                                    
                                    // Warten auf das (basic - Scren Resolution)-Widget:
                                    //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                                    waitForElement($('body'),'#w00003', 'dummy', 'dummy', function () {
                                       // Widget ist geladen, auf das Widget klicken:
                                        setTimeout( () => {
                                    //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                                            $('#w00003>div>span>span').trigger('click');
                                            //Die vorhergehende Zeile sorft wohl dafür, daß das aktuelle VIS eine Instanz-ID erhält!?
                                        }, 1000);
                                        // und in einen Datenpunkt schreiben
                                        setTimeout( () => {
                                    //+++++ Achtung: In der nächsten Zeile die IP-Adresse anpassen!!! +++++
                                            $.getJSON('http://192.168.1.99/meineip.php', function(data) {
                                    
                                    //Leider funktioniert das Special-Binding "instance" nicht, sodaß man einen "workaround" braucht, um die VIS-Instanz-ID durch ein Widget an ioBroker übergeben zu können, damit ioBroker individuell auf Aktionen in verschiedenen VIS-Instanzen reagieren kann. Dazu eine lokale Variable innerhalb VIS erzeugen. Diese kann dann via Binding in einem VIS-Widget genutzt werden. Da sie VIS-lokal ist, hat sie trotz einheitlichem Initialisierungscode hier in jeder VIS-Instanz einen anderen/den jeweils passenden Wert:
                                    vis.setValue('local_Instanz_ID', vis.instance);
                                    
                                    //Aus den Semikolon-getrennten-Strings jeweils eine Liste machen:
                                    Liste_Instanznamen = Vis_Instanznamen.split(';');
                                    Liste_IP_Adressen = Vis_IP_Adressen.split(';');
                                    Liste_Vis_Hauptmenu_Nummern = Vis_Hauptmenu_Nummern.split(';');
                                    
                                    //Konsolenausgaben für das Austesten des Skripts:
                                    console.info(Vis_IP_Adressen.length);
                                    console.info(Liste_IP_Adressen.length);
                                    console.info(Vis_Hauptmenu_Nummern);
                                    console.info(Vis_Hauptmenu_Nummern.length);
                                    console.info(Liste_Vis_Hauptmenu_Nummern.length);
                                    
                                    //Schleife über alle möglichen/vorgesehenen VIS-Instanzen:
                                    for (var i_index in Liste_Instanznamen) {
                                      i = Liste_Instanznamen[i_index];
                                      if (data.ipAddress == Liste_IP_Adressen[i_index]) {
                                        vis.setValue((['0_userdata.0.Vis_Navigation.Instanzen.',i,'.Instanz_ID'].join('')), { val: vis.instance, ack: true }, parseInt(((0) || '').toString(), 10), false);
                                    console.info(i);
                                      }
                                    }
                                    
                                    //Schleife über alle Hauptmenu_Nummern, um die VIS-Instanz-ID vor jede Nummer zu schreiben:
                                    Dummy = '';
                                    for (var j_index in Liste_Vis_Hauptmenu_Nummern) {
                                      j = Liste_Vis_Hauptmenu_Nummern[j_index];
                                    Dummy = Dummy + vis.instance + '__' + j + ';';
                                    console.info(Dummy);
                                    }
                                    //Das letzte Semikolon wieder löschen:
                                    Dummy = Dummy.slice(0, Dummy.length-1);
                                    console.info(Dummy);
                                    //VIS-lokale Variable erzeugen und setzen, um damit das Menü-Widget zu befüllen:
                                    vis.setValue('local_Hauptnavigation_Werte', Dummy);
                                    
                                            });
                                        }, 3000);
                                    }, 0, true);
                                    

                                    Nun ergibt das Binding

                                    {local_Hauptnavigation_Werte}
                                    

                                    im Feld "Werte" des "Select ValueList-Widgets":
                                    InstanzID__0;InstanzID__1;InstanzID__2;...
                                    Via Trigger/Blockly kann man die Instanz-ID dann im ioBroker wieder abtrennen und auf den "reinen" Auswahl-Wert (also 0, 1, 2 ...) reagieren:
                                    Unbenannt.PNG

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

                                      @andersmacher sagte in [gelöst] - VIS Intance Id - automatisiert ermitteln:

                                      wie/wo genau Du Dein Skript speicherst und wann/wodurch Du es startest/beendest

                                      es wird als skript im javascript-adapter gespeichert und dort normal gestartet.

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

                                        @oliverio Danke für die Antwort!
                                        Habe das Skript da mal hingetan und gestartet. Danach dann den Apache gestoppt und via Browser http://192.168.1.99:16384 aufgerufen und bekomme dann auch im Browser die erhoffte Rückmeldung mit ip_Address und reverseDNS.
                                        Damit das dann aus VIS-heraus für die Instanz-Bestimmung genutzt werden kann, hätte ich gedacht, daß im folgenden Ausschnitt von @BananaJoe s Skript

                                        // Warten auf das (basic - Scren Resolution)-Widget:
                                        //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                                        waitForElement($('body'),'#w00003', 'dummy', 'dummy', function () {
                                           // Widget ist geladen, auf das Widget klicken:
                                            setTimeout( () => {
                                        //+++++ Achtung: In der nächsten Zeile die Widget-ID anpassen!!! +++++
                                                $('#w00003>div>span>span').trigger('click');
                                                //Die vorhergehende Zeile sorft wohl dafür, daß das aktuelle VIS eine Instanz-ID erhält!?
                                            }, 1000);
                                            // und in einen Datenpunkt schreiben
                                            setTimeout( () => {
                                        //+++++ Achtung: In der nächsten Zeile die IP-Adresse anpassen!!! +++++
                                                $.getJSON('http://192.168.1.99/meineip.php', function(data) {
                                        

                                        dann Zeile 13 geändert werden muß in

                                                $.getJSON('http://0.0.0.0:16384', function(data) {
                                        

                                        aber leider wird dann nichts in meinen Datenpunkt für die Instanz-ID geschrieben und in der Browserkonsole steht:

                                        21:03:25.965 Quellübergreifende (Cross-Origin) Anfrage blockiert: Die Gleiche-Quelle-Regel verbietet das Lesen der externen Ressource auf http://0.0.0.0:16384/. (Grund: CORS-Anfrage schlug fehl). Statuscode: (null).
                                        

                                        "Unter weitere Hinweise" verlinkt die Fehlermeldung auf
                                        Link Text
                                        Einige der dort genannten Punkte (z. B. Add-Blocker oder privates Fenster) könnten auf meinen Browser (FireFox 115.16.1esr (64-Bit)) zutreffen, das blick ich aber nicht. Daher habe ich das mal von der VIS-Android-App aus probiert, aber auch damit wird keine VIS-Instanz in den ioBroker-Datenpunkt eingetragen.

                                        Hättet Ihr da noch Hinweise für mich?

                                        OliverIO BananaJoe 2 Replies Last reply Reply Quote 0
                                        • OliverIO
                                          OliverIO @Andersmacher last edited by

                                          @andersmacher

                                          wenn der iobroker auf dem gleichen rechner gelaufen ist wie zuvor der apache, musst du nix ändern.
                                          ansonsten halt die ip adresse des iobrokers + den port, der im skript definiert wurde. falls iobroker im docker container und nicht im hostmode läuft muss der port bei docker noch freigeschaltet werden.

                                          0.0.0.0 ist aber definitiv falsch. das hat in dem zusammenhang nix zu suchen.
                                          für eine serveranwendung bedeutet das, das sie alle anfragen annimmt.

                                          1 Reply Last reply Reply Quote 0
                                          • BananaJoe
                                            BananaJoe Most Active @Andersmacher last edited by

                                            @andersmacher wie @OliverIO schreibt: Im Skript steht die 0.0.0.0 dafür das an allen Netzwerkschnittstellen gelauscht/geantwortet wird.
                                            In der VIS muss da die IP + Port deines ioBrokers hin (das wird ja vom Tablet & Co aus aufgerufen)

                                            $.getJSON('http://192.168.1.8:16384', function(data) {
                                            

                                            wäre es bei mir, du must deine IP nehmen.
                                            Port 16384 war auch rein willkürlich von mir gewählt (weil bestimmt unbenutzt)

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

                                            Support us

                                            ioBroker
                                            Community Adapters
                                            Donate

                                            937
                                            Online

                                            31.9k
                                            Users

                                            80.1k
                                            Topics

                                            1.3m
                                            Posts

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