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.2k

  • 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.
  • O ottokar

    @moelski

    Ja einer hat Bindestrich einer Unterstrich...

    Entfernt, geht... Reihenfolge scheint auch zu passen..

    Kannst du - und _ vielleicht noch in die Regex packen, ich denke das ist nicht zuuu selten..

    Oli

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

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

    Kannst du - und _ vielleicht noch in die Regex packen, ich denke das ist nicht zuuu selten..

    Zeile 119:

    var rx = new RegExp( /class=\"name\"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_]*)\"\sdatalabel/g );
    

    Gleich unter " console.log(" > Decode Names")"

    Grüße Dominik

    O 1 Antwort Letzte Antwort
    0
    • moelskiM moelski

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

      Kannst du - und _ vielleicht noch in die Regex packen, ich denke das ist nicht zuuu selten..

      Zeile 119:

      var rx = new RegExp( /class=\"name\"\stitle=\"([a-zA-Z0-9 äöüÄÖÜ\-\_]*)\"\sdatalabel/g );
      

      Gleich unter " console.log(" > Decode Names")"

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

      @moelski

      Eingetragen und getestet,
      Funktioniert...

      Oli

      moelskiM 1 Antwort Letzte Antwort
      0
      • O ottokar

        @moelski

        Eingetragen und getestet,
        Funktioniert...

        Oli

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

        @ottokar
        Sehr schön, dann mal schauen was die beiden anderen Experten so schreiben :-)

        Grüße Dominik

        ? 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
          #28

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

          Hier aber nun das Script:

          Hallo

          Ist das Script das aktuelle?

          ° 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:

            Hallo

            Ist das Script das aktuelle?

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

            @Negalein
            Ich habe das Script angepasst damit die Regex mit - und _ klar kommen.
            Den Part mit dem User + Passwort musst du dir im Moment aus den Infos von ottokar ruassuchen.
            Das war aber auch nur eine kleine Ergänzung.

            sonst hat sich noch nichts geändert.

            Grüße Dominik

            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 ...

              F Offline
              F Offline
              fastfoot
              schrieb am zuletzt editiert von
              #30

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

              Ok jetzt bin ich gespannt auf das Feedback

              Funktioniert nach der Änderung mit FbUser einwandfrei auf einer 7590

              iobroker läuft unter Docker auf QNAP TS-451+
              SkriptRecovery: https://forum.iobroker.net/post/930558

              moelskiM 1 Antwort Letzte Antwort
              0
              • F fastfoot

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

                Ok jetzt bin ich gespannt auf das Feedback

                Funktioniert nach der Änderung mit FbUser einwandfrei auf einer 7590

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

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

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

                Ok jetzt bin ich gespannt auf das Feedback

                Funktioniert nach der Änderung mit FbUser einwandfrei auf einer 7590

                Sehr schön.
                Dann werde ich den Part noch ergänzen und auch die Beschreibung anpassen.

                Und dann schreibe ich evtl. noch ein paar Sätze wie man das über die VIS dann steuern könnte ...

                Grüße Dominik

                NegaleinN 1 Antwort Letzte Antwort
                0
                • moelskiM moelski

                  @ottokar
                  Sehr schön, dann mal schauen was die beiden anderen Experten so schreiben :-)

                  ? Offline
                  ? Offline
                  Ein ehemaliger Benutzer
                  schrieb am zuletzt editiert von
                  #32

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

                  @ottokar
                  Sehr schön, dann mal schauen was die beiden anderen Experten so schreiben :-)

                  Also meiner Einer meint, es funktioniert jetzt einwandfrei! Super!

                  Allen anderen Experten vielen Dank :-)

                  Jetzt braucht man nur noch Datenpunkte anzulegen und einzubinden für das Device und den Filter, um das ganze zu automatisieren..
                  oder man legt mehrere Scripte an, das ist aber unschön :-)

                  1 Antwort Letzte Antwort
                  0
                  • moelskiM moelski

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

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

                    Ok jetzt bin ich gespannt auf das Feedback

                    Funktioniert nach der Änderung mit FbUser einwandfrei auf einer 7590

                    Sehr schön.
                    Dann werde ich den Part noch ergänzen und auch die Beschreibung anpassen.

                    Und dann schreibe ich evtl. noch ein paar Sätze wie man das über die VIS dann steuern könnte ...

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

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

                    Sehr schön.
                    Dann werde ich den Part noch ergänzen und auch die Beschreibung anpassen.
                    Und dann schreibe ich evtl. noch ein paar Sätze wie man das über die VIS dann steuern könnte ...

                    sehr gut, dann warte ich noch :)

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

                    1 Antwort Letzte Antwort
                    0
                    • ? Ein ehemaliger Benutzer

                      @moelski
                      Hi,
                      super, vielen Dank! Glaub ich dir, dass da sehr viel Arbeit drin steckt, wenn ich mir das Script ansehe!

                      Es gibt den fb-checkpresence-Adapter, dort wurden schon viele solcher Anfragen gestellt, ich hab mal n Querverweis gemacht, vielleicht kann sich der Programmierer @afuerhoff sich bei dir ein paar Teile von dem Script holen und dort in seinen Adapter einbauen??

                      Aber das müsst ihr untereinander ausmachen :-)
                      Ich selbst komm auch mit dem Javascript zurecht :-)

                      afuerhoffA Offline
                      afuerhoffA Offline
                      afuerhoff
                      Developer
                      schrieb am zuletzt editiert von afuerhoff
                      #34

                      @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.

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

                      moelskiM 1 Antwort Letzte Antwort
                      0
                      • 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
                                          Antworten
                                          • In einem neuen Thema antworten
                                          Anmelden zum Antworten
                                          • Älteste zuerst
                                          • Neuste zuerst
                                          • Meiste Stimmen


                                          Support us

                                          ioBroker
                                          Community Adapters
                                          Donate

                                          631

                                          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