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

  1. ioBroker Community Home
  2. Deutsch
  3. Skripten / Logik
  4. JavaScript
  5. Internet Filter in der Fritzbox mit JavaScript setzen

NEWS

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

  • Monatsrückblick – September 2025
    BluefoxB
    Bluefox
    13
    1
    1.9k

  • Neues Video "KI im Smart Home" - ioBroker plus n8n
    BluefoxB
    Bluefox
    15
    1
    2.3k

Internet Filter in der Fritzbox mit JavaScript setzen

Geplant Angeheftet Gesperrt Verschoben JavaScript
fritzboxkindersicherungfilter
145 Beiträge 24 Kommentatoren 25.6k Aufrufe 36 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.
  • afuerhoffA afuerhoff

    @ilovegym @moelski
    Ich schau mir das mal an . Würde dann ggfs Teile des Scripts verwenden. Kann allerdings noch etwas dauern, da ich erst noch das Mesh Feature implementieren möchte.
    EDIT: Script läuft bei mir ebenfalls ohne Probleme auf einer 7490.

    moelskiM Offline
    moelskiM Offline
    moelski
    schrieb am zuletzt editiert von
    #35

    Moin !

    Ich habe mal Version 1.03 im 7ten Posting eingefügt.

    Changelog:

    • 02.09.20 V1.03 Erweiterung zum automatischen Start mittels ioB Datenpunkt
    • 02.09.20 V1.02 Script Abbruch wenn der Login nicht geklappt hat
    • 02.09.20 V1.01 Regex Erweiterung, Login auch mit User/Passwort
    • 01.09.20 V1.00 erste öffentliches Release

    Also damit sollte der Login mit Username klappen und man kann das Script auch über einen Datenpunkt im ioB anschubsen. Dazu muss man nur einen Datenpunkt anlegen in den man dann "Comp-L-Pi;filtprof1" als Beispiel schreibt. Also <Computername>;<Profil>.

    Infos wie man das einrichtet habe ich ins Script gepackt :-)

    Als nächstes würde ich dann noch einen Umbau vornehmen so das man die Profilnamen und nicht die Profil ID verwenden kann.

    Grüße

    Grüße Dominik

    1 Antwort Letzte Antwort
    3
    • moelskiM moelski

      Moin !

      Gestern Abend doch glatt am TV verschlummert :face_with_rolling_eyes: Aber nun ...

      Hier aber nun das Script:

      /*******************************************************
      * Basis: 
      * https://community.openhab.org/t/disable-internet-connection-of-specific-echo-device-on-fritz-box-via-script/91374
      * https://forum.fhem.de/index.php?topic=109689.0
      * https://www.kuketz-blog.de/fritzbox-wlan-toggle-per-skript/
      *******************************************************/ 
      
      /*******************************************************
      * Script SetDeviceFilter
      * 
      * 13.09.20 V1.09    Regex mit Escape beim . und Fix für Liste
      * 12.09.20 V1.08    Regex um . erweitert
      * 12.09.20 V1.07    Erweritertes Logging aktivierbar für Fehlersuche
      * 08.09.20 V1.06    Anpassungen an FirtzOS 7.20 (Regex Auswertung für Listenerstellung)
      * 05.09.20 V1.05    Ausgabe einer Device / Filter Liste möglich als JSON 
      *                   Logging reduziert wenn FbListOnly = false
      * 04.09.20 V1.04    Fix für doppelte Einträge in Listen
      * 02.09.20 V1.03    Erweiterung zum automatischen Start mittels ioB Datenpunkt
      * 02.09.20 V1.02    Script Abbruch wenn der Login nicht geklappt hat
      * 02.09.20 V1.01    Regex Erweiterung, Login auch mit User/Passwort
      * 01.09.20 V1.00    erste öffentliches Release
      *******************************************************/ 
      const Version = "1.09";
      
      // TODO
      // * Liste erzeugen mit Devices und aktuellem Filter
      //   > als JSON in einen DP speichern -> für Tabellenauswertung
      
      // https://forum.iobroker.net/topic/16184/http-request-in-javascript/5
      var request = require('request');
      var headers = { 
          'Content-Type': 'application/x-www-form-urlencoded', 
          'User-Agent': 'curl/7.64.0', 
          'Accept': '*/*'
      }; 
      
      /*******************************************************
      * E I N S T E L L U N G E N 
      *******************************************************/
      
      // Die IP Der Fritzbox
      const FbIp          = "192.168.30.1";              
      // Der User der Fritzbox 
      // HINWEIS : Den Benutzer leer lassen wenn die Anmeldung an der FB nur mit Passwort erfolgt !
      const FbUser        = ""; 
      // Das Password der Fritzbox                  
      const FbPassword    = "";   
      // welcher Rechner soll "bearbeitet" werden                
      var   FbDevice      = "Comp-L-Pi";   
      // Das neue Profil für den Rechner              
      var   FbProfile     = "filtprof4";  
      // erweitertes Logging aktivieren für Fehlersuche
      const FbDebugging   = false;
      // Keine Änderung an der Fritzbox (true) -> Listet dann nur alle Profile und Rechner               
      const FbListOnly    = false; 
      // Device / Filter Liste erzeugen Ja (true), Nein (false)
      const FbCreateList  = true; 
      // Datenpunkt (string) für einen automatischen Script Start (muss angelegt werden!)
      // Der Datenpunkt wir mit Device;Profil beschrieben. Bsp: Comp-L-Pi;filtprof1
      const FbIobObject   = "Global.0.Fritzbox.ChangeDeviceFilter"  
      // Datenpunkt in den die Device / Filter Liste als JSON geschrieben wird                    
      const FbIobJsonList = "Global.0.Fritzbox.DeviceFilterListJson" 
      
      var secChallenge;
      var secMd5;
      var secLogin;
      var secSid;
      var secProfileNames = [];
      var secProfileIds   = [];
      var secDeviceNames  = [];
      var secDeviceIds    = [];
      var JsonList        = "";
      
      /*******************************************************
      * Object Trigger Version
      * Wenn auf einen Datenpunkt getriggert werden soll, dann muss der folgende Block auskommentiert werden. 
      * FbIobObject muss mit einem string Datenpunkt versehen werden.
      * Den Static Teil dann auskommentieren !
      * Diese Variante erlaubt es das Script dynamisch für alle Devices / Profile der FB zu verwenden. 
      *******************************************************/
      on({id: FbIobObject, change: "ne"}, function (obj) {
          console.log("Profile Changer started ... (Version : " + Version + ")");
          var data = obj.state.val.split(";");
          if (data.length != 2) {
              console.log("Wrong parameter : " + obj.state.val);
              return;
          }
      
          if (FbDebugging){
              console.log(" > Datapoint : " + data);
          }
      
          console.log("Computer : " + data[0]);
          console.log("Filter   : " + data[1]);
          FbDevice  = data[0];
          FbProfile = data[1];
      
          getFbChallenge();
          
          console.log("Profile Changer done");
      });
      
      /*******************************************************
      * Static Version
      * Die folgenden 3 Zeilen lassen das Script  sofort laufen. 
      * Dabei sollte der Object Trigger Block auskommentiert werden. 
      *******************************************************/
      // console.log("Profile Changer started ... (Version : " + Version + ")");
      // getFbChallenge();
      // console.log("Profile Changer done");
      
      // Get the Challenge String from the FritzBox
      // Compute the md5 Hash
      function getFbChallenge(){
          console.log("function getFbChallenge");
          request.get({
              url:        'http://' + FbIp + '/login_sid.lua?username=' + FbUser,
              headers:    headers
          }, function(error, response, body) {
              if (error) log(error, 'error');
              //console.log(response);
              // https://developer.mozilla.org/de/docs/Web/JavaScript/Reference/Global_Objects/RegExp
              // <?xml version="1.0" encoding="utf-8"?><SessionInfo><SID>0000000000000000</SID><Challenge>d36de231</Challenge><BlockTime>0</BlockTime><Rights></Rights></SessionI
              secChallenge = body.match(/Challenge>(.*)<\/Challenge/)[1];
              console.log(" > Challenge : " + secChallenge);
      
              var uft16le = str2rstr_utf16le(secChallenge + "-" +FbPassword );
              var md5     = rstr_md5(uft16le);
              secMd5      = rstr2hex(md5);
                      
              console.log(" > MD5       : " + secMd5);   
      
              // response="${challenge}-${md5}"
              // sid=$(curl -i -s -k -d "response=${response}&username=" "http://$1" | grep -Po -m 1 '(?<=sid=)[a-f\d]+' | tail -1)
              secLogin = "response=" + secChallenge + "-" + secMd5 + "&username=" + FbUser;
      
              console.log(" > Login     : " + secLogin);
      
              getFbSid();
          });
      }
      
      // Get the SID from the Fritzbox
      function getFbSid() {
          console.log("function getFbSid");
      
          request.post({
              url:        'http://' + FbIp,
              headers:    headers, 
              form:       secLogin
          }, function(error, response, body) {
              if (error) log(error, 'error');
              // SID filtern 
              // "sid":"c503b24dae458086"
              try {
                  secSid = response.body.match(/\"sid\":\"(.*)\"/)[1];
              }
              catch (e) {
                  if (secSid == undefined) {
                      console.log("Your login was not successful. End Script", Error)
                      return;
                  }
              }
      
              console.log(" > SID       : " + secSid);
      
              //getFbDeviceInfos();
              getFbProfiles();
          });
      } 
      
      // TBD: 
      // Netzwerkinfos mit allen Devices lesen: 
      // var req = "xhr=1&sid=" + secSid + "&lang=de&page=netDev&xhrId=cleanup&useajax=1&no_sidrenew=" 
      function getFbDeviceInfos(){
          console.log("function getFbDeviceInfos");
          var req = "xhr=1&sid=" + secSid + "&lang=de&page=netDev&xhrId=cleanup&useajax=1&no_sidrenew=" 
      
          request.post({
              url:        'http://' + FbIp + '/data.lua',
              headers:    headers, 
              form:       req
          }, function(error, response, body) {
              if (error) log(error, 'error');
              console.log(response.body);
          });    
      }
      
      // Get all Profiles from the Fritzbox
      function getFbProfiles() {
          console.log("function getFbProfiles");
          // #curl -d "xhr=1&sid=${sid}&lang=de&no_sidrenew=&page=kidPro" "http://$1/data.lua"
          var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidPro";
      
          request.post({
              url:        'http://' + FbIp + '/data.lua',
              headers:    headers, 
              form:       req
          }, function(error, response, body) {
              if (error) log(error, 'error');
              
              if (FbDebugging){
                  console.log(" > response.body : \n" + response.body);
              }
      
              secProfileIds   = [];
              secProfileNames = [];
      
              console.log(" > Decode Names")
              var rx = new RegExp( /class=\"name\"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g );
              while( (match = rx.exec( body )) != null ) {
                  secProfileNames.push(match[1]);
              }
              if (FbDebugging){
                  console.log(" > secProfileNames : \n" + secProfileNames);
              }
      
              console.log(" > Decode Filters")
              // submit" name="edit" value="
              rx = new RegExp( /submit\"\sname=\"edit\"\svalue=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sclass=\"icon/g );
              while( (match = rx.exec( body )) != null ) {
                  secProfileIds.push(match[1]);
              }
              if (FbDebugging){
                  console.log(" > secProfileIds : \n" + secProfileIds);
              }
      
              if (FbListOnly) {
                  console.log("Filter Count : " + secProfileIds.length);
                  for (var i = 0; i < secProfileIds.length; i++) {
                      console.log("Filter named '"+ secProfileNames[i] + "' has ID : " + secProfileIds[i]);
                  }
              }
      
              getFbDevices();
          });
      }
      
      // IMPORTANT: 
      // The DeviceId changes when you switch the profile. 
      // If you use default profiles it is something like "landevice308962"
      // If you use your own profiles it is something like "user7749"
      function getFbDevices() {
          console.log("function getFbDevices");
          // xhr=1&sid=5f5ba302815594d8&lang=de&no_sidrenew=&page=kidLis
          var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidLis";
      
          request.post({
              url:        'http://' + FbIp + '/data.lua',
              headers:    headers, 
              form:       req
          }, function(error, response, body) {
              if (error) log(error, 'error');
              if (FbDebugging){
                  console.log(" > response.body : \n" + response.body);
              }
      
              // TESTING
              //body = getState("Global.0.Testing.StringValue").val;
      
              secDeviceNames = [];
              secDeviceIds   = [];
      
              console.log(" > Decode Device Names")
              var rx = new RegExp( /class=\"name"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g );
              while( (match = rx.exec( body )) != null ) {
                  secDeviceNames.push(match[1]);
              }
              if (FbDebugging){
                  console.log(" > secDeviceNames : \n" + secDeviceNames);
              }
      
              console.log(" > Decode Device Ids")
              rx = new RegExp( /name=\"profile:([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"><option/g );
              while( (match = rx.exec( body )) != null ) {
                  secDeviceIds.push(match[1]);
              }
              if (FbDebugging){
                  console.log(" > secDeviceIds : \n" + secDeviceIds);
              }
              
              if (FbListOnly) {
                  console.log("Device Count : " + secDeviceIds.length);
                  for (var i = 0; i < secDeviceNames.length; i++) {
                      console.log(i + " - Device named '"+ secDeviceNames[i] + "' has ID : " + secDeviceIds[i]);
                  }
              }
      
              if (!FbListOnly) { 
                  setFbSperre(); 
              } else { 
                  console.log("DONE : Listmode - No Device blocking ..."); 
              }
          });
      }
      
      //curl -d "sid=${sid}&edit=$3&time=$4&budget=unlimited&apply=&page=kids_profileedit" "http://$1/data.lua" >/dev/null 2>&1
      function setFbSperre() {
          console.log("function setFbSperre");
          
          // Device suchen ... 
          var devId = -1;
          for (var i = 0; i < secDeviceNames.length; i++) {
              if (secDeviceNames[i] == FbDevice) {
                  devId = i;
              }
          }
          
          console.log(" > Device Nr   : " + devId); 
          console.log(" > Device ID   : " + secDeviceIds[devId]); 
          console.log(" > Device Name : " + secDeviceNames[devId]);
          //var profile = "filtprof4";
          //var profile = "filtprof3079";
      
          // xhr=1&sid=b46f3d48b3b0fcc2&lang=de&no_sidrenew=&profile%3Alandevice308962=filtprof4&apply=&oldpage=%2Finternet%2Fkids_userlist.lua
          var req2 = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&profile%3A" + secDeviceIds[devId] + "=" + FbProfile + "&apply=&oldpage=%2Finternet%2Fkids_userlist.lua";
          console.log(" > req : " + req2);
          //var req2 = "xhr=1&sid=" + sid + "&lang=de&no_sidrenew=&profile%3Alandevice308962=filtprof3079&profile%3Auser7725=filtprof4&profile%3Alandevice2671=filtprof1&profile%3Alandevice2444=filtprof1&profile%3Alandevice2330=filtprof1&profile%3Alandevice2195=filtprof1&profile%3Alandevice229903=filtprof1&profile%3Alandevice8486=filtprof1&profile%3Alandevice1908=filtprof1&profile%3Alandevice1865=filtprof1&profile%3Alandevice516865=filtprof1&profile%3Alandevice518551=filtprof1&profile%3Alandevice169594=filtprof1&profile%3Alandevice171398=filtprof1&profile%3Alandevice175510=filtprof1&profile%3Alandevice177635=filtprof1&profile%3Alandevice178875=filtprof1&profile%3Alandevice179702=filtprof1&profile%3Alandevice180995=filtprof1&profile%3Alandevice182196=filtprof1&profile%3Alandevice183607=filtprof1&profile%3Alandevice184405=filtprof1&profile%3Alandevice638220=filtprof1&profile%3Alandevice634583=filtprof1&profile%3Alandevice7650=filtprof1&profile%3Alandevice353720=filtprof1&profile%3Auser7731=filtprof4&profile%3Alandevice170191=filtprof1&profile%3Alandevice941076=filtprof1&profile%3Alandevice358722=filtprof1&profile%3Alandevice2859=filtprof1&profile%3Auser7739=filtprof4&profile%3Alandevice8035=filtprof1&profile%3Auser7733=filtprof4&profile%3Auser7740=filtprof4&profile%3Alandevice1333=filtprof1&profile%3Alandevice6211=filtprof1&profile%3Alandevice35977=filtprof1&profile%3Alandevice8101=filtprof1&profile%3Alandevice8174=filtprof1&profile%3Auser7729=filtprof4&profile%3Alandevice1125=filtprof1&profile%3Alandevice1124=filtprof1&profile%3Alandevice7715=filtprof1&profile%3Alandevice52361=filtprof1&profile%3Alandevice30084=filtprof1&profile%3Alandevice26151=filtprof1&profile%3Alandevice26138=filtprof1&profile%3Alandevice6162=filtprof1&profile%3Alandevice5204=filtprof1&profile%3Alandevice1129367=filtprof1&profile%3Alandevice9399=filtprof1&profile%3Alandevice232149=filtprof1&profile%3Alandevice6935=filtprof1&profile%3Alandevice941333=filtprof1&profile%3Alandevice941569=filtprof1&apply=&oldpage=%2Finternet%2Fkids_userlist.lua";
          //var req2 = "xhr=1&sid=" + sid + "&lang=de&no_sidrenew=&profile:landevice308962=filtprof4&apply=&oldpage=/internet/kids_userlist.lua";
      
          request.post({
              url:        'http://' + FbIp + '/data.lua',
              headers:    headers, 
              form:       req2
          }, function(error, response, body) {
              if (error) log(error, 'error');
              if (FbDebugging){
                  console.log(" > response : \n" + response);
              }
      
              console.log("Change done. Check your FB / Device if it worked :-)");
      
              if (FbCreateList) {
                  getFbDeviceFilterList();
              }
          });    
      }
      
      function getFbDeviceFilterList() {
          console.log("function getFbDeviceFilterList");
          
          // xhr=1&sid=e2d99419592fb780&lang=de&no_sidrenew=&page=kidLis
          var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidLis";
      
          request.post({
              url:        'http://' + FbIp + '/data.lua',
              headers:    headers, 
              form:       req
          }, function(error, response, body) {
              //if (error) log(error, 'error');
              //console.log(response.body);
      
              JsonList = "";
              var tmpDevices      = [];
              var tmpUsage        = [];
              var tmpProfilesId   = [];
              var tmpProfilesName = [];
              var tmpTimes        = [];
      
              // TESTING
              //body = getState("Global.0.Testing.StringValue").val;
              
              console.log(" > Decode Device Names")
              var rx = new RegExp(/class=\"name"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g);
              while ((match = rx.exec(body)) != null) {
                  tmpDevices.push(match[1]);
              }
              //console.log(" > " + tmpDevices);
      
              console.log(" > Decode Device Profile ID");
              //rx = new RegExp(/button\s.*?value=\"([a-zA-Z0-9 äöüÄÖÜ]*?)\"/g);
              rx = new RegExp(/option\svalue=\"([a-zA-Z0-9 äöüÄÖÜ\-]*?)\"\sselected/g);
              while ((match = rx.exec(body)) != null) {
                  tmpProfilesId.push(match[1]);
              }
              //console.log(" > " + tmpProfilesId);
      
              console.log(" > Decode Device Profile Names");
              for (var i = 0; i < tmpProfilesId.length; i++) {
                  for (var j = 0; j < secProfileIds.length; j++) { 
                      //console.log(tmpProfilesId[i] + " > " + secProfileIds[j]);
                      if (tmpProfilesId[i] == secProfileIds[j]){
                          tmpProfilesName.push(secProfileNames[j]);
                      }
                  }
              }
              //console.log(" > " + tmpProfilesName);
      
              console.log(" > Decode Internetnutzung")
              rx = new RegExp(/td\sdatalabel=\"[a-zA-Z0-9 äöüÄÖÜ\-]+\"\sclass=\"usage\">(.*?)<\/td>/g);
              while ((match = rx.exec(body)) != null) { 
                  var tmp = match[1];
                  tmp = tmp.replace("<span>", "").replace("<\/span>", "");
                  tmpUsage.push(tmp);
              }
              //console.log(" > " + tmpUsage);
      
              console.log(" > Decode Times")
              rx = new RegExp(/td\sdatalabel=\"[a-zA-Z0-9 äöüÄÖÜ\-]+\"\sclass=\"bar\stime\">(.*?)<\/td>/g);
              while ((match = rx.exec(body)) != null) {
                  var tmp = match[1];
                  if (tmp.includes("\"")) {
                      tmp = tmp.replace("<span title=\"", "");
                      tmp = tmp.slice(0, tmp.indexOf("\""));
                      tmpTimes.push(tmp);
                  } else {
                      tmpTimes.push(tmp);
                  }
              }
              //console.log(" > " + tmpTimes);
      
              // console.log("Device Count : " + secDeviceIds.length);
              JsonList = "[";
              for (var i = 0; i < tmpDevices.length; i++) {
                  JsonList += "{";
                  JsonList += "\"Device\": \"" + tmpDevices[i] + "\"," 
                  JsonList += "\"Profile ID\": \"" + tmpProfilesId[i] + "\","
                  JsonList += "\"Profile Name\": \"" + tmpProfilesName[i] + "\","
                  JsonList += "\"Usage\": \"" + tmpUsage[i] + "\","
                  JsonList += "\"Time\": \"" + tmpTimes[i] + "\""
                  if (i != tmpDevices.length - 1) { JsonList += "},"; } else { JsonList += "}"; }
              }
              JsonList += "]";
      
              //console.log(JsonList);
              setState(FbIobJsonList,  JsonList, true);
      
              console.log("DONE : List generated ...");
          });   
      }
      
      
      // https://gist.githubusercontent.com/josedaniel/951664/raw/33c7a6b44d6cc53dc75cb5230fb551bc4ba7a46a/md5.js
      // http://pajhome.org.uk/crypt/md5/instructions.html
      
      /*
       * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
       * Digest Algorithm, as defined in RFC 1321.
       * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
       * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
       * Distributed under the BSD License
       * See http://pajhome.org.uk/crypt/md5 for more info.
       */
      
      /*
       * Configurable variables. You may need to tweak these to be compatible with
       * the server-side, but the defaults work in most cases.
       */
      var hexcase = 0;   /* hex output format. 0 - lowercase; 1 - uppercase        */
      var b64pad  = "";  /* base-64 pad character. "=" for strict RFC compliance   */
      
      /*
       * These are the functions you'll usually want to call
       * They take string arguments and return either hex or base-64 encoded strings
       */
      function hex_md5(s)    { return rstr2hex(rstr_md5(str2rstr_utf8(s))); }
      function b64_md5(s)    { return rstr2b64(rstr_md5(str2rstr_utf8(s))); }
      function any_md5(s, e) { return rstr2any(rstr_md5(str2rstr_utf8(s)), e); }
      function hex_hmac_md5(k, d)
        { return rstr2hex(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
      function b64_hmac_md5(k, d)
        { return rstr2b64(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
      function any_hmac_md5(k, d, e)
        { return rstr2any(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)), e); }
      
      /*
       * Perform a simple self-test to see if the VM is working
       */
      function md5_vm_test()
      {
        return hex_md5("abc").toLowerCase() == "900150983cd24fb0d6963f7d28e17f72";
      }
      
      /*
       * Calculate the MD5 of a raw string
       */
      function rstr_md5(s)
      {
        return binl2rstr(binl_md5(rstr2binl(s), s.length * 8));
      }
      
      /*
       * Calculate the HMAC-MD5, of a key and some data (raw strings)
       */
      function rstr_hmac_md5(key, data)
      {
        var bkey = rstr2binl(key);
        if(bkey.length > 16) bkey = binl_md5(bkey, key.length * 8);
      
        var ipad = Array(16), opad = Array(16);
        for(var i = 0; i < 16; i++)
        {
          ipad[i] = bkey[i] ^ 0x36363636;
          opad[i] = bkey[i] ^ 0x5C5C5C5C;
        }
      
        var hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
        return binl2rstr(binl_md5(opad.concat(hash), 512 + 128));
      }
      
      /*
       * Convert a raw string to a hex string
       */
      function rstr2hex(input)
      {
        try { hexcase } catch(e) { hexcase=0; }
        var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef";
        var output = "";
        var x;
        for(var i = 0; i < input.length; i++)
        {
          x = input.charCodeAt(i);
          output += hex_tab.charAt((x >>> 4) & 0x0F)
                 +  hex_tab.charAt( x        & 0x0F);
        }
        return output;
      }
      
      /*
       * Convert a raw string to a base-64 string
       */
      function rstr2b64(input)
      {
        try { b64pad } catch(e) { b64pad=''; }
        var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
        var output = "";
        var len = input.length;
        for(var i = 0; i < len; i += 3)
        {
          var triplet = (input.charCodeAt(i) << 16)
                      | (i + 1 < len ? input.charCodeAt(i+1) << 8 : 0)
                      | (i + 2 < len ? input.charCodeAt(i+2)      : 0);
          for(var j = 0; j < 4; j++)
          {
            if(i * 8 + j * 6 > input.length * 8) output += b64pad;
            else output += tab.charAt((triplet >>> 6*(3-j)) & 0x3F);
          }
        }
        return output;
      }
      
      /*
       * Convert a raw string to an arbitrary string encoding
       */
      function rstr2any(input, encoding)
      {
        var divisor = encoding.length;
        var i, j, q, x, quotient;
      
        /* Convert to an array of 16-bit big-endian values, forming the dividend */
        var dividend = Array(Math.ceil(input.length / 2));
        for(i = 0; i < dividend.length; i++)
        {
          dividend[i] = (input.charCodeAt(i * 2) << 8) | input.charCodeAt(i * 2 + 1);
        }
      
        /*
         * Repeatedly perform a long division. The binary array forms the dividend,
         * the length of the encoding is the divisor. Once computed, the quotient
         * forms the dividend for the next step. All remainders are stored for later
         * use.
         */
        var full_length = Math.ceil(input.length * 8 /
                                          (Math.log(encoding.length) / Math.log(2)));
        var remainders = Array(full_length);
        for(j = 0; j < full_length; j++)
        {
          quotient = Array();
          x = 0;
          for(i = 0; i < dividend.length; i++)
          {
            x = (x << 16) + dividend[i];
            q = Math.floor(x / divisor);
            x -= q * divisor;
            if(quotient.length > 0 || q > 0)
              quotient[quotient.length] = q;
          }
          remainders[j] = x;
          dividend = quotient;
        }
      
        /* Convert the remainders to the output string */
        var output = "";
        for(i = remainders.length - 1; i >= 0; i--)
          output += encoding.charAt(remainders[i]);
      
        return output;
      }
      
      /*
       * Encode a string as utf-8.
       * For efficiency, this assumes the input is valid utf-16.
       */
      function str2rstr_utf8(input)
      {
        var output = "";
        var i = -1;
        var x, y;
      
        while(++i < input.length)
        {
          /* Decode utf-16 surrogate pairs */
          x = input.charCodeAt(i);
          y = i + 1 < input.length ? input.charCodeAt(i + 1) : 0;
          if(0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF)
          {
            x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
            i++;
          }
      
          /* Encode output as utf-8 */
          if(x <= 0x7F)
            output += String.fromCharCode(x);
          else if(x <= 0x7FF)
            output += String.fromCharCode(0xC0 | ((x >>> 6 ) & 0x1F),
                                          0x80 | ( x         & 0x3F));
          else if(x <= 0xFFFF)
            output += String.fromCharCode(0xE0 | ((x >>> 12) & 0x0F),
                                          0x80 | ((x >>> 6 ) & 0x3F),
                                          0x80 | ( x         & 0x3F));
          else if(x <= 0x1FFFFF)
            output += String.fromCharCode(0xF0 | ((x >>> 18) & 0x07),
                                          0x80 | ((x >>> 12) & 0x3F),
                                          0x80 | ((x >>> 6 ) & 0x3F),
                                          0x80 | ( x         & 0x3F));
        }
        return output;
      }
      
      /*
       * Encode a string as utf-16
       */
      function str2rstr_utf16le(input)
      {
        var output = "";
        for(var i = 0; i < input.length; i++)
          output += String.fromCharCode( input.charCodeAt(i)        & 0xFF,
                                        (input.charCodeAt(i) >>> 8) & 0xFF);
        return output;
      }
      
      function str2rstr_utf16be(input)
      {
        var output = "";
        for(var i = 0; i < input.length; i++)
          output += String.fromCharCode((input.charCodeAt(i) >>> 8) & 0xFF,
                                         input.charCodeAt(i)        & 0xFF);
        return output;
      }
      
      /*
       * Convert a raw string to an array of little-endian words
       * Characters >255 have their high-byte silently ignored.
       */
      function rstr2binl(input)
      {
        var output = Array(input.length >> 2);
        for(var i = 0; i < output.length; i++)
          output[i] = 0;
        for(var i = 0; i < input.length * 8; i += 8)
          output[i>>5] |= (input.charCodeAt(i / 8) & 0xFF) << (i%32);
        return output;
      }
      
      /*
       * Convert an array of little-endian words to a string
       */
      function binl2rstr(input)
      {
        var output = "";
        for(var i = 0; i < input.length * 32; i += 8)
          output += String.fromCharCode((input[i>>5] >>> (i % 32)) & 0xFF);
        return output;
      }
      
      /*
       * Calculate the MD5 of an array of little-endian words, and a bit length.
       */
      function binl_md5(x, len)
      {
        /* append padding */
        x[len >> 5] |= 0x80 << ((len) % 32);
        x[(((len + 64) >>> 9) << 4) + 14] = len;
      
        var a =  1732584193;
        var b = -271733879;
        var c = -1732584194;
        var d =  271733878;
      
        for(var i = 0; i < x.length; i += 16)
        {
          var olda = a;
          var oldb = b;
          var oldc = c;
          var oldd = d;
      
          a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
          d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
          c = md5_ff(c, d, a, b, x[i+ 2], 17,  606105819);
          b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
          a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
          d = md5_ff(d, a, b, c, x[i+ 5], 12,  1200080426);
          c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
          b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
          a = md5_ff(a, b, c, d, x[i+ 8], 7 ,  1770035416);
          d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
          c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
          b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
          a = md5_ff(a, b, c, d, x[i+12], 7 ,  1804603682);
          d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
          c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
          b = md5_ff(b, c, d, a, x[i+15], 22,  1236535329);
      
          a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
          d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
          c = md5_gg(c, d, a, b, x[i+11], 14,  643717713);
          b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
          a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
          d = md5_gg(d, a, b, c, x[i+10], 9 ,  38016083);
          c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
          b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
          a = md5_gg(a, b, c, d, x[i+ 9], 5 ,  568446438);
          d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
          c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
          b = md5_gg(b, c, d, a, x[i+ 8], 20,  1163531501);
          a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
          d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
          c = md5_gg(c, d, a, b, x[i+ 7], 14,  1735328473);
          b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
      
          a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
          d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
          c = md5_hh(c, d, a, b, x[i+11], 16,  1839030562);
          b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
          a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
          d = md5_hh(d, a, b, c, x[i+ 4], 11,  1272893353);
          c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
          b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
          a = md5_hh(a, b, c, d, x[i+13], 4 ,  681279174);
          d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
          c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
          b = md5_hh(b, c, d, a, x[i+ 6], 23,  76029189);
          a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
          d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
          c = md5_hh(c, d, a, b, x[i+15], 16,  530742520);
          b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
      
          a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
          d = md5_ii(d, a, b, c, x[i+ 7], 10,  1126891415);
          c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
          b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
          a = md5_ii(a, b, c, d, x[i+12], 6 ,  1700485571);
          d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
          c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
          b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
          a = md5_ii(a, b, c, d, x[i+ 8], 6 ,  1873313359);
          d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
          c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
          b = md5_ii(b, c, d, a, x[i+13], 21,  1309151649);
          a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
          d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
          c = md5_ii(c, d, a, b, x[i+ 2], 15,  718787259);
          b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
      
          a = safe_add(a, olda);
          b = safe_add(b, oldb);
          c = safe_add(c, oldc);
          d = safe_add(d, oldd);
        }
        return Array(a, b, c, d);
      }
      
      /*
       * These functions implement the four basic operations the algorithm uses.
       */
      function md5_cmn(q, a, b, x, s, t)
      {
        return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b);
      }
      function md5_ff(a, b, c, d, x, s, t)
      {
        return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
      }
      function md5_gg(a, b, c, d, x, s, t)
      {
        return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
      }
      function md5_hh(a, b, c, d, x, s, t)
      {
        return md5_cmn(b ^ c ^ d, a, b, x, s, t);
      }
      function md5_ii(a, b, c, d, x, s, t)
      {
        return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
      }
      
      /*
       * Add integers, wrapping at 2^32. This uses 16-bit operations internally
       * to work around bugs in some JS interpreters.
       */
      function safe_add(x, y)
      {
        var lsw = (x & 0xFFFF) + (y & 0xFFFF);
        var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
        return (msw << 16) | (lsw & 0xFFFF);
      }
      
      /*
       * Bitwise rotate a 32-bit number to the left.
       */
      function bit_rol(num, cnt)
      {
        return (num << cnt) | (num >>> (32 - cnt));
      }
      

      Ursprung
      Das Script basiert auf einem Post den ich mal gefunden habe und zwar hier:
      https://community.openhab.org/t/disable-internet-connection-of-specific-echo-device-on-fritz-box-via-script/91374
      Das ist nur ein 5 Zeilen Shell Script und ich habe mir gedacht dass das auch in JavaScript gehen müsste - tut es auch :-)

      Funktionsweise
      Das Script läuft in mehreren Stufen ab ... Erst muss eine "Challange" von der FB abgeholt werden. Damit wird dann zusammen mit dem Passwort ein MD5 Hash gebildet.
      Das nutzt man dann wiederum um die SID von der Fritzbox zu bekommen. Jetzt kann man mit der SID diverse Anfragen an die FB stellen.
      Ich lese dann z.B. alle verfügbaren Filter Profile aus und die bekannten Geräte in der Fritzbox nebst ihren IDs.
      Dieser Part ist in dem ursprünglichen Script etwas rudimentär. Da geht es dann mit F12 und Developer Tools ans eingemachte. Aber das lasse ich hier erstmal weg.
      Am Ende ruft das Script dann eine Seite der FB auf und setzt damit dann den neuen Filter für das Device.

      Ich sage es mal gleich vorweg ... Die ist keine FB API ! Im Grunde ist es ein Hack der den Login und die Datenbewegung der FB Webseite nutzt - nur eben ohne die Webseite.
      Und ja ... Man kann mit diesem "Toolset" noch mehr anstellen. Ich kann z.B. alle Geräte der FB lesen mit ihren IPs. Im Grunde muss man über die Developer Tools nur mal schauen was die Webseite so an Daten schiebt wenn man etwas einstellt.
      Daraus kann man dann interessante Rückschlüsse ziehen. Aber das ist out of Scope hier ...

      Einrichtung und Nutzung
      Tja ihr müsst das Script kopieren und im JavaScript Adapter ein neues JavaScript erstellen. Da dann den Inhalt rein kopieren. 90% erledigt :-)
      Jetzt etwas "Schnüffeln" und Settings setzen.

      • Zum Testen solltet ihr erstmal die "Static Version" nutzen. Damit läuft das Script nur 1x durch. Wie man das aktiviert steht im Script. Wenn das Script dann sauber läuft kann man auf die "Object Trigger Version" umschwenken und alles über einen Datenpunkt in ioB steuern. Infos dazu stehen im Script.
      • Erstmal müsst ihr natürlich bei FbIp und FbPassword die entsprechenden Werte eintragen. Also einmal die FB IP und das FB Passwort.
      • Wenn der Login auf eure FritzBox mittels User und Passwort erfolgt, dann muss in FbUser der User noch eingetragen werden. Ansonsten dieses Feld einfach leer lassen.
      • FbListOnly jetzt auf true setzen. Denn wir wollen erstmal nur sehen ob es a) klappt und b) die Filter und Devices listen.
      • Wenn alles eingetragen ist (FbDevice & FbProfile sind an dieser Stelle noch unwichtig) dann mal das Script abspulen lassen ...
      • Ihr bekommt jetzt im Logging diverse Ausgaben (hoffentlich keine Fehler). Einmal folgendes:
      06:29:58.055	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: > Decode Filters
      06:29:58.056	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Standard' has ID : filtprof1
      06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Gast' has ID : filtprof2
      06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Unbeschränkt' has ID : filtprof3
      06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Gesperrt' has ID : filtprof4
      

      Und dann noch sowas:

      06:30:01.191	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-L-Acer-L' has ID : user2482
      06:30:01.191	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-L-Pi' has ID : landevice37174
      06:30:01.192	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-M-Tablet-W' has ID : landevice2671
      

      Wenn das gelistet wird dann funktioniert def. schon mal der Zugriff auf die FB. Wenn nicht ... Nun dann müssen wir mal guggn :-)

      • Was wir jetzt brauchen ist erstmal das Gerät was wir einstellen wollen. Ihr nehmt also aus der Device Liste die ID.
        Beispiel:
        Device named 'Comp-L-Acer-L' has ID : user2482
        Wir brauchen den Namen und NICHT die ID!! Also kopieren wir uns Comp-L-Acer-L als Beispiel.
        Diesen Namen kopieren wir im Script hinter FbDevice.
      • Jetzt brauchen wir noch den Filter. Auch die wurden ja im ersten Lauf gelistet.
        Beispiel:
        Filter named 'Gesperrt' has ID : filtprof4
        Wir brauchen die ID und NICHT den Namen. Wir kopieren uns also filtprof4 als Beispiel.
        Diese ID tragen wir dann bei FbProfile ein.
        Anmerkung ... Ok das könnte man noch auf den Namen umstellen merke ich gerade :-)
      • FbListOnly jetzt auf false setzen.
      • Jetzt kommt der Teil wo die Kuh ins Gras beisst :-) Wir lassen das Script laufen.
        Und wenn nicht alles daneben geht, dann ändert sich wie von Zauberhand der Filter für den entsprechenden Rechner.
        Und in meinem Fall gingen 2 Jahre Lösungssuche für dieses Problem zu Ende ....

      Hinweise
      Mir ist bewusst das man an dem Script noch einiges verbessern könnte. Und ich würde es demnächst auch in github einstellen inkl. Anleitung.
      Man könnte es auch in einen Adapter überführen. Aber erstmal wäre gut zu wissen ob es bei anderen auch klappt ...
      Wenn man das Script mit der VIS verheiraten will, dann muss man da noch etwas dran umbauen. Da braucht es dann 2-3 Datenpunkte.
      Aber ein riesen Drama sollte das nicht sein ...

      Ok jetzt bin ich gespannt auf das Feedback :-)

      Bis dahin ...

      NegaleinN Offline
      NegaleinN Offline
      Negalein
      Global Moderator
      schrieb am zuletzt editiert von
      #36

      @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

      Evtl. muss das dann auch mal nach Scripting verschoben werden ...

      done :)

      ° Node.js: 20.17.0 NPM: 10.8.2
      ° Proxmox, Ubuntu 22.04.3 LTS
      ° Fixer ---> iob fix

      1 Antwort Letzte Antwort
      0
      • moelskiM moelski

        Moin !

        Gestern Abend doch glatt am TV verschlummert :face_with_rolling_eyes: Aber nun ...

        Hier aber nun das Script:

        /*******************************************************
        * Basis: 
        * https://community.openhab.org/t/disable-internet-connection-of-specific-echo-device-on-fritz-box-via-script/91374
        * https://forum.fhem.de/index.php?topic=109689.0
        * https://www.kuketz-blog.de/fritzbox-wlan-toggle-per-skript/
        *******************************************************/ 
        
        /*******************************************************
        * Script SetDeviceFilter
        * 
        * 13.09.20 V1.09    Regex mit Escape beim . und Fix für Liste
        * 12.09.20 V1.08    Regex um . erweitert
        * 12.09.20 V1.07    Erweritertes Logging aktivierbar für Fehlersuche
        * 08.09.20 V1.06    Anpassungen an FirtzOS 7.20 (Regex Auswertung für Listenerstellung)
        * 05.09.20 V1.05    Ausgabe einer Device / Filter Liste möglich als JSON 
        *                   Logging reduziert wenn FbListOnly = false
        * 04.09.20 V1.04    Fix für doppelte Einträge in Listen
        * 02.09.20 V1.03    Erweiterung zum automatischen Start mittels ioB Datenpunkt
        * 02.09.20 V1.02    Script Abbruch wenn der Login nicht geklappt hat
        * 02.09.20 V1.01    Regex Erweiterung, Login auch mit User/Passwort
        * 01.09.20 V1.00    erste öffentliches Release
        *******************************************************/ 
        const Version = "1.09";
        
        // TODO
        // * Liste erzeugen mit Devices und aktuellem Filter
        //   > als JSON in einen DP speichern -> für Tabellenauswertung
        
        // https://forum.iobroker.net/topic/16184/http-request-in-javascript/5
        var request = require('request');
        var headers = { 
            'Content-Type': 'application/x-www-form-urlencoded', 
            'User-Agent': 'curl/7.64.0', 
            'Accept': '*/*'
        }; 
        
        /*******************************************************
        * E I N S T E L L U N G E N 
        *******************************************************/
        
        // Die IP Der Fritzbox
        const FbIp          = "192.168.30.1";              
        // Der User der Fritzbox 
        // HINWEIS : Den Benutzer leer lassen wenn die Anmeldung an der FB nur mit Passwort erfolgt !
        const FbUser        = ""; 
        // Das Password der Fritzbox                  
        const FbPassword    = "";   
        // welcher Rechner soll "bearbeitet" werden                
        var   FbDevice      = "Comp-L-Pi";   
        // Das neue Profil für den Rechner              
        var   FbProfile     = "filtprof4";  
        // erweitertes Logging aktivieren für Fehlersuche
        const FbDebugging   = false;
        // Keine Änderung an der Fritzbox (true) -> Listet dann nur alle Profile und Rechner               
        const FbListOnly    = false; 
        // Device / Filter Liste erzeugen Ja (true), Nein (false)
        const FbCreateList  = true; 
        // Datenpunkt (string) für einen automatischen Script Start (muss angelegt werden!)
        // Der Datenpunkt wir mit Device;Profil beschrieben. Bsp: Comp-L-Pi;filtprof1
        const FbIobObject   = "Global.0.Fritzbox.ChangeDeviceFilter"  
        // Datenpunkt in den die Device / Filter Liste als JSON geschrieben wird                    
        const FbIobJsonList = "Global.0.Fritzbox.DeviceFilterListJson" 
        
        var secChallenge;
        var secMd5;
        var secLogin;
        var secSid;
        var secProfileNames = [];
        var secProfileIds   = [];
        var secDeviceNames  = [];
        var secDeviceIds    = [];
        var JsonList        = "";
        
        /*******************************************************
        * Object Trigger Version
        * Wenn auf einen Datenpunkt getriggert werden soll, dann muss der folgende Block auskommentiert werden. 
        * FbIobObject muss mit einem string Datenpunkt versehen werden.
        * Den Static Teil dann auskommentieren !
        * Diese Variante erlaubt es das Script dynamisch für alle Devices / Profile der FB zu verwenden. 
        *******************************************************/
        on({id: FbIobObject, change: "ne"}, function (obj) {
            console.log("Profile Changer started ... (Version : " + Version + ")");
            var data = obj.state.val.split(";");
            if (data.length != 2) {
                console.log("Wrong parameter : " + obj.state.val);
                return;
            }
        
            if (FbDebugging){
                console.log(" > Datapoint : " + data);
            }
        
            console.log("Computer : " + data[0]);
            console.log("Filter   : " + data[1]);
            FbDevice  = data[0];
            FbProfile = data[1];
        
            getFbChallenge();
            
            console.log("Profile Changer done");
        });
        
        /*******************************************************
        * Static Version
        * Die folgenden 3 Zeilen lassen das Script  sofort laufen. 
        * Dabei sollte der Object Trigger Block auskommentiert werden. 
        *******************************************************/
        // console.log("Profile Changer started ... (Version : " + Version + ")");
        // getFbChallenge();
        // console.log("Profile Changer done");
        
        // Get the Challenge String from the FritzBox
        // Compute the md5 Hash
        function getFbChallenge(){
            console.log("function getFbChallenge");
            request.get({
                url:        'http://' + FbIp + '/login_sid.lua?username=' + FbUser,
                headers:    headers
            }, function(error, response, body) {
                if (error) log(error, 'error');
                //console.log(response);
                // https://developer.mozilla.org/de/docs/Web/JavaScript/Reference/Global_Objects/RegExp
                // <?xml version="1.0" encoding="utf-8"?><SessionInfo><SID>0000000000000000</SID><Challenge>d36de231</Challenge><BlockTime>0</BlockTime><Rights></Rights></SessionI
                secChallenge = body.match(/Challenge>(.*)<\/Challenge/)[1];
                console.log(" > Challenge : " + secChallenge);
        
                var uft16le = str2rstr_utf16le(secChallenge + "-" +FbPassword );
                var md5     = rstr_md5(uft16le);
                secMd5      = rstr2hex(md5);
                        
                console.log(" > MD5       : " + secMd5);   
        
                // response="${challenge}-${md5}"
                // sid=$(curl -i -s -k -d "response=${response}&username=" "http://$1" | grep -Po -m 1 '(?<=sid=)[a-f\d]+' | tail -1)
                secLogin = "response=" + secChallenge + "-" + secMd5 + "&username=" + FbUser;
        
                console.log(" > Login     : " + secLogin);
        
                getFbSid();
            });
        }
        
        // Get the SID from the Fritzbox
        function getFbSid() {
            console.log("function getFbSid");
        
            request.post({
                url:        'http://' + FbIp,
                headers:    headers, 
                form:       secLogin
            }, function(error, response, body) {
                if (error) log(error, 'error');
                // SID filtern 
                // "sid":"c503b24dae458086"
                try {
                    secSid = response.body.match(/\"sid\":\"(.*)\"/)[1];
                }
                catch (e) {
                    if (secSid == undefined) {
                        console.log("Your login was not successful. End Script", Error)
                        return;
                    }
                }
        
                console.log(" > SID       : " + secSid);
        
                //getFbDeviceInfos();
                getFbProfiles();
            });
        } 
        
        // TBD: 
        // Netzwerkinfos mit allen Devices lesen: 
        // var req = "xhr=1&sid=" + secSid + "&lang=de&page=netDev&xhrId=cleanup&useajax=1&no_sidrenew=" 
        function getFbDeviceInfos(){
            console.log("function getFbDeviceInfos");
            var req = "xhr=1&sid=" + secSid + "&lang=de&page=netDev&xhrId=cleanup&useajax=1&no_sidrenew=" 
        
            request.post({
                url:        'http://' + FbIp + '/data.lua',
                headers:    headers, 
                form:       req
            }, function(error, response, body) {
                if (error) log(error, 'error');
                console.log(response.body);
            });    
        }
        
        // Get all Profiles from the Fritzbox
        function getFbProfiles() {
            console.log("function getFbProfiles");
            // #curl -d "xhr=1&sid=${sid}&lang=de&no_sidrenew=&page=kidPro" "http://$1/data.lua"
            var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidPro";
        
            request.post({
                url:        'http://' + FbIp + '/data.lua',
                headers:    headers, 
                form:       req
            }, function(error, response, body) {
                if (error) log(error, 'error');
                
                if (FbDebugging){
                    console.log(" > response.body : \n" + response.body);
                }
        
                secProfileIds   = [];
                secProfileNames = [];
        
                console.log(" > Decode Names")
                var rx = new RegExp( /class=\"name\"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g );
                while( (match = rx.exec( body )) != null ) {
                    secProfileNames.push(match[1]);
                }
                if (FbDebugging){
                    console.log(" > secProfileNames : \n" + secProfileNames);
                }
        
                console.log(" > Decode Filters")
                // submit" name="edit" value="
                rx = new RegExp( /submit\"\sname=\"edit\"\svalue=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sclass=\"icon/g );
                while( (match = rx.exec( body )) != null ) {
                    secProfileIds.push(match[1]);
                }
                if (FbDebugging){
                    console.log(" > secProfileIds : \n" + secProfileIds);
                }
        
                if (FbListOnly) {
                    console.log("Filter Count : " + secProfileIds.length);
                    for (var i = 0; i < secProfileIds.length; i++) {
                        console.log("Filter named '"+ secProfileNames[i] + "' has ID : " + secProfileIds[i]);
                    }
                }
        
                getFbDevices();
            });
        }
        
        // IMPORTANT: 
        // The DeviceId changes when you switch the profile. 
        // If you use default profiles it is something like "landevice308962"
        // If you use your own profiles it is something like "user7749"
        function getFbDevices() {
            console.log("function getFbDevices");
            // xhr=1&sid=5f5ba302815594d8&lang=de&no_sidrenew=&page=kidLis
            var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidLis";
        
            request.post({
                url:        'http://' + FbIp + '/data.lua',
                headers:    headers, 
                form:       req
            }, function(error, response, body) {
                if (error) log(error, 'error');
                if (FbDebugging){
                    console.log(" > response.body : \n" + response.body);
                }
        
                // TESTING
                //body = getState("Global.0.Testing.StringValue").val;
        
                secDeviceNames = [];
                secDeviceIds   = [];
        
                console.log(" > Decode Device Names")
                var rx = new RegExp( /class=\"name"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g );
                while( (match = rx.exec( body )) != null ) {
                    secDeviceNames.push(match[1]);
                }
                if (FbDebugging){
                    console.log(" > secDeviceNames : \n" + secDeviceNames);
                }
        
                console.log(" > Decode Device Ids")
                rx = new RegExp( /name=\"profile:([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"><option/g );
                while( (match = rx.exec( body )) != null ) {
                    secDeviceIds.push(match[1]);
                }
                if (FbDebugging){
                    console.log(" > secDeviceIds : \n" + secDeviceIds);
                }
                
                if (FbListOnly) {
                    console.log("Device Count : " + secDeviceIds.length);
                    for (var i = 0; i < secDeviceNames.length; i++) {
                        console.log(i + " - Device named '"+ secDeviceNames[i] + "' has ID : " + secDeviceIds[i]);
                    }
                }
        
                if (!FbListOnly) { 
                    setFbSperre(); 
                } else { 
                    console.log("DONE : Listmode - No Device blocking ..."); 
                }
            });
        }
        
        //curl -d "sid=${sid}&edit=$3&time=$4&budget=unlimited&apply=&page=kids_profileedit" "http://$1/data.lua" >/dev/null 2>&1
        function setFbSperre() {
            console.log("function setFbSperre");
            
            // Device suchen ... 
            var devId = -1;
            for (var i = 0; i < secDeviceNames.length; i++) {
                if (secDeviceNames[i] == FbDevice) {
                    devId = i;
                }
            }
            
            console.log(" > Device Nr   : " + devId); 
            console.log(" > Device ID   : " + secDeviceIds[devId]); 
            console.log(" > Device Name : " + secDeviceNames[devId]);
            //var profile = "filtprof4";
            //var profile = "filtprof3079";
        
            // xhr=1&sid=b46f3d48b3b0fcc2&lang=de&no_sidrenew=&profile%3Alandevice308962=filtprof4&apply=&oldpage=%2Finternet%2Fkids_userlist.lua
            var req2 = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&profile%3A" + secDeviceIds[devId] + "=" + FbProfile + "&apply=&oldpage=%2Finternet%2Fkids_userlist.lua";
            console.log(" > req : " + req2);
            //var req2 = "xhr=1&sid=" + sid + "&lang=de&no_sidrenew=&profile%3Alandevice308962=filtprof3079&profile%3Auser7725=filtprof4&profile%3Alandevice2671=filtprof1&profile%3Alandevice2444=filtprof1&profile%3Alandevice2330=filtprof1&profile%3Alandevice2195=filtprof1&profile%3Alandevice229903=filtprof1&profile%3Alandevice8486=filtprof1&profile%3Alandevice1908=filtprof1&profile%3Alandevice1865=filtprof1&profile%3Alandevice516865=filtprof1&profile%3Alandevice518551=filtprof1&profile%3Alandevice169594=filtprof1&profile%3Alandevice171398=filtprof1&profile%3Alandevice175510=filtprof1&profile%3Alandevice177635=filtprof1&profile%3Alandevice178875=filtprof1&profile%3Alandevice179702=filtprof1&profile%3Alandevice180995=filtprof1&profile%3Alandevice182196=filtprof1&profile%3Alandevice183607=filtprof1&profile%3Alandevice184405=filtprof1&profile%3Alandevice638220=filtprof1&profile%3Alandevice634583=filtprof1&profile%3Alandevice7650=filtprof1&profile%3Alandevice353720=filtprof1&profile%3Auser7731=filtprof4&profile%3Alandevice170191=filtprof1&profile%3Alandevice941076=filtprof1&profile%3Alandevice358722=filtprof1&profile%3Alandevice2859=filtprof1&profile%3Auser7739=filtprof4&profile%3Alandevice8035=filtprof1&profile%3Auser7733=filtprof4&profile%3Auser7740=filtprof4&profile%3Alandevice1333=filtprof1&profile%3Alandevice6211=filtprof1&profile%3Alandevice35977=filtprof1&profile%3Alandevice8101=filtprof1&profile%3Alandevice8174=filtprof1&profile%3Auser7729=filtprof4&profile%3Alandevice1125=filtprof1&profile%3Alandevice1124=filtprof1&profile%3Alandevice7715=filtprof1&profile%3Alandevice52361=filtprof1&profile%3Alandevice30084=filtprof1&profile%3Alandevice26151=filtprof1&profile%3Alandevice26138=filtprof1&profile%3Alandevice6162=filtprof1&profile%3Alandevice5204=filtprof1&profile%3Alandevice1129367=filtprof1&profile%3Alandevice9399=filtprof1&profile%3Alandevice232149=filtprof1&profile%3Alandevice6935=filtprof1&profile%3Alandevice941333=filtprof1&profile%3Alandevice941569=filtprof1&apply=&oldpage=%2Finternet%2Fkids_userlist.lua";
            //var req2 = "xhr=1&sid=" + sid + "&lang=de&no_sidrenew=&profile:landevice308962=filtprof4&apply=&oldpage=/internet/kids_userlist.lua";
        
            request.post({
                url:        'http://' + FbIp + '/data.lua',
                headers:    headers, 
                form:       req2
            }, function(error, response, body) {
                if (error) log(error, 'error');
                if (FbDebugging){
                    console.log(" > response : \n" + response);
                }
        
                console.log("Change done. Check your FB / Device if it worked :-)");
        
                if (FbCreateList) {
                    getFbDeviceFilterList();
                }
            });    
        }
        
        function getFbDeviceFilterList() {
            console.log("function getFbDeviceFilterList");
            
            // xhr=1&sid=e2d99419592fb780&lang=de&no_sidrenew=&page=kidLis
            var req = "xhr=1&sid=" + secSid + "&lang=de&no_sidrenew=&page=kidLis";
        
            request.post({
                url:        'http://' + FbIp + '/data.lua',
                headers:    headers, 
                form:       req
            }, function(error, response, body) {
                //if (error) log(error, 'error');
                //console.log(response.body);
        
                JsonList = "";
                var tmpDevices      = [];
                var tmpUsage        = [];
                var tmpProfilesId   = [];
                var tmpProfilesName = [];
                var tmpTimes        = [];
        
                // TESTING
                //body = getState("Global.0.Testing.StringValue").val;
                
                console.log(" > Decode Device Names")
                var rx = new RegExp(/class=\"name"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_\.]*)\"\sdatalabel/g);
                while ((match = rx.exec(body)) != null) {
                    tmpDevices.push(match[1]);
                }
                //console.log(" > " + tmpDevices);
        
                console.log(" > Decode Device Profile ID");
                //rx = new RegExp(/button\s.*?value=\"([a-zA-Z0-9 äöüÄÖÜ]*?)\"/g);
                rx = new RegExp(/option\svalue=\"([a-zA-Z0-9 äöüÄÖÜ\-]*?)\"\sselected/g);
                while ((match = rx.exec(body)) != null) {
                    tmpProfilesId.push(match[1]);
                }
                //console.log(" > " + tmpProfilesId);
        
                console.log(" > Decode Device Profile Names");
                for (var i = 0; i < tmpProfilesId.length; i++) {
                    for (var j = 0; j < secProfileIds.length; j++) { 
                        //console.log(tmpProfilesId[i] + " > " + secProfileIds[j]);
                        if (tmpProfilesId[i] == secProfileIds[j]){
                            tmpProfilesName.push(secProfileNames[j]);
                        }
                    }
                }
                //console.log(" > " + tmpProfilesName);
        
                console.log(" > Decode Internetnutzung")
                rx = new RegExp(/td\sdatalabel=\"[a-zA-Z0-9 äöüÄÖÜ\-]+\"\sclass=\"usage\">(.*?)<\/td>/g);
                while ((match = rx.exec(body)) != null) { 
                    var tmp = match[1];
                    tmp = tmp.replace("<span>", "").replace("<\/span>", "");
                    tmpUsage.push(tmp);
                }
                //console.log(" > " + tmpUsage);
        
                console.log(" > Decode Times")
                rx = new RegExp(/td\sdatalabel=\"[a-zA-Z0-9 äöüÄÖÜ\-]+\"\sclass=\"bar\stime\">(.*?)<\/td>/g);
                while ((match = rx.exec(body)) != null) {
                    var tmp = match[1];
                    if (tmp.includes("\"")) {
                        tmp = tmp.replace("<span title=\"", "");
                        tmp = tmp.slice(0, tmp.indexOf("\""));
                        tmpTimes.push(tmp);
                    } else {
                        tmpTimes.push(tmp);
                    }
                }
                //console.log(" > " + tmpTimes);
        
                // console.log("Device Count : " + secDeviceIds.length);
                JsonList = "[";
                for (var i = 0; i < tmpDevices.length; i++) {
                    JsonList += "{";
                    JsonList += "\"Device\": \"" + tmpDevices[i] + "\"," 
                    JsonList += "\"Profile ID\": \"" + tmpProfilesId[i] + "\","
                    JsonList += "\"Profile Name\": \"" + tmpProfilesName[i] + "\","
                    JsonList += "\"Usage\": \"" + tmpUsage[i] + "\","
                    JsonList += "\"Time\": \"" + tmpTimes[i] + "\""
                    if (i != tmpDevices.length - 1) { JsonList += "},"; } else { JsonList += "}"; }
                }
                JsonList += "]";
        
                //console.log(JsonList);
                setState(FbIobJsonList,  JsonList, true);
        
                console.log("DONE : List generated ...");
            });   
        }
        
        
        // https://gist.githubusercontent.com/josedaniel/951664/raw/33c7a6b44d6cc53dc75cb5230fb551bc4ba7a46a/md5.js
        // http://pajhome.org.uk/crypt/md5/instructions.html
        
        /*
         * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
         * Digest Algorithm, as defined in RFC 1321.
         * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
         * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
         * Distributed under the BSD License
         * See http://pajhome.org.uk/crypt/md5 for more info.
         */
        
        /*
         * Configurable variables. You may need to tweak these to be compatible with
         * the server-side, but the defaults work in most cases.
         */
        var hexcase = 0;   /* hex output format. 0 - lowercase; 1 - uppercase        */
        var b64pad  = "";  /* base-64 pad character. "=" for strict RFC compliance   */
        
        /*
         * These are the functions you'll usually want to call
         * They take string arguments and return either hex or base-64 encoded strings
         */
        function hex_md5(s)    { return rstr2hex(rstr_md5(str2rstr_utf8(s))); }
        function b64_md5(s)    { return rstr2b64(rstr_md5(str2rstr_utf8(s))); }
        function any_md5(s, e) { return rstr2any(rstr_md5(str2rstr_utf8(s)), e); }
        function hex_hmac_md5(k, d)
          { return rstr2hex(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
        function b64_hmac_md5(k, d)
          { return rstr2b64(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d))); }
        function any_hmac_md5(k, d, e)
          { return rstr2any(rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)), e); }
        
        /*
         * Perform a simple self-test to see if the VM is working
         */
        function md5_vm_test()
        {
          return hex_md5("abc").toLowerCase() == "900150983cd24fb0d6963f7d28e17f72";
        }
        
        /*
         * Calculate the MD5 of a raw string
         */
        function rstr_md5(s)
        {
          return binl2rstr(binl_md5(rstr2binl(s), s.length * 8));
        }
        
        /*
         * Calculate the HMAC-MD5, of a key and some data (raw strings)
         */
        function rstr_hmac_md5(key, data)
        {
          var bkey = rstr2binl(key);
          if(bkey.length > 16) bkey = binl_md5(bkey, key.length * 8);
        
          var ipad = Array(16), opad = Array(16);
          for(var i = 0; i < 16; i++)
          {
            ipad[i] = bkey[i] ^ 0x36363636;
            opad[i] = bkey[i] ^ 0x5C5C5C5C;
          }
        
          var hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8);
          return binl2rstr(binl_md5(opad.concat(hash), 512 + 128));
        }
        
        /*
         * Convert a raw string to a hex string
         */
        function rstr2hex(input)
        {
          try { hexcase } catch(e) { hexcase=0; }
          var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef";
          var output = "";
          var x;
          for(var i = 0; i < input.length; i++)
          {
            x = input.charCodeAt(i);
            output += hex_tab.charAt((x >>> 4) & 0x0F)
                   +  hex_tab.charAt( x        & 0x0F);
          }
          return output;
        }
        
        /*
         * Convert a raw string to a base-64 string
         */
        function rstr2b64(input)
        {
          try { b64pad } catch(e) { b64pad=''; }
          var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
          var output = "";
          var len = input.length;
          for(var i = 0; i < len; i += 3)
          {
            var triplet = (input.charCodeAt(i) << 16)
                        | (i + 1 < len ? input.charCodeAt(i+1) << 8 : 0)
                        | (i + 2 < len ? input.charCodeAt(i+2)      : 0);
            for(var j = 0; j < 4; j++)
            {
              if(i * 8 + j * 6 > input.length * 8) output += b64pad;
              else output += tab.charAt((triplet >>> 6*(3-j)) & 0x3F);
            }
          }
          return output;
        }
        
        /*
         * Convert a raw string to an arbitrary string encoding
         */
        function rstr2any(input, encoding)
        {
          var divisor = encoding.length;
          var i, j, q, x, quotient;
        
          /* Convert to an array of 16-bit big-endian values, forming the dividend */
          var dividend = Array(Math.ceil(input.length / 2));
          for(i = 0; i < dividend.length; i++)
          {
            dividend[i] = (input.charCodeAt(i * 2) << 8) | input.charCodeAt(i * 2 + 1);
          }
        
          /*
           * Repeatedly perform a long division. The binary array forms the dividend,
           * the length of the encoding is the divisor. Once computed, the quotient
           * forms the dividend for the next step. All remainders are stored for later
           * use.
           */
          var full_length = Math.ceil(input.length * 8 /
                                            (Math.log(encoding.length) / Math.log(2)));
          var remainders = Array(full_length);
          for(j = 0; j < full_length; j++)
          {
            quotient = Array();
            x = 0;
            for(i = 0; i < dividend.length; i++)
            {
              x = (x << 16) + dividend[i];
              q = Math.floor(x / divisor);
              x -= q * divisor;
              if(quotient.length > 0 || q > 0)
                quotient[quotient.length] = q;
            }
            remainders[j] = x;
            dividend = quotient;
          }
        
          /* Convert the remainders to the output string */
          var output = "";
          for(i = remainders.length - 1; i >= 0; i--)
            output += encoding.charAt(remainders[i]);
        
          return output;
        }
        
        /*
         * Encode a string as utf-8.
         * For efficiency, this assumes the input is valid utf-16.
         */
        function str2rstr_utf8(input)
        {
          var output = "";
          var i = -1;
          var x, y;
        
          while(++i < input.length)
          {
            /* Decode utf-16 surrogate pairs */
            x = input.charCodeAt(i);
            y = i + 1 < input.length ? input.charCodeAt(i + 1) : 0;
            if(0xD800 <= x && x <= 0xDBFF && 0xDC00 <= y && y <= 0xDFFF)
            {
              x = 0x10000 + ((x & 0x03FF) << 10) + (y & 0x03FF);
              i++;
            }
        
            /* Encode output as utf-8 */
            if(x <= 0x7F)
              output += String.fromCharCode(x);
            else if(x <= 0x7FF)
              output += String.fromCharCode(0xC0 | ((x >>> 6 ) & 0x1F),
                                            0x80 | ( x         & 0x3F));
            else if(x <= 0xFFFF)
              output += String.fromCharCode(0xE0 | ((x >>> 12) & 0x0F),
                                            0x80 | ((x >>> 6 ) & 0x3F),
                                            0x80 | ( x         & 0x3F));
            else if(x <= 0x1FFFFF)
              output += String.fromCharCode(0xF0 | ((x >>> 18) & 0x07),
                                            0x80 | ((x >>> 12) & 0x3F),
                                            0x80 | ((x >>> 6 ) & 0x3F),
                                            0x80 | ( x         & 0x3F));
          }
          return output;
        }
        
        /*
         * Encode a string as utf-16
         */
        function str2rstr_utf16le(input)
        {
          var output = "";
          for(var i = 0; i < input.length; i++)
            output += String.fromCharCode( input.charCodeAt(i)        & 0xFF,
                                          (input.charCodeAt(i) >>> 8) & 0xFF);
          return output;
        }
        
        function str2rstr_utf16be(input)
        {
          var output = "";
          for(var i = 0; i < input.length; i++)
            output += String.fromCharCode((input.charCodeAt(i) >>> 8) & 0xFF,
                                           input.charCodeAt(i)        & 0xFF);
          return output;
        }
        
        /*
         * Convert a raw string to an array of little-endian words
         * Characters >255 have their high-byte silently ignored.
         */
        function rstr2binl(input)
        {
          var output = Array(input.length >> 2);
          for(var i = 0; i < output.length; i++)
            output[i] = 0;
          for(var i = 0; i < input.length * 8; i += 8)
            output[i>>5] |= (input.charCodeAt(i / 8) & 0xFF) << (i%32);
          return output;
        }
        
        /*
         * Convert an array of little-endian words to a string
         */
        function binl2rstr(input)
        {
          var output = "";
          for(var i = 0; i < input.length * 32; i += 8)
            output += String.fromCharCode((input[i>>5] >>> (i % 32)) & 0xFF);
          return output;
        }
        
        /*
         * Calculate the MD5 of an array of little-endian words, and a bit length.
         */
        function binl_md5(x, len)
        {
          /* append padding */
          x[len >> 5] |= 0x80 << ((len) % 32);
          x[(((len + 64) >>> 9) << 4) + 14] = len;
        
          var a =  1732584193;
          var b = -271733879;
          var c = -1732584194;
          var d =  271733878;
        
          for(var i = 0; i < x.length; i += 16)
          {
            var olda = a;
            var oldb = b;
            var oldc = c;
            var oldd = d;
        
            a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
            d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
            c = md5_ff(c, d, a, b, x[i+ 2], 17,  606105819);
            b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
            a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
            d = md5_ff(d, a, b, c, x[i+ 5], 12,  1200080426);
            c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
            b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
            a = md5_ff(a, b, c, d, x[i+ 8], 7 ,  1770035416);
            d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
            c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
            b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
            a = md5_ff(a, b, c, d, x[i+12], 7 ,  1804603682);
            d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
            c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
            b = md5_ff(b, c, d, a, x[i+15], 22,  1236535329);
        
            a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
            d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
            c = md5_gg(c, d, a, b, x[i+11], 14,  643717713);
            b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
            a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
            d = md5_gg(d, a, b, c, x[i+10], 9 ,  38016083);
            c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
            b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
            a = md5_gg(a, b, c, d, x[i+ 9], 5 ,  568446438);
            d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
            c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
            b = md5_gg(b, c, d, a, x[i+ 8], 20,  1163531501);
            a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
            d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
            c = md5_gg(c, d, a, b, x[i+ 7], 14,  1735328473);
            b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
        
            a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
            d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
            c = md5_hh(c, d, a, b, x[i+11], 16,  1839030562);
            b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
            a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
            d = md5_hh(d, a, b, c, x[i+ 4], 11,  1272893353);
            c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
            b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
            a = md5_hh(a, b, c, d, x[i+13], 4 ,  681279174);
            d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
            c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
            b = md5_hh(b, c, d, a, x[i+ 6], 23,  76029189);
            a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
            d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
            c = md5_hh(c, d, a, b, x[i+15], 16,  530742520);
            b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
        
            a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
            d = md5_ii(d, a, b, c, x[i+ 7], 10,  1126891415);
            c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
            b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
            a = md5_ii(a, b, c, d, x[i+12], 6 ,  1700485571);
            d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
            c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
            b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
            a = md5_ii(a, b, c, d, x[i+ 8], 6 ,  1873313359);
            d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
            c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
            b = md5_ii(b, c, d, a, x[i+13], 21,  1309151649);
            a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
            d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
            c = md5_ii(c, d, a, b, x[i+ 2], 15,  718787259);
            b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
        
            a = safe_add(a, olda);
            b = safe_add(b, oldb);
            c = safe_add(c, oldc);
            d = safe_add(d, oldd);
          }
          return Array(a, b, c, d);
        }
        
        /*
         * These functions implement the four basic operations the algorithm uses.
         */
        function md5_cmn(q, a, b, x, s, t)
        {
          return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s),b);
        }
        function md5_ff(a, b, c, d, x, s, t)
        {
          return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
        }
        function md5_gg(a, b, c, d, x, s, t)
        {
          return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
        }
        function md5_hh(a, b, c, d, x, s, t)
        {
          return md5_cmn(b ^ c ^ d, a, b, x, s, t);
        }
        function md5_ii(a, b, c, d, x, s, t)
        {
          return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
        }
        
        /*
         * Add integers, wrapping at 2^32. This uses 16-bit operations internally
         * to work around bugs in some JS interpreters.
         */
        function safe_add(x, y)
        {
          var lsw = (x & 0xFFFF) + (y & 0xFFFF);
          var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
          return (msw << 16) | (lsw & 0xFFFF);
        }
        
        /*
         * Bitwise rotate a 32-bit number to the left.
         */
        function bit_rol(num, cnt)
        {
          return (num << cnt) | (num >>> (32 - cnt));
        }
        

        Ursprung
        Das Script basiert auf einem Post den ich mal gefunden habe und zwar hier:
        https://community.openhab.org/t/disable-internet-connection-of-specific-echo-device-on-fritz-box-via-script/91374
        Das ist nur ein 5 Zeilen Shell Script und ich habe mir gedacht dass das auch in JavaScript gehen müsste - tut es auch :-)

        Funktionsweise
        Das Script läuft in mehreren Stufen ab ... Erst muss eine "Challange" von der FB abgeholt werden. Damit wird dann zusammen mit dem Passwort ein MD5 Hash gebildet.
        Das nutzt man dann wiederum um die SID von der Fritzbox zu bekommen. Jetzt kann man mit der SID diverse Anfragen an die FB stellen.
        Ich lese dann z.B. alle verfügbaren Filter Profile aus und die bekannten Geräte in der Fritzbox nebst ihren IDs.
        Dieser Part ist in dem ursprünglichen Script etwas rudimentär. Da geht es dann mit F12 und Developer Tools ans eingemachte. Aber das lasse ich hier erstmal weg.
        Am Ende ruft das Script dann eine Seite der FB auf und setzt damit dann den neuen Filter für das Device.

        Ich sage es mal gleich vorweg ... Die ist keine FB API ! Im Grunde ist es ein Hack der den Login und die Datenbewegung der FB Webseite nutzt - nur eben ohne die Webseite.
        Und ja ... Man kann mit diesem "Toolset" noch mehr anstellen. Ich kann z.B. alle Geräte der FB lesen mit ihren IPs. Im Grunde muss man über die Developer Tools nur mal schauen was die Webseite so an Daten schiebt wenn man etwas einstellt.
        Daraus kann man dann interessante Rückschlüsse ziehen. Aber das ist out of Scope hier ...

        Einrichtung und Nutzung
        Tja ihr müsst das Script kopieren und im JavaScript Adapter ein neues JavaScript erstellen. Da dann den Inhalt rein kopieren. 90% erledigt :-)
        Jetzt etwas "Schnüffeln" und Settings setzen.

        • Zum Testen solltet ihr erstmal die "Static Version" nutzen. Damit läuft das Script nur 1x durch. Wie man das aktiviert steht im Script. Wenn das Script dann sauber läuft kann man auf die "Object Trigger Version" umschwenken und alles über einen Datenpunkt in ioB steuern. Infos dazu stehen im Script.
        • Erstmal müsst ihr natürlich bei FbIp und FbPassword die entsprechenden Werte eintragen. Also einmal die FB IP und das FB Passwort.
        • Wenn der Login auf eure FritzBox mittels User und Passwort erfolgt, dann muss in FbUser der User noch eingetragen werden. Ansonsten dieses Feld einfach leer lassen.
        • FbListOnly jetzt auf true setzen. Denn wir wollen erstmal nur sehen ob es a) klappt und b) die Filter und Devices listen.
        • Wenn alles eingetragen ist (FbDevice & FbProfile sind an dieser Stelle noch unwichtig) dann mal das Script abspulen lassen ...
        • Ihr bekommt jetzt im Logging diverse Ausgaben (hoffentlich keine Fehler). Einmal folgendes:
        06:29:58.055	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: > Decode Filters
        06:29:58.056	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Standard' has ID : filtprof1
        06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Gast' has ID : filtprof2
        06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Unbeschränkt' has ID : filtprof3
        06:29:58.057	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Filter named 'Gesperrt' has ID : filtprof4
        

        Und dann noch sowas:

        06:30:01.191	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-L-Acer-L' has ID : user2482
        06:30:01.191	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-L-Pi' has ID : landevice37174
        06:30:01.192	info	javascript.0 (20276) script.js.Fritzbox.CurlTestFB: Device named 'Comp-M-Tablet-W' has ID : landevice2671
        

        Wenn das gelistet wird dann funktioniert def. schon mal der Zugriff auf die FB. Wenn nicht ... Nun dann müssen wir mal guggn :-)

        • Was wir jetzt brauchen ist erstmal das Gerät was wir einstellen wollen. Ihr nehmt also aus der Device Liste die ID.
          Beispiel:
          Device named 'Comp-L-Acer-L' has ID : user2482
          Wir brauchen den Namen und NICHT die ID!! Also kopieren wir uns Comp-L-Acer-L als Beispiel.
          Diesen Namen kopieren wir im Script hinter FbDevice.
        • Jetzt brauchen wir noch den Filter. Auch die wurden ja im ersten Lauf gelistet.
          Beispiel:
          Filter named 'Gesperrt' has ID : filtprof4
          Wir brauchen die ID und NICHT den Namen. Wir kopieren uns also filtprof4 als Beispiel.
          Diese ID tragen wir dann bei FbProfile ein.
          Anmerkung ... Ok das könnte man noch auf den Namen umstellen merke ich gerade :-)
        • FbListOnly jetzt auf false setzen.
        • Jetzt kommt der Teil wo die Kuh ins Gras beisst :-) Wir lassen das Script laufen.
          Und wenn nicht alles daneben geht, dann ändert sich wie von Zauberhand der Filter für den entsprechenden Rechner.
          Und in meinem Fall gingen 2 Jahre Lösungssuche für dieses Problem zu Ende ....

        Hinweise
        Mir ist bewusst das man an dem Script noch einiges verbessern könnte. Und ich würde es demnächst auch in github einstellen inkl. Anleitung.
        Man könnte es auch in einen Adapter überführen. Aber erstmal wäre gut zu wissen ob es bei anderen auch klappt ...
        Wenn man das Script mit der VIS verheiraten will, dann muss man da noch etwas dran umbauen. Da braucht es dann 2-3 Datenpunkte.
        Aber ein riesen Drama sollte das nicht sein ...

        Ok jetzt bin ich gespannt auf das Feedback :-)

        Bis dahin ...

        NegaleinN Offline
        NegaleinN Offline
        Negalein
        Global Moderator
        schrieb am zuletzt editiert von
        #37

        @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

        Hier aber nun das Script:

        das heisst für jedes Gerät 2 Scripte (Profil "gesperrt" & Profil "frei")?

        ° Node.js: 20.17.0 NPM: 10.8.2
        ° Proxmox, Ubuntu 22.04.3 LTS
        ° Fixer ---> iob fix

        moelskiM 1 Antwort Letzte Antwort
        0
        • NegaleinN Negalein

          @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

          Hier aber nun das Script:

          das heisst für jedes Gerät 2 Scripte (Profil "gesperrt" & Profil "frei")?

          moelskiM Offline
          moelskiM Offline
          moelski
          schrieb am zuletzt editiert von
          #38

          @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

          das heisst für jedes Gerät 2 Scripte (Profil "gesperrt" & Profil "frei")?

          Also ich habe in meiner Visu ein paar Button ala
          "Kind sperren"
          "Kind darf"

          Diese Button setzen dann den Datenpunkt den man im Script definieren kann.
          Wie das geht habe ich ja im Script auch beschrieben.

          Und wenn dann der datenpunkt beschrieben wird, dann führt das Script die Aktion mit dem Rechner und dem Profil aus.
          Also hast du das Script nur 1x am laufen.

          Wenn du es statisch lösen willst ... Ja dann müsste man 2x das Script haben.
          Aber das würde ich wann immer es geht vermeiden :-)

          Grüße

          Grüße Dominik

          NegaleinN 2 Antworten Letzte Antwort
          0
          • moelskiM moelski

            @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

            das heisst für jedes Gerät 2 Scripte (Profil "gesperrt" & Profil "frei")?

            Also ich habe in meiner Visu ein paar Button ala
            "Kind sperren"
            "Kind darf"

            Diese Button setzen dann den Datenpunkt den man im Script definieren kann.
            Wie das geht habe ich ja im Script auch beschrieben.

            Und wenn dann der datenpunkt beschrieben wird, dann führt das Script die Aktion mit dem Rechner und dem Profil aus.
            Also hast du das Script nur 1x am laufen.

            Wenn du es statisch lösen willst ... Ja dann müsste man 2x das Script haben.
            Aber das würde ich wann immer es geht vermeiden :-)

            Grüße

            NegaleinN Offline
            NegaleinN Offline
            Negalein
            Global Moderator
            schrieb am zuletzt editiert von
            #39

            @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

            Diese Button setzen dann den Datenpunkt den man im Script definieren kann.

            Ah, jetzt versteh ich es

            const FbIobObject = "Global.0.Fritzbox.ChangeDeviceFilter"
            und Global.0gehört zb in javascript.0geändert.

            ° Node.js: 20.17.0 NPM: 10.8.2
            ° Proxmox, Ubuntu 22.04.3 LTS
            ° Fixer ---> iob fix

            moelskiM 1 Antwort Letzte Antwort
            0
            • NegaleinN Negalein

              @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

              Diese Button setzen dann den Datenpunkt den man im Script definieren kann.

              Ah, jetzt versteh ich es

              const FbIobObject = "Global.0.Fritzbox.ChangeDeviceFilter"
              und Global.0gehört zb in javascript.0geändert.

              moelskiM Offline
              moelskiM Offline
              moelski
              schrieb am zuletzt editiert von moelski
              #40

              @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

              und Global.0gehört zb in javascript.0geändert.

              Ja nö ja ne.
              wo der Datenpunkt liegt ist total Wurst.
              Und ich persönlich mag diese verteilten Datenpunkte auch nicht.
              Ich habe einen Ordner Global.0 und die liegt bei mir alles drin was ich an selbst definierten Datenpunkten habe.
              Aber wie gesagt ... wo das liegt ist Wumpe ;-)

              Grüße Dominik

              1 Antwort Letzte Antwort
              0
              • NegaleinN Offline
                NegaleinN Offline
                Negalein
                Global Moderator
                schrieb am zuletzt editiert von
                #41

                @moelski

                Ich bekomm im Log nur den normalen Start angezeigt.

                21:31:34.030	info	javascript.0 (595) Stop script script.js.Fritz.Geraetesperre
                21:31:34.183	info	javascript.0 (595) Start javascript script.js.Fritz.Geraetesperre
                21:31:34.188	info	javascript.0 (595) script.js.Fritz.Geraetesperre: registered 1 subscription and 0 schedules
                

                Die Einstellungen im Script sehen so aus.

                /*******************************************************
                * E I N S T E L L U N G E N 
                *******************************************************/
                 
                // Die IP Der Fritzbox
                const FbIp        = "10.0.1.1";              
                // Der User der Fritzbox 
                // HINWEIS : Den Benutzer leer lassen wenn die Anmeldung an der FB nur mit Passwort erfolgt !
                const FbUser      = "xxxxxxxx"; 
                // Das Password der Fritzbox                  
                const FbPassword  = "xxxxxxxx";   
                // welcher Rechner soll "bearbeitet" werden                
                var   FbDevice    = "Comp-L-Pi";   
                // Das neue Profil für den Rechner              
                var   FbProfile   = "filtprof1";  
                // Keine Änderung an der Fritzbox (true) -> Listet dann nur alle Profile und Rechner               
                const FbListOnly  = true;  
                // Datenpunkt (string) für einen automatischen Script Start (muss angelegt werden!)
                // Der Datenpunkt wir mit Device;Profil beschrieben. Bsp: Comp-L-Pi;filtprof1
                const FbIobObject = "0_userdata.0.Fritzbox.ChangeDeviceFilter" 
                

                ° Node.js: 20.17.0 NPM: 10.8.2
                ° Proxmox, Ubuntu 22.04.3 LTS
                ° Fixer ---> iob fix

                moelskiM 1 Antwort Letzte Antwort
                0
                • NegaleinN Negalein

                  @moelski

                  Ich bekomm im Log nur den normalen Start angezeigt.

                  21:31:34.030	info	javascript.0 (595) Stop script script.js.Fritz.Geraetesperre
                  21:31:34.183	info	javascript.0 (595) Start javascript script.js.Fritz.Geraetesperre
                  21:31:34.188	info	javascript.0 (595) script.js.Fritz.Geraetesperre: registered 1 subscription and 0 schedules
                  

                  Die Einstellungen im Script sehen so aus.

                  /*******************************************************
                  * E I N S T E L L U N G E N 
                  *******************************************************/
                   
                  // Die IP Der Fritzbox
                  const FbIp        = "10.0.1.1";              
                  // Der User der Fritzbox 
                  // HINWEIS : Den Benutzer leer lassen wenn die Anmeldung an der FB nur mit Passwort erfolgt !
                  const FbUser      = "xxxxxxxx"; 
                  // Das Password der Fritzbox                  
                  const FbPassword  = "xxxxxxxx";   
                  // welcher Rechner soll "bearbeitet" werden                
                  var   FbDevice    = "Comp-L-Pi";   
                  // Das neue Profil für den Rechner              
                  var   FbProfile   = "filtprof1";  
                  // Keine Änderung an der Fritzbox (true) -> Listet dann nur alle Profile und Rechner               
                  const FbListOnly  = true;  
                  // Datenpunkt (string) für einen automatischen Script Start (muss angelegt werden!)
                  // Der Datenpunkt wir mit Device;Profil beschrieben. Bsp: Comp-L-Pi;filtprof1
                  const FbIobObject = "0_userdata.0.Fritzbox.ChangeDeviceFilter" 
                  
                  moelskiM Offline
                  moelskiM Offline
                  moelski
                  schrieb am zuletzt editiert von
                  #42

                  @Negalein
                  Soweit auch alles ok.
                  Du hast diesen Part aktiv:

                  on({id: FbIobObject, change: "ne"}, function (obj) {
                  .....
                  

                  Bedeutet das Script läuft nur los wenn dein Datenpunkt "0_userdata.0.Fritzbox.ChangeDeviceFilter" sich ändert.

                  Mach folgendes ...
                  Geh in die Objekte und trag bei deinem Datenpunkt folgendes ein:
                  test;test

                  Damit läuft das Script los und listet dir alle Geräte und alle Filter.
                  Und damit kannst du dann die Geräte sperren oder freigeben indem du dann in den Datenpunkt korrekte Werte schreibst:
                  In den Datenpunkt muss halt immer folgendes: <Computername>;<Profil ID> getrennt mit Semikolon.

                  Da bei dir FbListOnly aber noch auf true steht werden eh nur die Listen ausgegeben.
                  Wenn du dann reale Daten in den Datenpunkt schreibst musst du das noch auf false setzen.

                  Grüße Dominik

                  O 1 Antwort Letzte Antwort
                  0
                  • moelskiM moelski

                    @Negalein
                    Soweit auch alles ok.
                    Du hast diesen Part aktiv:

                    on({id: FbIobObject, change: "ne"}, function (obj) {
                    .....
                    

                    Bedeutet das Script läuft nur los wenn dein Datenpunkt "0_userdata.0.Fritzbox.ChangeDeviceFilter" sich ändert.

                    Mach folgendes ...
                    Geh in die Objekte und trag bei deinem Datenpunkt folgendes ein:
                    test;test

                    Damit läuft das Script los und listet dir alle Geräte und alle Filter.
                    Und damit kannst du dann die Geräte sperren oder freigeben indem du dann in den Datenpunkt korrekte Werte schreibst:
                    In den Datenpunkt muss halt immer folgendes: <Computername>;<Profil ID> getrennt mit Semikolon.

                    Da bei dir FbListOnly aber noch auf true steht werden eh nur die Listen ausgegeben.
                    Wenn du dann reale Daten in den Datenpunkt schreibst musst du das noch auf false setzen.

                    O Offline
                    O Offline
                    ottokar
                    schrieb am zuletzt editiert von
                    #43

                    @moelski @Negalein

                    Ja ich hab mich auch gefreut das Alexa endlich das tablet ein und ausschaltet(dacht ich) aber die List onlyfunktion war noch an...

                    Äh ich hab auch noch eine Noob Frage, ich mach zwar schon seit Jahren mit IOB rum aber Javascripts setzt ich eher selten ein,
                    hab noch die alte Version, der schaltet mir regelmäßig das Tablet aus wenn IOB neu startet.
                    ich hab das auch irgendwann mal gelesen das halt das script bei neustart getrigggert wird.
                    Wie kann ich das denn nochmal verhindern,
                    Aktuell hab ich nen Blockly das die den Alexa history DatenPunkt auf "Tablet aus" abfragt und dann script.enable auf wahr setzt ...
                    kann ich das Script dann über den Datenpunkt dann starten , ist das dafür gedacht?

                    Oli

                    moelskiM HomoranH 2 Antworten Letzte Antwort
                    0
                    • O ottokar

                      @moelski @Negalein

                      Ja ich hab mich auch gefreut das Alexa endlich das tablet ein und ausschaltet(dacht ich) aber die List onlyfunktion war noch an...

                      Äh ich hab auch noch eine Noob Frage, ich mach zwar schon seit Jahren mit IOB rum aber Javascripts setzt ich eher selten ein,
                      hab noch die alte Version, der schaltet mir regelmäßig das Tablet aus wenn IOB neu startet.
                      ich hab das auch irgendwann mal gelesen das halt das script bei neustart getrigggert wird.
                      Wie kann ich das denn nochmal verhindern,
                      Aktuell hab ich nen Blockly das die den Alexa history DatenPunkt auf "Tablet aus" abfragt und dann script.enable auf wahr setzt ...
                      kann ich das Script dann über den Datenpunkt dann starten , ist das dafür gedacht?

                      Oli

                      moelskiM Offline
                      moelskiM Offline
                      moelski
                      schrieb am zuletzt editiert von moelski
                      #44

                      Oha, da müssen wir noch etwas weiter ausholen, wie :grinning:

                      Also, im Script gibt es zwei Stellen und nur eine davon sollte aktiv sein.

                      Variante 1:

                      console.log("Profile Changer started ... (Version : " + Version + ")");
                      getFbChallenge();
                      console.log("Profile Changer done");
                      

                      Wenn das nicht auskommentiert ist, dann startet das Script immer durch wenn das Script anläuft.
                      Eigentlich arg unschön, weil das startet dann auch wenn der JavaScript Adapter neu startet.
                      Im Grunde ist dieses Stück Code eher zum Testen als für den produktiven Einsatz.
                      Also das mal am besten auskommentieren :-)

                      Variante 2:

                      on({id: FbIobObject, change: "ne"}, function (obj) {
                          console.log("Profile Changer started ... (Version : " + Version + ")");
                          var data = obj.state.val.split(";");
                          if (data.length != 2) {
                              console.log("Wrong parameter : " + obj.state.val);
                              return;
                          }
                       
                          console.log("Computer : " + data[0]);
                          console.log("Filter   : " + data[1]);
                          FbDevice  = data[0];
                          FbProfile = data[1];
                       
                          getFbChallenge();
                          
                          console.log("Profile Changer done");
                      });
                      

                      Hier wir das Script immer dann ausgeführt wenn sich im eingestellten Datenpunkt FbIobObject etwas ändert.
                      Heißt sobald in dem Datenpunkt ein neuer Wert kommt mit <Computername>;<FilterID>, dann decodiert das Script die beiden Werte und schiebt sie durch das folgende Script.
                      Diese Variante solltet ihr produktiv einsetzen :-)

                      Damit muss man auch nicht irgendwie das Script über einen Datenpunkt explizit anstarten.
                      Einfach nur in den eingestellten Datenpunkt die beiden Werte mit Semikolon dazwischen schreiben und der Rest geht von selbst.

                      Und diese Variante startet auch nicht einfach unkontrolliert wenn z.B. der JavaScript Adapter durchstartet.

                      @ottokar said in Internet Filter in der Fritzbox mit JavaScript setzen:

                      hab noch die alte Version,

                      Ich würde auf den letzten Stand wechseln. Weil viel einfacher und alle bekannten Bugs beseitigt ;-)

                      @ottokar said in Internet Filter in der Fritzbox mit JavaScript setzen:

                      Aktuell hab ich nen Blockly das die den Alexa history DatenPunkt auf "Tablet aus" abfragt und dann script.enable auf wahr setzt ...

                      Ullalla ... Sowas kompliziertes hab ich mir noch nie ausgedacht :joy:

                      Grüße Dominik

                      O 1 Antwort Letzte Antwort
                      0
                      • moelskiM moelski

                        @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                        das heisst für jedes Gerät 2 Scripte (Profil "gesperrt" & Profil "frei")?

                        Also ich habe in meiner Visu ein paar Button ala
                        "Kind sperren"
                        "Kind darf"

                        Diese Button setzen dann den Datenpunkt den man im Script definieren kann.
                        Wie das geht habe ich ja im Script auch beschrieben.

                        Und wenn dann der datenpunkt beschrieben wird, dann führt das Script die Aktion mit dem Rechner und dem Profil aus.
                        Also hast du das Script nur 1x am laufen.

                        Wenn du es statisch lösen willst ... Ja dann müsste man 2x das Script haben.
                        Aber das würde ich wann immer es geht vermeiden :-)

                        Grüße

                        NegaleinN Offline
                        NegaleinN Offline
                        Negalein
                        Global Moderator
                        schrieb am zuletzt editiert von
                        #45

                        @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                        Also ich habe in meiner Visu ein paar Button ala
                        "Kind sperren"
                        "Kind darf"
                        Diese Button setzen dann den Datenpunkt den man im Script definieren kann.

                        nochmal ich :)

                        Im Script hab ich

                        // welcher Rechner soll "bearbeitet" werden                
                        var   FbDevice    = "user3847";   
                        // Das neue Profil für den Rechner              
                        var   FbProfile   = "filtprof5151";  
                        

                        filtprof5151 ist das gesperrte.

                        Im Widget übergebe ich user3847;filtprof5151 an den erstellten DP.
                        Gerät wird gesperrt
                        Übergebe ich user3847;filtprof3 ist es wieder frei.

                        var FbProfile = "filtprof5151"; muss aber trotzdem fix im Script stehen?

                        Und noch eine Frage:
                        sieht man es in den Widgets irgendwie, ob das Gerät gesperrt ist? Weißt du, ob es da eine Möglichkeit gibt?

                        ° Node.js: 20.17.0 NPM: 10.8.2
                        ° Proxmox, Ubuntu 22.04.3 LTS
                        ° Fixer ---> iob fix

                        moelskiM 1 Antwort Letzte Antwort
                        0
                        • moelskiM moelski

                          Oha, da müssen wir noch etwas weiter ausholen, wie :grinning:

                          Also, im Script gibt es zwei Stellen und nur eine davon sollte aktiv sein.

                          Variante 1:

                          console.log("Profile Changer started ... (Version : " + Version + ")");
                          getFbChallenge();
                          console.log("Profile Changer done");
                          

                          Wenn das nicht auskommentiert ist, dann startet das Script immer durch wenn das Script anläuft.
                          Eigentlich arg unschön, weil das startet dann auch wenn der JavaScript Adapter neu startet.
                          Im Grunde ist dieses Stück Code eher zum Testen als für den produktiven Einsatz.
                          Also das mal am besten auskommentieren :-)

                          Variante 2:

                          on({id: FbIobObject, change: "ne"}, function (obj) {
                              console.log("Profile Changer started ... (Version : " + Version + ")");
                              var data = obj.state.val.split(";");
                              if (data.length != 2) {
                                  console.log("Wrong parameter : " + obj.state.val);
                                  return;
                              }
                           
                              console.log("Computer : " + data[0]);
                              console.log("Filter   : " + data[1]);
                              FbDevice  = data[0];
                              FbProfile = data[1];
                           
                              getFbChallenge();
                              
                              console.log("Profile Changer done");
                          });
                          

                          Hier wir das Script immer dann ausgeführt wenn sich im eingestellten Datenpunkt FbIobObject etwas ändert.
                          Heißt sobald in dem Datenpunkt ein neuer Wert kommt mit <Computername>;<FilterID>, dann decodiert das Script die beiden Werte und schiebt sie durch das folgende Script.
                          Diese Variante solltet ihr produktiv einsetzen :-)

                          Damit muss man auch nicht irgendwie das Script über einen Datenpunkt explizit anstarten.
                          Einfach nur in den eingestellten Datenpunkt die beiden Werte mit Semikolon dazwischen schreiben und der Rest geht von selbst.

                          Und diese Variante startet auch nicht einfach unkontrolliert wenn z.B. der JavaScript Adapter durchstartet.

                          @ottokar said in Internet Filter in der Fritzbox mit JavaScript setzen:

                          hab noch die alte Version,

                          Ich würde auf den letzten Stand wechseln. Weil viel einfacher und alle bekannten Bugs beseitigt ;-)

                          @ottokar said in Internet Filter in der Fritzbox mit JavaScript setzen:

                          Aktuell hab ich nen Blockly das die den Alexa history DatenPunkt auf "Tablet aus" abfragt und dann script.enable auf wahr setzt ...

                          Ullalla ... Sowas kompliziertes hab ich mir noch nie ausgedacht :joy:

                          O Offline
                          O Offline
                          ottokar
                          schrieb am zuletzt editiert von
                          #46

                          @moelski

                          das mit dem History ist echt genial du hast nur im IOT Adapter ein Fake Gerät (bei mir heißt das halt Trigger) das in den Alexa Routinen angegeben wird damit die Tante auch was tut , weil schalte auf Pro7 oder so kommt dann halt kenn ich nicht weiß ich nicht..

                          Ich bau dann mal das neue Script ein und schau wie ich das hinbekomm..

                          Danke
                          Oli

                          1 Antwort Letzte Antwort
                          0
                          • O ottokar

                            @moelski @Negalein

                            Ja ich hab mich auch gefreut das Alexa endlich das tablet ein und ausschaltet(dacht ich) aber die List onlyfunktion war noch an...

                            Äh ich hab auch noch eine Noob Frage, ich mach zwar schon seit Jahren mit IOB rum aber Javascripts setzt ich eher selten ein,
                            hab noch die alte Version, der schaltet mir regelmäßig das Tablet aus wenn IOB neu startet.
                            ich hab das auch irgendwann mal gelesen das halt das script bei neustart getrigggert wird.
                            Wie kann ich das denn nochmal verhindern,
                            Aktuell hab ich nen Blockly das die den Alexa history DatenPunkt auf "Tablet aus" abfragt und dann script.enable auf wahr setzt ...
                            kann ich das Script dann über den Datenpunkt dann starten , ist das dafür gedacht?

                            Oli

                            HomoranH Nicht stören
                            HomoranH Nicht stören
                            Homoran
                            Global Moderator Administrators
                            schrieb am zuletzt editiert von Homoran
                            #47

                            @ottokar sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                            und dann script.enable auf wahr setzt ...
                            kann ich das Script dann über den Datenpunkt dann starten , ist das dafür gedacht?

                            Nein, scriptenabled soll nicht zur Steuerung verwendet werden.
                            Ein Skript soll immer laufen, und die Logik über einen Trigger starten

                            kein Support per PN! - Fragen im Forum stellen - es gibt fast nichts, was nicht auch für andere interessant ist.

                            Benutzt das Voting rechts unten im Beitrag wenn er euch geholfen hat.

                            der Installationsfixer: curl -fsL https://iobroker.net/fix.sh | bash -

                            1 Antwort Letzte Antwort
                            0
                            • NegaleinN Negalein

                              @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                              Also ich habe in meiner Visu ein paar Button ala
                              "Kind sperren"
                              "Kind darf"
                              Diese Button setzen dann den Datenpunkt den man im Script definieren kann.

                              nochmal ich :)

                              Im Script hab ich

                              // welcher Rechner soll "bearbeitet" werden                
                              var   FbDevice    = "user3847";   
                              // Das neue Profil für den Rechner              
                              var   FbProfile   = "filtprof5151";  
                              

                              filtprof5151 ist das gesperrte.

                              Im Widget übergebe ich user3847;filtprof5151 an den erstellten DP.
                              Gerät wird gesperrt
                              Übergebe ich user3847;filtprof3 ist es wieder frei.

                              var FbProfile = "filtprof5151"; muss aber trotzdem fix im Script stehen?

                              Und noch eine Frage:
                              sieht man es in den Widgets irgendwie, ob das Gerät gesperrt ist? Weißt du, ob es da eine Möglichkeit gibt?

                              moelskiM Offline
                              moelskiM Offline
                              moelski
                              schrieb am zuletzt editiert von moelski
                              #48

                              @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                              var FbProfile = "filtprof5151"; muss aber trotzdem fix im Script stehen?

                              Nein.
                              Wie man hier sehen kann wird das mit den Daten aus dem Datenpunkt überschrieben:

                                  console.log("Computer : " + data[0]);
                                  console.log("Filter   : " + data[1]);
                                  
                                  FbDevice  = data[0];
                                  FbProfile = data[1];**
                              

                              @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                              sieht man es in den Widgets irgendwie, ob das Gerät gesperrt ist?

                              Nein.
                              Das müsste man extra raus filtern.

                              @Homoran said in Internet Filter in der Fritzbox mit JavaScript setzen:

                              Nein, scriptenabled soll nicht zur Steuerung verwendet werden.
                              Ein Skript soll immer laufen, und die Logik über einen Trigger starten

                              u made my day :-)

                              Grüße Dominik

                              NegaleinN 2 Antworten Letzte Antwort
                              0
                              • moelskiM moelski

                                @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                var FbProfile = "filtprof5151"; muss aber trotzdem fix im Script stehen?

                                Nein.
                                Wie man hier sehen kann wird das mit den Daten aus dem Datenpunkt überschrieben:

                                    console.log("Computer : " + data[0]);
                                    console.log("Filter   : " + data[1]);
                                    
                                    FbDevice  = data[0];
                                    FbProfile = data[1];**
                                

                                @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                sieht man es in den Widgets irgendwie, ob das Gerät gesperrt ist?

                                Nein.
                                Das müsste man extra raus filtern.

                                @Homoran said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                Nein, scriptenabled soll nicht zur Steuerung verwendet werden.
                                Ein Skript soll immer laufen, und die Logik über einen Trigger starten

                                u made my day :-)

                                NegaleinN Offline
                                NegaleinN Offline
                                Negalein
                                Global Moderator
                                schrieb am zuletzt editiert von
                                #49

                                @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                Wie man hier sehen kann wird das mit den Daten aus dem Datenpunkt überschrieben:

                                Danke

                                Kannst du mir mal deine Buttons exportieren?

                                ° Node.js: 20.17.0 NPM: 10.8.2
                                ° Proxmox, Ubuntu 22.04.3 LTS
                                ° Fixer ---> iob fix

                                moelskiM 1 Antwort Letzte Antwort
                                0
                                • moelskiM moelski

                                  @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                  var FbProfile = "filtprof5151"; muss aber trotzdem fix im Script stehen?

                                  Nein.
                                  Wie man hier sehen kann wird das mit den Daten aus dem Datenpunkt überschrieben:

                                      console.log("Computer : " + data[0]);
                                      console.log("Filter   : " + data[1]);
                                      
                                      FbDevice  = data[0];
                                      FbProfile = data[1];**
                                  

                                  @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                  sieht man es in den Widgets irgendwie, ob das Gerät gesperrt ist?

                                  Nein.
                                  Das müsste man extra raus filtern.

                                  @Homoran said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                  Nein, scriptenabled soll nicht zur Steuerung verwendet werden.
                                  Ein Skript soll immer laufen, und die Logik über einen Trigger starten

                                  u made my day :-)

                                  NegaleinN Offline
                                  NegaleinN Offline
                                  Negalein
                                  Global Moderator
                                  schrieb am zuletzt editiert von
                                  #50

                                  @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                  Nein.
                                  Das müsste man extra raus filtern.

                                  schade. Aber hauptsache es geht jetzt einfach zu sperren

                                  ° Node.js: 20.17.0 NPM: 10.8.2
                                  ° Proxmox, Ubuntu 22.04.3 LTS
                                  ° Fixer ---> iob fix

                                  moelskiM 1 Antwort Letzte Antwort
                                  0
                                  • NegaleinN Negalein

                                    @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                    Wie man hier sehen kann wird das mit den Daten aus dem Datenpunkt überschrieben:

                                    Danke

                                    Kannst du mir mal deine Buttons exportieren?

                                    moelskiM Offline
                                    moelskiM Offline
                                    moelski
                                    schrieb am zuletzt editiert von
                                    #51

                                    @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                    Kannst du mir mal deine Buttons exportieren?

                                    [{"tpl":"tplIconState","data":{"oid":"Global.0.Fritzbox.ChangeDeviceFilter","g_fixed":false,"g_visibility":false,"g_css_font_text":false,"g_css_background":true,"g_css_shadow_padding":false,"g_css_border":true,"g_gestures":false,"g_signals":false,"g_last_change":false,"visibility-cond":"==","visibility-val":1,"visibility-groups-action":"hide","signals-cond-0":"==","signals-val-0":true,"signals-icon-0":"/vis/signals/lowbattery.png","signals-icon-size-0":0,"signals-blink-0":false,"signals-horz-0":0,"signals-vert-0":0,"signals-hide-edit-0":false,"signals-cond-1":"==","signals-val-1":true,"signals-icon-1":"/vis/signals/lowbattery.png","signals-icon-size-1":0,"signals-blink-1":false,"signals-horz-1":0,"signals-vert-1":0,"signals-hide-edit-1":false,"signals-cond-2":"==","signals-val-2":true,"signals-icon-2":"/vis/signals/lowbattery.png","signals-icon-size-2":0,"signals-blink-2":false,"signals-horz-2":0,"signals-vert-2":0,"signals-hide-edit-2":false,"lc-type":"last-change","lc-is-interval":true,"lc-is-moment":false,"lc-format":"","lc-position-vert":"top","lc-position-horz":"right","lc-offset-vert":0,"lc-offset-horz":0,"lc-font-size":"12px","lc-font-family":"","lc-font-style":"","lc-bkg-color":"","lc-color":"","lc-border-width":"0","lc-border-style":"","lc-border-color":"","lc-border-radius":10,"lc-zindex":0,"src":"/icons-mfd-png/control_x.png","value":"rechnerabc;filterid"},"style":{"left":"208px","top":"508px","background":"#3c5977","background-color":"#3c5977","border-style":"none","border-radius":"0px","width":"62px","height":"58px","z-index":"3"},"widgetSet":"jqui"}]
                                    

                                    So in etwa :-)

                                    Grüße Dominik

                                    1 Antwort Letzte Antwort
                                    0
                                    • NegaleinN Negalein

                                      @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                      Nein.
                                      Das müsste man extra raus filtern.

                                      schade. Aber hauptsache es geht jetzt einfach zu sperren

                                      moelskiM Offline
                                      moelskiM Offline
                                      moelski
                                      schrieb am zuletzt editiert von
                                      #52

                                      @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                      schade. Aber hauptsache es geht jetzt einfach zu sperren

                                      Das wäre aber nicht so kompliziert ...
                                      Das könnte ein weiterer Aufruf im Script für alle Devices ermitteln welche die FB kennt.
                                      Und das könnte man sicherlich auch in einen JSON Datenpunkt wegschreiben zwecks Auswertung.

                                      Ich packe das mal auf die Todo.
                                      Das wäre nämlich ne interessante Funktion für meine Frau :-D

                                      Grüße Dominik

                                      NegaleinN 1 Antwort Letzte Antwort
                                      0
                                      • moelskiM moelski

                                        @Negalein said in Internet Filter in der Fritzbox mit JavaScript setzen:

                                        schade. Aber hauptsache es geht jetzt einfach zu sperren

                                        Das wäre aber nicht so kompliziert ...
                                        Das könnte ein weiterer Aufruf im Script für alle Devices ermitteln welche die FB kennt.
                                        Und das könnte man sicherlich auch in einen JSON Datenpunkt wegschreiben zwecks Auswertung.

                                        Ich packe das mal auf die Todo.
                                        Das wäre nämlich ne interessante Funktion für meine Frau :-D

                                        NegaleinN Offline
                                        NegaleinN Offline
                                        Negalein
                                        Global Moderator
                                        schrieb am zuletzt editiert von
                                        #53

                                        @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                        Ich packe das mal auf die Todo.

                                        perfekt

                                        Das wäre nämlich ne interessante Funktion für meine Frau

                                        Ja, so eine hab ich auch daheim :D

                                        ° Node.js: 20.17.0 NPM: 10.8.2
                                        ° Proxmox, Ubuntu 22.04.3 LTS
                                        ° Fixer ---> iob fix

                                        moelskiM 1 Antwort Letzte Antwort
                                        0
                                        • NegaleinN Negalein

                                          @moelski sagte in Internet Filter in der Fritzbox mit JavaScript setzen:

                                          Ich packe das mal auf die Todo.

                                          perfekt

                                          Das wäre nämlich ne interessante Funktion für meine Frau

                                          Ja, so eine hab ich auch daheim :D

                                          moelskiM Offline
                                          moelskiM Offline
                                          moelski
                                          schrieb am zuletzt editiert von moelski
                                          #54

                                          @Negalein

                                          Ich habe mal kurz F12 wieder genutzt um mir die Daten anzusehen ....

                                          <tr>
                                          	<td class="name" title="Comp-L-Pi" datalabel="Comp-L-Pi">
                                          		<span>Comp-L-Pi</span>
                                          	</td>
                                          	<td datalabel="Internetnutzung" class="usage">unbeschränkt</td>
                                          	<td datalabel="Onlinezeit heute" class="bar time">
                                          		<span title="00:00 von 24:00 Stunden">
                                          			<span style="width:0%;"/>
                                          		</span>
                                          	</td>
                                          	<td datalabel="Zugangsprofil" class="profile">
                                          		<select name="profile:landevice37174">
                                          			<option value="filtprof1" selected>Standard</option>
                                          			<option value="filtprof3">Unbeschränkt</option>
                                          			<option value="filtprof4">Gesperrt</option>
                                          			<option value="filtprof1118">Lennard ohne Youtube</option>
                                          			<option value="filtprof3079">Lennard Tablet</option>
                                          			<option value="filtprof4042">Test-Whi_te</option>
                                          			<option value="filtprof3827">TV ohne YT</option>
                                          		</select>
                                          	</td>
                                          	<td datalabel="" class="btncolumn">
                                          		<button type="submit" name="edit" id="uiEdit:landevice37174" value="filtprof1" class="icon edit" title="Bearbeiten"/>
                                          	</td>
                                          </tr>
                                          

                                          Also mit ein bissel Regex kriege ich da alle Infos zusammen :male-astronaut:

                                          Grüße Dominik

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


                                          Support us

                                          ioBroker
                                          Community Adapters
                                          Donate

                                          487

                                          Online

                                          32.4k

                                          Benutzer

                                          81.4k

                                          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