Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Skripten / Logik
    4. Blockly
    5. [gelöst] Skript läuft nicht... Lüften oder nicht...

    NEWS

    • Wir empfehlen: Node.js 22.x

    • Neuer Blog: Fotos und Eindrücke aus Solingen

    • ioBroker goes Matter ... Matter Adapter in Stable

    [gelöst] Skript läuft nicht... Lüften oder nicht...

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

      @David-Froebus sagte:

      Außen-Luftfeuchtigkeit - 3 * (Innentemperatur - Außentemperatur) => Zu erwartende Innen-Luftfeuchtigkeit.

      Dieser einfache Zusammenhang gilt nur in einem begrenzten Temperaturbereich.

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

        @paul53 Vielen Dank! Das sieht sehr gut aus!

        1 Reply Last reply Reply Quote 0
        • D
          Dragon last edited by

           
          // Berechnet Taupunkt, absolute Luftfeuchtigkeit, Enthalpie, Lüftungsempfehlung, 
          //gemessene Temperatur & Luftfeuctigkeit inkl. Offset zwecks Kalibrierung
          // -----------------------------------------------------------------------------
          // von paul53 übernommen und angepasst
          // http://forum.iobroker.net/viewtopic.php?f=20&t=2437&hilit=L%C3%BCftung%2A#p21476
          // und Solear: http://forum.iobroker.net/viewtopic.php?f=21&t=2645&p=23381#p23282
           
           
          var nn          = 68;                // eigene Höhe nn (normalnull), z.B. über http://de.mygeoposition.com zu ermitteln
          var pfad        = "Raumklima"   +".";   // Pfad unter dem die Datenpunkte in der Javascript-Instanz angelegt werden
          var controlPfad = "CONTROL"     +".";   // Pfad innerhalb des Raums
          var skriptConf  = true;                 // true: Raumwerte werden über das Skript geändert / false: Raumwerte werden über Objekte (oder VIS) geändert
          var defaultTemp = 21.75;                // Default TEMP_Zielwert, wenn im Raum nicht angegeben
           
           
          // -----------------------------------------------------------------------------
          // Räume mit Sensoren
          // -----------------------------------------------------------------------------
          var raeume = { // Keine Leerzeichen (Name wird als Datenpunktname verwendet!)
              // Sensoren Aussen
              "Terrasse" : {
                  "Sensor_TEMP"           :   "zigbee.0.00158d00044484a0.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "zigbee.0.00158d00044484a0.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
              },
              // Sensoren Innen
              "Badezimmer" : {
                  "Sensor_TEMP"           :   "ble.0.58:2d:34:37:a7:42.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "ble.0.58:2d:34:37:a7:42.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
                  "TEMP_Zielwert"         :   21.75,
                  "Aussensensor"          :   "Terrasse"
              },    
              "Wohnzimmer" : {
                  "Sensor_TEMP"           :   "ble.0.58:2d:34:37:ad:0a.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "ble.0.58:2d:34:37:ad:0a.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
                  "TEMP_Zielwert"         :   21.75,
                  "Aussensensor"          :   "Terrasse"
              },
              "Schlafzimmer" : {
                  "Sensor_TEMP"           :   "ble.0.58:2d:34:37:aa:c6.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "ble.0.58:2d:34:37:aa:c6.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
                  "TEMP_Zielwert"         :   21.75,
                  "Aussensensor"          :   "Terrasse"
              },
              "Kueche" : {
                  "Sensor_TEMP"           :   "zigbee.0.00158d00044500f2.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "zigbee.0.00158d00044500f2.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
                  "TEMP_Zielwert"         :   21.75,
                  "Aussensensor"          :   "Terrasse"
              },
               "Flur" : {
                  "Sensor_TEMP"           :   "ble.0.58:2d:34:37:ab:0d.temperature"/*Temperature*/,
                  "Sensor_HUM"            :   "ble.0.58:2d:34:37:ab:0d.humidity"/*Humidity*/,
                  "Sensor_TEMP_OFFSET"    :   0.0,
                  "Sensor_HUM_OFFSET"     :   0,
                  "TEMP_Zielwert"         :   21.75,
                  "Aussensensor"          :   "Terrasse"
              },
          };
           
           
           
           
          // =============================================================================
          // Skriptbereich. Ab hier muss nichts mehr eingestellt / verändert werden.
          // =============================================================================
          var raumDatenpunkte = {
              "x" : {
                  "DpName" : "Feuchtegehalt_Absolut",
                  "init": 0,
                  "dp": {
                      "name": 'absoluter Feuchtegehalt',
                      "desc": 'absoluter Feuchtegehalt, errechnet',
                      "type": 'number',
                      "role": 'value',
                      "unit": 'g/kg'
                  }
              },
              "rh" : {
                  "DpName" : "relative_Luftfeuchtigkeit",
                  "init": 0,
                  "dp": {
                      "name": 'gemessene relative Luftfeuchtigkeit (inkl. Offset)',
                      "desc": 'relative Luftfeuchtigkeit, vom Sensor + Offset zum Ausgleich von Messungenauigkeiten des Geräts',
                      "type": 'number',
                      "role": 'value',
                      "unit": '%'
                  }
              },
              "dp" : {
                  "DpName" : "Taupunkt",
                  "init": 0,
                  "dp": {
                      "name": 'Taupunkt',
                      "desc": 'Taupunkt. Temperatur von Wänden, Fenstern, usw. ab der sich die Feuchtigkeit niederschlägt.',
                      "type": 'number',
                      "role": 'value',
                      "unit": '°C'
                  }
              },
              "t" : {
                  "DpName" : "Temperatur",
                  "init": 0,
                  "dp": {
                      "name": 'gemessene Temperatur (inkl. Offset)',
                      "desc": 'gemessene Temperatur vom Sensor zzgl. eines Offsets um Geräteungenauigkeiten auszugleichen',
                      "type": 'number',
                      "role": 'value',
                      "unit": '°C'
                  }
              },
              "h" : {
                  "DpName" : "Enthalpie",
                  "init": 0,
                  "dp": {
                      "name": 'Enthalpie',
                      "desc": 'Enthalpie',
                      "type": 'number',
                      "role": 'value',
                      "unit": 'kJ/kg'
                  }
              },
              "lüften" : {
                  "DpName" : "Lüftungsempfehlung",
                  "init": false,
                  "dp": {
                      "name": 'Lüftungsempfehlung',
                      "desc": 'Lüftungsempfehlung',
                     // "type"  : 'logic',
                      //"role": 'value'
                  }
              }
          };
           
          var raumControl = {
              "Sensor_TEMP_OFFSET" : {
                  "DpName" : "Sensor_TEMP_OFFSET",
                  "init": 0,
                  "dp": {
                      "name": 'Offset Temperatur zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                      "desc": 'Offset Temperatur zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                      "type": 'number',
                      "role": 'control.value',
                      "unit": '°C'
                  }
              },
              "Sensor_HUM_OFFSET" : {
                  "DpName" : "Sensor_HUM_OFFSET",
                  "init": 0,
                  "dp": {
                      "name": 'Offset Luftfeuchtigkeit zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                      "desc": 'Offset Luftfeuchtigkeit zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                      "type": 'number',
                      "role": 'control.value',
                      "unit": '%'
                  }
              },
              "TEMP_Zielwert" : {
                  "DpName" : "TEMP_Zielwert",
                  "init": 0,
                  "dp": {
                      "name": 'Temperatursteuerwert zum lüften',
                      "desc": 'Temperatursteuerwert zum lüften',
                      "type": 'number',
                      "role": 'control.value',
                      "unit": '°C'
                  }
              },
              "Aussensensor" : {
                  "DpName" : "Aussensensor",
                  "init": "",
                  "dp": {
                      "name": 'Aussensensor, der zum Vergleich genommen wird',
                      "desc": 'Aussensensor, der zum Vergleich genommen wird',
                      "type": 'string',
                      "role": 'control.value'
                  }
              }
          };
           
          var  DP = require('dewpoint');                      // Das Modul dewpoint einlesen
          var xdp = new DP(nn);
           
           
           
          function createDp() {
              var name;
              var init;
              var forceCreation;
              var common;
              for (var raum in raeume) {
                  for (var datenpunktID in raumDatenpunkte) {
                      name = pfad + raum + "." + raumDatenpunkte[datenpunktID].DpName;
                      init = raumDatenpunkte[datenpunktID].init;
                      forceCreation = false; // Init der Datenpunkte wird nur beim ersten Star angelegt. Danach bleiben die Wert auch nach Skritpstart enthalten.
                      common = raumDatenpunkte[datenpunktID].dp;
                      createState(name, init , forceCreation, common);
                      log("neuer Datenpunkt: " + name,"debug");
                  }
                  for (var control in raumControl) {
                      name = pfad + raum + "." + controlPfad + raumControl[control].DpName;
                      //init = raumControl[control].init;
                      forceCreation = skriptConf;
                      common = raumControl[control].dp;
                      if (typeof raeume[raum][raumControl[control].DpName] !=="undefined") {
                          init = raeume[raum][raumControl[control].DpName];
                          createState(name, init , forceCreation, common);
                      }
                  }
              }
          } 
          function runden(val, digits) {
              return Math.round(val * Math.pow(10, digits)) / Math.pow(10, digits);
          }
          function calc(raum) {                                           // Über Modul Dewpoint absolute Feuchte berechnen
              var t           = getState(raeume[raum].Sensor_TEMP).val;   // Temperatur auslesen     
              var rh          = getState(raeume[raum].Sensor_HUM).val;    // Feuchtigkeit relativ auslesen 
              var y           = xdp.Calc(t, rh);                        
              var toffset     = 0.0;                                      // Offset in °C
              var rhoffset    = 0;                                        // Offset in %
              if(typeof raeume[raum].Sensor_TEMP_OFFSET !=="undefined") {
                  var idtoffset = pfad + raum + "." + controlPfad + "Sensor_TEMP_OFFSET";
                  toffset = getState(idtoffset).val;  // Offset aus den Objekten/Datenpunkt auslesen
              }
              if(typeof raeume[raum].Sensor_HUM_OFFSET !=="undefined") {
                  var idrhoffset = pfad + raum + "." + controlPfad + "Sensor_HUM_OFFSET";
                  rhoffset = getState(idrhoffset).val;  // Offset aus den Objekten/Datenpunkt auslesen
              }
           
              t       = t     + toffset;
              rh      = rh    + rhoffset;
           
              var x   =  y.x;                                 // Zu errechnende Variable für Feuchtegehalt in g/kg
              var dp  = y.dp;                                 // Zu errechnende Variable für Taupunkt in °C
           
              var h   = 1.00545 * t + (2.500827 + 0.00185894 * t) * x;
           
           
              var idx     = pfad + raum + "." + raumDatenpunkte["x"].DpName;
              var iddp    = pfad + raum + "." + raumDatenpunkte["dp"].DpName;
              var idt     = pfad + raum + "." + raumDatenpunkte["t"].DpName;
              var idrh    = pfad + raum + "." + raumDatenpunkte["rh"].DpName;
              var ih      = pfad + raum + "." + raumDatenpunkte["h"].DpName;
              
              setState(idx    , runden(x,2));     // errechnete absolute Feuchte in Variable schreiben
              setState(iddp   , runden(dp,1));    // errechneter Taupunkt in Variable schreiben
              setState(idt    , t);               // Sensor Temperatur        inkl. Offset
              setState(idrh   , rh);              // Sensor Relative Feuchte  inkl. Offset
              setState(ih     , runden(h,2));     // Enthalpie
           
              // Lüften
              if(typeof raeume[raum].Aussensensor !=="undefined") {
                  var aussen = raeume[raum].Aussensensor;
                  var idta = pfad + aussen + "." + raumDatenpunkte["t"].DpName;
                  var idxa = pfad + aussen + "." + raumDatenpunkte["rh"].DpName;
              } else {
                  return;
              }
           
              var ti = t;                     // Raumtemperatur in °C
              var xi = rh;                    // Raumfeuchtegehalt in g/kg
              var ta = getState(idta).val;    // Aussentemperatur in °C
              var xa = getState(idxa).val;    // Aussenfeuchtegehalt in g/kg
              if (xa === 0) return; 
              
              var mi = defaultTemp;           // Temperaturmittelwert auf Default
           
              if(typeof raeume[raum].TEMP_Zielwert !=="undefined") {
                  mi = raeume[raum].TEMP_Zielwert;
              }
              var mih = mi + 0.25; // Temperaturmittelwert hoch
              var mit = mi - 0.25; // Temperaturmittelwert tief
           
              var idLueften = pfad + raum + "." + raumDatenpunkte["lüften"].DpName;
              
              // Lüftungsempfehlung steuern mit 0,3 g/kg und 0,5 K Hysterese
              if      (xa <= (xi - 0.4) && ta <= (ti - 0.6) && ti >= mih) setState(idLueften, true);
              else if (xa >= (xi - 0.1) || ta >= (ti - 0.1) || ti <= mit) setState(idLueften, false);
           
              log("Raum: " + raum+", ti:"+ti+", ta: "+ta+", xi:"+xi+", xa: "+xa+", mih:"+mih+", mit:"+mit,"debug");
          }
           
          function calcAll () {
              for (var raum in raeume) {
                  calc(raum);
              }
          }
           
          /*
          on({id: idSensorTemperaturBad   ,change:'ne'}, function (obj) {
              calc();
          });
          {1}
          on({id: idSensorLuftfeuchteBad  ,change:'ne'}, function (obj) {
              calc();
          });
          */
           
          // alle 30 Minuten neu berechnen
          schedule("30 * * * *", function () {
              calcAll();
          });
           
           
          function main() {
              calcAll();
              calcAll();
          }
           
          createDp();                 // Datenpunkte anlegen
          setTimeout(main,    500);   // Zum Skriptstart ausführen
           
           
           
           
           
          
          

          Folgende Fehlermeldung:

          javascript.0 2020-09-14 18:47:59.084 error (15655) at Script.runInContext (vm.js:130:18)
          javascript.0 2020-09-14 18:47:59.084 error (15655) at script.js.common.Komfort.Lüften:192:11
          javascript.0 2020-09-14 18:47:59.083 error (15655) TypeError: DP is not a constructor
          javascript.0 2020-09-14 18:47:59.083 error (15655) ^
          javascript.0 2020-09-14 18:47:59.083 error (15655) var xdp = new DP(nn);
          javascript.0 2020-09-14 18:47:59.082 error (15655) script.js.common.Komfort.Lüften: script.js.common.Komfort.Lüften:192
          javascript.0 2020-09-14 18:47:59.078 error (15655) at Script.runInContext (vm.js:130:18)
          javascript.0 2020-09-14 18:47:59.077 error (15655) at script.js.common.Komfort.Lüften:191:11
          javascript.0 2020-09-14 18:47:59.077 error (15655) at require (/opt/iobroker/node_modules/iobroker.javascript/lib/sandbox.js:390:36)
          javascript.0 2020-09-14 18:47:59.076 error (15655) at require (internal/modules/cjs/helpers.js:72:18)
          javascript.0 2020-09-14 18:47:59.076 error (15655) at Module.require (internal/modules/cjs/loader.js:1025:19)
          javascript.0 2020-09-14 18:47:59.075 error (15655) script.js.common.Komfort.Lüften: Error: Cannot find module '/opt/iobroker/node_modules/iobroker.javascript/lib/../../dewpoint'

          1 Reply Last reply Reply Quote 0
          • D
            Dragon last edited by Homoran

            zweites skript. eigentlich selbe funktion:

            //
            // Raumklima - v0.6.5
            //
            // Berechnet Taupunkt, absolute Luftfeuchtigkeit, Enthalpie, Lüftungsempfehlung,
            // gemessene Temperatur & Luftfeuchtigkeit inkl. Offset zwecks Kalibrierung
            // -----------------------------------------------------------------------------
            // benötigt in der Javascript das Modul "dewpoint"
            // (in der Javascript-Instanz Einstellungen unter "Zusätzliche NPM-Module")
            // -----------------------------------------------------------------------------
            //
            // Formeln zur Berechnung der Luftfeuchtigkeit:
            // http://www.nabu-eibelshausen.de/Rechner/feuchte_luft_enthalpie.html
            //
            // Empfehlung Paul53:
            // Kalibrierung der Offsetwerte in einer für den Vergleich relevanten Umgebung
            // z.B. 22°C, 65% Luftfeuchte (nicht im Winter).
            //
            // gute Infos zum Raumklima:
            // https://www.energie-lexikon.info/luftfeuchtigkeit.html
            // http://www.energiebuero-online.de/bauphysik/richtigluften.htm
             
            // Autoren des Skripts:
            // -----------------------------------------------------------------------------
            // - Paul53:
            //   Formeln, Idee, Experte im Bereich Raumklima, Korrekturen am gr. Skript
            // - Solear:
            //   Zusammenfassung der Skripte/Formeln von Paul53
            // - ruhr70:
            //   Ein Skript für alle vorhandenen Räume
            // - eric 2905:
            //   Optimierungen, viele neue Ideen, JSON-Ausgabe, globale Datenpunkte
            // - Andy3268:
            //   Hinzufügen der 4.ten Bedingung für Raumfuchte Grenzwerte
             
            // https://forum.iobroker.net/topic/2313/skript-absolute-feuchte-berechnen/437 
            // TODO:
            // -----------------------------------------------------------------------------
            // - Verzicht auf das node module ""dewpoint"
            //
            // - Einstellungen Hysterese (Expertenmodus)
            //
            // - setState / getState, die es nicht gibt: Fehler abfangen und Warnung ausgeben, damit der Adapter sich nicht beendet
            //
            // - Luftdruck alternativ vom Messgerät und nicht über Skript (ggf. per Raum)
            //
            // - Auswählbar: Datenpunkte ohne Einheit (zusätzlich) erzeugen (z.B. für vis justgage, value & indicator)
            //
            // - Auswählbar:
            //   Zweig Raum:    NICHT anlegen
            //   JSON:          NICHT anlegen
            //   DETAILS:       NICHT anlegen
            //   CONTROL:       NICHT anlegen
            //
            // - JSON wird recht groß: ggf. Datenpunkte für JSON auswählbar machen
            //
            // - ggf. JSON nicht als String zusammenbauen, sondern als json-Objekt (dann JSON.stringify(json))
            //
            // - Zähler einbauen: Anzahl Räume in Hysterese (Grenzbereich)
            //
            // # "Lüftungsengine":
            // -------------------
            // - möglichst an die individuellen Situationen und Vorlieben anpassbar
            // - differenziertere Lüftungsempfehlung
            // - CO2, Luftgüte einbeziehen
            // - Experteneinstellungen (welche Werte sind einem wichtig)
            // - Modus mit Werten/Prioritäten (wie dringend muss gelüftet werden)
            // - Kellerentlüftung einbauen (Raum markierbar als Keller)
            // - Sommer / Winter (Heizperiode) berücksichtigen
            // - dringend lüften, ab 70% rel. Luftfeuchtigkeit und geeigneter Außenluft (Vergl. absolute Luftfeuchtigkeit)
            // - Massnahme: zu trockene Luft (rel. Luftfeuchtigkeit < 40%)
            // - Massnahme: Luft rel. Feuch > 60% oder 65% (?)
            // - Feuchtigkeitstrend berücksichtigen. Ist ie Tendenz fallend, Bedingung "Entfeuchten" überstimmen.
             
            // Ideensammlung Lüftungsengine
            // - zentraler Datenpunkt: Heizperiode
            // - je Raum eine opt. Datenpunkt für eine zugeordnete Heizung (Zieltemperatur und Heizung an/aus)
            // - je Raum die Wunschtemperatur
            // - Prio: schlechte Luftqualität
            // - Prio: kühlen, wenn Temperaturunterschied zu groß
            // - Prio: zu trockene Luft (rel.)
            // - Prio: zu feuchte Luft (rel.)
             
            // berücksichtigen / Beobachtungen:
            //
            // wenn draussen zu kalt ist, macht das lüften tlw. keinen Sinn mehr
            // wenn die Zimmertemperatur bis zum Minimum abkühlt kann torz Unterschid xi/xa
            // xi und die rel. Luftfeuchte weiter steigen, da die dann kältere Raumluft weniger 
            // Luftfeuchtigkeittragen kann.
             
             
             
             
            var  DP =   require('dewpoint');        // Das Modul dewpoint einlesen
             
             
            // -----------------------------------------------------------------------------
            // Einstellungen Skriptverhalten, eigene Parameter -  !! bitte anpassen !!
            // -----------------------------------------------------------------------------
             
            // Wichtig:                             // betrifft den CONTROL Zweig bei den Raumdatepunkten (Offsets, Raummindestemperatur (Auskühlschutz))
            var skriptConf  = true;                 // Anwender kann sich aussuchen, ob er die Werte im Skript oder über die Objekte pflegen möchte
                                                   // true:  Raumwerte werden über das Skript geändert/überschrieben (var raeume)
                                                   // false: Raumwerte werden über Objekte (z.B. im Admin, Zustände oder VIS) geändert
             
            var debug = false;                      // true: erweitertes Logging einschalten
             
             
            // eigene Parameter:
            var hunn            = 68;           // eigene Höhe über nn (normalnull), z.B. über http://de.mygeoposition.com zu ermitteln
            var defaultTemp     = 18.00;     // Default TEMP_Minimum, wenn im Raum nicht angegeben (Auskühlschutz, tiefer soll eine Raumtemperatur durchs lüften nicht sinken)
            var defaultMinFeu   = 45.00;     // Default Mindest Feuchte wenn nicht angegeben.
            var defaultMaxFeu   = 55.00;     // Default Maximal Feuchte wenn nicht angegeben.
             
            var cronStr         = "*/30 * * * *";       // Zeit, in der alle Räume aktualisiert werden (da auf Änderung der Sensoren aktualisiert wird, kann die Zeit sehr hoch sein)
            var strDatum        = "DD-MM-JJJJ SS:mm:ss";// Format, in dem das Aktualisierungsdatum für das JSON ausgegeben wird
             
             
             
            // ### Experteneinstellungen ###
             
            // Lüftungsengine
             
            var hysMinTemp      = 0.5;              // Default 0.5, Hysterese Mindesttemperatur (Auskühlschutz). Innerhalb dieser Deltatemperatur bleibt die alte Lüftungsempfehlung für den Auskühlschutz bestehen.
            var hysEntfeuchten  = 0.2;              // Default 0.3, Hysterese Entfeuhten: Delta g/kG absolute Luftfeuchte. In dem Delta findet keine Änderung der alten Lüftungsempfehlung statt    
             
             
            // Skriptverhalten
            var delayRooms      = 500;              // Zeit in ms als Verzögerung, wie die Räume abgearbeitet werden
             
             
            // Pfade für die Datenpunkte:
            var pfad        = "Raumklima"   +".";   // Pfad unter dem die Datenpunkte in der Javascript-Instanz angelegt werden
             
            // Unterpfade unterhalb des Hauptpfads
            var raumPfad    = "Raum"        +".";   // Pfad unterhalb des Hauptpfads für die Räume
            var controlPfad = "CONTROL"     +".";   // Pfad innerhalb des Raums für Kontrollparameter
            var detailPfad  = "DETAILS"     +".";   // Pfad innerhalb des Raums für Detailparameter ("" und ohne ".", wenn kein Detailpfad gewünscht)
            var detailEnginePfad = "DETAILS_Lüftungsempfehlung" + "."; // Pfad innerhalb des Raums für Detailparameter zur Lüftungsengine
             
            var infoPfad    = "Skriptinfos" +".";   // Pfad für globale Skriptparameter zur Info
             
             
            // -----------------------------------------------------------------------------
            // Räume mit Sensoren, Parametrisierung -           !! bitte anpassen !!
            // -----------------------------------------------------------------------------
             
            // jeder Sensor darf nur einmal verwendet werden!
             
            // wird kein Aussensensor angegeben, wird der Sensor als Aussensensor behandelt!
             
            var raeume = { // Keine Leerzeichen (Name wird als Datenpunktname verwendet!)
                // Sensoren Aussen
                "Terrasse" : {
                    "Sensor_TEMP"           :   "zigbee.0.00158d00044484a0.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "zigbee.0.00158d00044484a0.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                },
                // Sensoren Innen
                "Badezimmer" : {
                    "Sensor_TEMP"           :   "ble.0.58:2d:34:37:a7:42.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "ble.0.58:2d:34:37:a7:42.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                    "TEMP_Minimum"         :   defaultTemp, // oder Zieltemperatur in Form von: 20.00 angeben
                    "Aussensensor"          :   "Terrasse"
                },    
                "Wohnzimmer" : {
                    "Sensor_TEMP"           :   "ble.0.58:2d:34:37:ad:0a.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "ble.0.58:2d:34:37:ad:0a.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                    "TEMP_Minimum"         :   defaultTemp, // oder Zieltemperatur in Form von: 20.00 angeben
                    "Aussensensor"          :   "Terrasse"
                },
                "Schlafzimmer" : {
                    "Sensor_TEMP"           :   "ble.0.58:2d:34:37:aa:c6.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "ble.0.58:2d:34:37:aa:c6.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                    "TEMP_Minimum"         :   defaultTemp, // oder Zieltemperatur in Form von: 20.00 angeben
                    "Aussensensor"          :   "Terrasse"
                },
                "Kueche" : {
                    "Sensor_TEMP"           :   "zigbee.0.00158d00044500f2.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "zigbee.0.00158d00044500f2.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                    "TEMP_Minimum"         :   defaultTemp, // oder Zieltemperatur in Form von: 20.00 angeben
                    "Aussensensor"          :   "Terrasse"
                },
                 "Flur" : {
                    "Sensor_TEMP"           :   "ble.0.58:2d:34:37:ab:0d.temperature"/*Temperature*/,
                    "Sensor_HUM"            :   "ble.0.58:2d:34:37:ab:0d.humidity"/*Humidity*/,
                    "Sensor_TEMP_OFFSET"    :   0.0,
                    "Sensor_HUM_OFFSET"     :   0,
                    "TEMP_Minimum"         :   defaultTemp, // oder Zieltemperatur in Form von: 20.00 angeben
                    "Aussensensor"          :   "Terrasse"
                },
            };
             
            // =============================================================================
             
            // =============================================================================
            // Skriptbereich. Ab hier muss nichts mehr eingestellt / verändert werden.
            // =============================================================================
             
            // =============================================================================
             
             
            var idSkriptinfoBar         = pfad + infoPfad + "Luftdruck";
            var idSkriptinfoHunn        = pfad + infoPfad + "Höhe_über_NN";
             
            // forceCreation = true, damit bei geändert eigener Höhe im Konfigurationsbereich der Datenpunkt neu geschrieben wird
            createState(idSkriptinfoBar, luftdruck(hunn), true, {
               name: 'mittlerer Luftdruck in bar',
               desc: 'mittlerer Luftdruck in bar, errechnet anhand der eigenen Höhe über NN',
               type: 'number',
               unit: 'bar',
               role: 'info'
            });
             
            createState(idSkriptinfoHunn, hunn, true, {
               name: 'Eigene Höhe über NN',
               desc: 'Eigene Höhe über NN (Normal Null), als Basis für den mittleren Luftdruck',
               type: 'number',
               unit: 'm',
               role: 'info'
            });
             
             
            var raumDatenpunkte = {
               "x" : {
                   "DpName" : "Feuchtegehalt_Absolut",
                   "init": 0,
                   "dp": {
                       "name": 'absoluter Feuchtegehalt',
                       "desc": 'absoluter Feuchtegehalt, errechnet',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'g/kg'
                   }
               },
               "rh" : {
                   "DpName" : "relative_Luftfeuchtigkeit",
                   "init": 0,
                   "dp": {
                       "name": 'gemessene relative Luftfeuchtigkeit (inkl. Offset)',
                       "desc": 'relative Luftfeuchtigkeit, vom Sensor + Offset zum Ausgleich von Messungenauigkeiten des Geräts',
                       "type": 'number',
                       "role": 'value',
                       "unit": '%'
                   }
               },
               "dp" : {
                   "DpName" : "Taupunkt",
                   "init": 0,
                   "dp": {
                       "name": 'Taupunkt',
                       "desc": 'Taupunkt. Temperatur von Wänden, Fenstern, usw. ab der sich die Feuchtigkeit niederschlägt.',
                       "type": 'number',
                       "role": 'value',
                       "unit": '°C'
                   }
               },
               "t" : {
                   "DpName" : "Temperatur",
                   "init": 0,
                   "dp": {
                       "name": 'gemessene Temperatur (inkl. Offset)',
                       "desc": 'gemessene Temperatur vom Sensor zzgl. eines Offsets um Geräteungenauigkeiten auszugleichen',
                       "type": 'number',
                       "role": 'value',
                       "unit": '°C'
                   }
               },
               "h" : {
                   "DpName" : detailPfad + "Enthalpie",
                   "init": 0,
                   "dp": {
                       "name": 'Enthalpie',
                       "desc": 'Enthalpie',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'kJ/kg'
                   }
               },
               "sdd" : {
                   "DpName" : detailPfad +"Sättigungsdampfdruck",
                   "init": 0,
                   "dp": {
                       "name": 'Sättigungsdampfdruck',
                       "desc": 'Sättigungsdampfdruck',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'hPa'
                   }
               },
               "dd" : {
                   "DpName" : detailPfad + "Dampfdruck",
                   "init": 0,
                   "dp": {
                       "name": 'Dampfdruck',
                       "desc": 'Dampfdruck',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'hPa'
                   }
               },
               "rd" : {
                   "DpName" : "Dampfgewicht",
                   "init": 0,
                   "dp": {
                       "name": 'Dampfgewicht (Wassergehalt)',
                       "desc": 'Dampfgewicht (Wassergehalt)',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'g/m³'
                   }
               },
               "maxrd" : {
                   "DpName" : detailPfad + "Dampfgewicht_maximal",
                   "init": 0,
                   "dp": {
                       "name": 'max. Dampfgewicht (Wassergehalt)',
                       "desc": 'max. Dampfgewicht (Wassergehalt) bei aktueller Temperatur',
                       "type": 'number',
                       "role": 'value',
                       "unit": 'g/m³'
                   }
               },
               "lüften" : {
                   "DpName" : "Lüftungsempfehlung",
                   //"init": false,
                   "dp": {
                       "name": 'Lüftungsempfehlung',
                       "desc": 'Lüftungsempfehlung',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_b1" : {
                   "DpName" : detailEnginePfad + "Lüften_b1_Entfeuchten",
                   //"init": false,
                   "dp": {
                       "name": 'Lüften Bedingung 1 entfeuchten',
                       "desc": 'Lüften Bedingung 1 entfeuchten erfüllt',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_b2" : {
                   "DpName" : detailEnginePfad + "Lüften_b2_Kühlen",
                   //"init": false,
                   "dp": {
                       "name": 'Lüften Bedingung 2 kühlen',
                       "desc": 'Lüften Bedingung 2 kühlen erfüllt',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_b3" : {
                   "DpName" : detailEnginePfad + "Lüften_b3_Auskühlschutz",
                   //"init": false,
                   "dp": {
                       "name": 'Lüften Bedingung 3 Auskühlschutz',
                       "desc": 'Lüften Bedingung 3 Auskühlschutz erfüllt (Innentemperatur soll nicht unter Minimumteperatur fallen)',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_b4" : {
                   "DpName" : detailEnginePfad + "Lüften_b4_Raumfeuchte",
                   //"init": false,
                   "dp": {
                       "name": 'Lüften Bedingung 4 Raumfeuchte',
                       "desc": 'Lüften Bedingung 4 Raumfeuchte erfüllt',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_Hysterese" : {
                   "DpName" : detailEnginePfad + "Lüften_Hysterese",
                   //"init": false,
                   "dp": {
                       "name": 'Logik im Bereich der Hysterese. Keine Änderung der bestehenden Lüftungsempfehlung.',
                       "desc": 'Logik im Bereich der Hysterese. Keine Änderung der bestehenden Lüftungsempfehlung.',
                       "type": 'boolean',
                       "role": 'value'
                   }
               },
               "lüften_Beschreibung" : {
                   "DpName" : detailEnginePfad + "Lüftungsempfehlung_Beschreibung",
                   "init": "",
                   "dp": {
                       "name": 'Lüftungsempfehlung beschreibender Text',
                       "desc": 'Lüftungsempfehlung beschreibender Text',
                       "type": 'string',
                       "role": 'value'
                   }
               }
            };
             
               // #1 - Entfeuchten:    Außenluft ist mind. (hysEntfeuchten + 0,1) trockener als Innen
               // #2 - Kühlen:         Außentemperatur ist mindestens 0,6 Grad kühler als innen TODO: im Winter auch?
               // #3 - Auskühlschutz:  Innentemperatur ist höher als die Mindesttemperatur
             
             
            var raumControl = {
               "Sensor_TEMP_OFFSET" : {
                   "DpName" : "Sensor_TEMP_OFFSET",
                   "init": 0,
                   "dp": {
                       "name": 'Offset Temperatur zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                       "desc": 'Offset Temperatur zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                       "type": 'number',
                       "role": 'control.value',
                       "unit": '°C'
                   }
               },
               "Sensor_HUM_OFFSET" : {
                   "DpName" : "Sensor_HUM_OFFSET",
                   "init": 0,
                   "dp": {
                       "name": 'Offset Luftfeuchtigkeit zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                       "desc": 'Offset Luftfeuchtigkeit zum Sensormesswert (Ausgleich von Ungenauigkeiten)',
                       "type": 'number',
                       "role": 'control.value',
                       "unit": '%'
                   }
               },
               "TEMP_Minimum" : {
                   "DpName" : "TEMP_Minimum",
                   "init": 0,
                   "dp": {
                       "name": 'Auskühlschutz Mindestraumtemperatur',
                       "desc": 'Auskühlschutz Mindestraumtemperatur zum lüften',
                       "type": 'number',
                       "role": 'control.value',
                       "unit": '°C'
                   }
               },
               "FEUCH_Minimum" : {
                   "DpName" : "FEUCH_Minimum",
                   "init": 0,
                   "dp": {
                       "name": 'Mindest Rel. Raumfeuchte',
                       "desc": 'Mindest Rel. Raumfeuchte zum lüften',
                       "type": 'number',
                       "role": 'control.value',
                       "unit": '%'
                   }
               },
               "FEUCH_Maximum" : {
                   "DpName" : "FEUCH_Maximum",
                   "init": 0,
                   "dp": {
                       "name": 'Maximal Rel. Raumfeuchte ',
                       "desc": 'Maximal Rel. Raumfeuchte zum lüften',
                       "type": 'number',
                       "role": 'control.value',
                       "unit": '%'
                   }
               },
               "Aussensensor" : {
                   "DpName" : "Aussensensor",
                   "init": "",
                   "dp": {
                       "name": 'Aussensensor, der zum Vergleich genommen wird',
                       "desc": 'Aussensensor, der zum Vergleich genommen wird',
                       "type": 'string',
                       "role": 'control.value'
                   }
               }
            };
             
             
            // globale Skript-Variablen/Objekte
            //------------------------------------------------------------------------------
             
            var xdp     = new DP(hunn);
             
            var pbar    = luftdruck(hunn);          // individueller Luftdruck      in bar (eigene Höhe)
             
             
             
            //------------------------------------------------------------------------------
            // Funktionen
            //------------------------------------------------------------------------------
             
            function writeJson(json) {
               return JSON.stringify(json);
            }
             
             
            // prüft ob setObjects() für die Instanz zur Verfügung steht (true/false)
            function checkEnableSetObject() { 
               var enableSetObject = getObject("system.adapter.javascript." + instance).native.enableSetObject;
               return enableSetObject;
            }
             
             
            function setChannelName(channelId,channelName){
               if(checkEnableSetObject()) { // wenn setObject nicht in der Instanz freigeschaltet ist, wird der Channel nicht angelegt
               // CHANNEL anlegen
                   setObject("javascript." + instance + "." + channelId, {
                       common: {
                           name: channelName
                       },
                       type: 'channel'
                   }, function(err) {
                       if (err) logs('Cannot write object: ' + err,"error");
                   });
               }
            }
             
             
            function lueftenDp(datenpunktID) {
               return (datenpunktID == "lüften") || (datenpunktID == "lüften_Beschreibung") || (datenpunktID == "lüften_b1") || (datenpunktID == "lüften_b2") || (datenpunktID == "lüften_b3") || (datenpunktID == "lüften_b4") || (datenpunktID ==  "lüften_Hysterese");
            }
             
             
            function createDp() {
               var name;
               var init;
               var forceCreation;
               var common;
               for (var raum in raeume) {
                   for (var datenpunktID in raumDatenpunkte) {
                       name = pfad + raumPfad + raum + "." + raumDatenpunkte[datenpunktID].DpName;
                       init = raumDatenpunkte[datenpunktID].init;
                       forceCreation = false; // Init der Datenpunkte wird nur beim ersten Star angelegt. Danach bleiben die Wert auch nach Skritpstart enthalten.
                       common = raumDatenpunkte[datenpunktID].dp;
                       
                       if (lueftenDp(datenpunktID)) {
                           if (!raeume[raum].Aussensensor) {
                               if (datenpunktID == "lüften") {
                                   log(raum + ": kein Aussensensor angegeben.  ### Messpunkte werden als Aussensensoren behandelt. ###","info"); // Warnung ist im Log OK, wenn es sich um einen Außensensor handelt.
                                   setChannelName(pfad + raumPfad + raum,"Aussensensor");
                               }
                           } else {
                               createState(name, init , forceCreation, common);
                               if (debug) log("neuer Datenpunkt: " + name);
                           }
                       } else {
                           createState(name, init , forceCreation, common);
                           if (debug) log("neuer Datenpunkt: " + name);
                       }
                       
                   }
                   for (var control in raumControl) {
                       name = pfad + raumPfad + raum + "." + controlPfad + raumControl[control].DpName;
                       //init = raumControl[control].init;
                       forceCreation = skriptConf;
                       common = raumControl[control].dp;
                       if (typeof raeume[raum][raumControl[control].DpName] !=="undefined") {
                           init = raeume[raum][raumControl[control].DpName];
                           createState(name, init , forceCreation, common);
                           var channelname = "Nur Info. Werte aus dem Skript zählen. Kann im Skript umgestellt werden.";
                           if (!skriptConf) channelname = "Änderungen hier in den Objekten werden berechnet";
                           setChannelName(pfad + raumPfad + raum + "." + controlPfad.substr(0, controlPfad.length-1),channelname);
                       }
                   }
               }
               
               //eric2905 Datenpunkt "Lüften" erzeugen
               // -------------------------------------------------------------------------
               createState(pfad + 'Lüften', false, {
                name: 'Muss irgendwo gelüftet werden',
                desc: 'Muss irgendwo gelüftet werden',
                type: 'boolean',
                unit: '',
                role: 'value'
               });
             
               createState(pfad + 'Lüften_Liste', "[]", {
                name: 'Liste der Räume in denen gelüftet werden muss',
                desc: 'Liste der Räume in denen gelüftet werden muss',
                type: 'string',
                unit: '',
                role: 'value'
               });
             
               // eric2905 Ende -----------------------------------------------------------
             
               //eric2905 Datenpunkt "JSON" erzeugen
               // -------------------------------------------------------------------------
               createState(pfad + 'JSON', "", {
                name: 'JSON-Ausgabe aller Werte',
                desc: 'JSON-Ausgabe aller Werte',
                type: 'string',
                unit: '',
                role: 'value'
               });
               // eric2905 Ende -----------------------------------------------------------
             
               //eric2905 Datenpunkt "Aktualsierung" erzeugen
               // -------------------------------------------------------------------------
               createState(pfad + 'Aktualsierung', "", {
                name: 'Aktualisierungszeitpunkt der JSON-Ausgabe',
                desc: 'Aktualisierungszeitpunkt der JSON-Ausgabe',
                type: 'string',
                unit: '',
                role: 'value'
               });
               // eric2905 Ende -----------------------------------------------------------
              
              
               //eric2905 Datenpunkt "countLueften" erzeugen
               // -------------------------------------------------------------------------
               createState(pfad + 'Lüften_Anzahl', 0, {
                name: 'Anzahl Lüftungsempfehlungen',
                desc: 'Anzahl Lüftungsempfehlungen',
                type: 'number',
                unit: '',
                role: 'value'
               });
               // eric2905 Ende -----------------------------------------------------------
             
               log("Datenpunkte angelegt");
            }
             
             
            // rundet einen Float auf eine bestimmte Anzahl Nachkommastellen
            function runden(wert,stellen) {
               return Math.round(wert * Math.pow(10,stellen)) / Math.pow(10,stellen);
            }
             
            // berechnet den mittleren Luftdruck für eine Höhenangabe in NN 
            function luftdruck(hunn) {
               var pnn         = 1013.25;                                  // Mittlerer Luftdruck          in hPa bei NN
               var p           = pnn - (hunn / 8.0);                       // individueller Luftdruck      in hPa (eigenen Höhe)
               return p / 1000;                                            // Luftdruck von hPa in bar umrechnen
            }
             
            // Color Boolean (farbige Ausgabe Boolean als String, z.B. für das Log)
            function cob(boolean) { 
               var cobStr = (boolean) ? '<span style="color:lime;"><b>true</b></span>' : '<span style="color:red;"><b>false</b></span>';
               return cobStr;
            }
             
            function makeNumber(wert) {
               if(isNaN(wert)) {
                   wert = parseFloat(wert.match(/\d+[.|,]?\d+/g));
               }
               return wert;
            }
             
             
             
            // Berechnungen Luftwerte 
            // ----------------------
             
            function calcSaettigungsdampfdruck(t) {    // benötigt die aktuelle Temperatur
               // Quelle: http://www.wetterochs.de/wetter/feuchte.html#f1
               var sdd,a,b;
               a = 7.5;
               b = 237.3;
               sdd = 6.1078 * Math.pow(10,((a*t)/(b+t)));
               return sdd; // ssd = Sättigungsdampfdruck in hPa
            }
             
            function calcDampfdruck(sdd,r) {
               // Quelle: http://www.wetterochs.de/wetter/feuchte.html#f1
               var dd = r/100 *sdd;
               return dd;  // dd = Dampfdruck in hPa
            }
             
            function calcTemperaturKelvin(t) {
               var tk = t + 273.15;
               return tk;
            }
             
            function calcDampfgewicht(dd,t) { // Wassergehalt
               // Dampfgewicht rd oder AF(r,TK) = 10^5 * mw/R* * DD(r,T)/TK
               // Quelle: http://www.wetterochs.de/wetter/feuchte.html#f1
               var tk = calcTemperaturKelvin(t);
               var mw = 18.016; // kg/kmol (Molekulargewicht des Wasserdampfes)
               var R  = 8314.3; // J/(kmol*K) (universelle Gaskonstante)
               var rd = Math.pow(10,5) * mw/R * dd/tk; 
               return rd; // rd = Dampfgewicht in g/m^3
            }
             
            function calcMaxDampfgewicht(rd,r) {
               var maxrd = rd / r *100;
               return maxrd;
            }
             
             
             
             
            // Berechnung: alle Werte je Raum
            // -------------------------------
             
             
            function calc(raum) {                                           // Über Modul Dewpoint absolute Feuchte berechnen
             
               var t           = getState(raeume[raum].Sensor_TEMP).val;   // Temperatur auslesen
               var rh          = getState(raeume[raum].Sensor_HUM).val;    // Feuchtigkeit relativ auslesen
             
               t   = makeNumber(t);                                        // Temperatur in Number umwandeln
               rh  = makeNumber(rh);                                       // relative Luftfeuchtigkeit in Number umwandeln
             
               var toffset     = 0.0;                                      // Default Offset in °C
               var rhoffset    = 0;                                        // Default Offset in %
               if(typeof raeume[raum].Sensor_TEMP_OFFSET !=="undefined") {
                   // Temperatur, wenn ein Offset vorhanden ist, diesen auslesen und Default überschreiben
                   var idtoffset = pfad + raumPfad+ raum + "." + controlPfad + "Sensor_TEMP_OFFSET";
                   toffset = getState(idtoffset).val;  // Offset aus den Objekten/Datenpunkt auslesen
               }
               if(typeof raeume[raum].Sensor_HUM_OFFSET !=="undefined") {
                   // Luftfeuchtigkeit, wenn ein Offset vorhanden ist, diesen auslesen und Default überschreiben
                   var idrhoffset = pfad + raumPfad + raum + "." + controlPfad + "Sensor_HUM_OFFSET";
                   rhoffset = getState(idrhoffset).val;  // Offset aus den Objekten/Datenpunkt auslesen
               }
             
               t       = t     + toffset;      // Messwertanpassung: gemessene Temperatur um den Offset ergänzen
               rh      = rh    + rhoffset;     // Messwertanpassung: gemessene relative Luftfeuchtigkeit um Offset ergänzen
             
               var y           = xdp.Calc(t, rh);
               var x   = y.x;  // Zu errechnende Variable für Feuchtegehalt in g/kg
               var dp  = y.dp; // Zu errechnende Variable für Taupunkt in °C
             
               var h       = 1.00545 * t + (2.500827 + 0.00185894 * t) * x;    // Enthalpie in kJ/kg berechnen
             
               var sdd     = calcSaettigungsdampfdruck(t);                     // Sättigungsdampfdruck in hPa
               var dd      = calcDampfdruck(sdd,rh);                           // dd = Dampfdruck in hPa
               var rd      = calcDampfgewicht(dd,t);                           // rd = Dampfgewicht/Wassergehalt in g/m^3
               var maxrd   = calcMaxDampfgewicht(rd,rh);                       // maximales Dampfgewicht in g/m^3
               
             
               var idx     = pfad + raumPfad + raum + "." + raumDatenpunkte["x"].DpName;   // DP-ID absolute Luftfeuchte in g/kg
               var iddp    = pfad + raumPfad + raum + "." + raumDatenpunkte["dp"].DpName;  // DP-ID Taupunkt in °C
               var idt     = pfad + raumPfad + raum + "." + raumDatenpunkte["t"].DpName;   // DP-ID Temperatur inkl. Offset
               var idrh    = pfad + raumPfad + raum + "." + raumDatenpunkte["rh"].DpName;  // DP-ID relative Luftfeuhtigkeit inkl. Offset
               var ih      = pfad + raumPfad + raum + "." + raumDatenpunkte["h"].DpName;   // DP-ID Enthalpie in kJ/kg
               var isdd    = pfad + raumPfad + raum + "." + raumDatenpunkte["sdd"].DpName;
               var idd     = pfad + raumPfad + raum + "." + raumDatenpunkte["dd"].DpName;
               var ird     = pfad + raumPfad + raum + "." + raumDatenpunkte["rd"].DpName;
               var imaxrd  = pfad + raumPfad + raum + "." + raumDatenpunkte["maxrd"].DpName;
             
             
               setState(idx    , runden(x,2));     // errechnete absolute Feuchte in Datenpunkt schreiben
               setState(iddp   , runden(dp,1));    // errechneter Taupunkt in Datenpunkt schreiben
               setState(idt    , parseFloat(t));   // Sensor Temperatur        inkl. Offset
               setState(idrh   , parseFloat(rh));   // Sensor Relative Feuchte  inkl. Offset
               setState(ih     , runden(h,2));     // Enthalpie in kJ/kg
               setState(isdd   , runden(sdd,2));
               setState(idd    , runden(dd,2));
               setState(ird    , runden(rd,2));
               setState(imaxrd , runden(maxrd,2));
             
             
               // Logik-Engine: Lüftungsempfehlung berechnen
               // -------------------------------------------------------------------------
               if (!raeume[raum].Aussensensor) {
                   // kein Aussensensor, keine Lüftungsempfehlung
                   if (debug) log("<b>------ " + raum + " ------- Aussen, keine Lüftungsempfehlung -----------</b>");
                   return; 
               }
               
               var aussen;
               var idta, idxa;
               if(typeof raeume[raum].Aussensensor !=="undefined") {
                   aussen = raeume[raum].Aussensensor; // aussen = "Raumname" des zugehörigen Aussensensors
                   idta = pfad + raumPfad + aussen + "." + raumDatenpunkte["t"].DpName;    // DP-ID zugehöriger Aussensensor, Temperatur aussen
                   idxa = pfad + raumPfad + aussen + "." + raumDatenpunkte["x"].DpName;    // DP-ID zugehöriger Aussensensor, Luftfeuchtigkeit aussen
               } else {
                   return; // wenn es keinen zugehörigen Aussensensor gibt, Funktion beenden (dann muss kein Vergleich berechnet werden)
               }
             
               var ti = t;                     // Raumtemperatur in °C
               var xi = runden(x,2);           // Raumfeuchtegehalt in g/kg
               var ta = getState(idta).val;    // Aussentemperatur in °C
               var xa = getState(idxa).val;    // Aussenfeuchtegehalt in g/kg
               if (xa == 0) return;            // TODO: warum? hatte ich leider nciht dokumentiert (ruhr70)
             
               var mi = defaultTemp;           // Temperaturmindestwert auf Default (Auskühlschutz)
               var xh = defaultMaxFeu;         // Feuchtemaximalwert auf Default
               var xt = defaultMinFeu;         // Feuchteminimalwert auf Default
            	  
               //if(typeof raeume[raum].TEMP_Minimum !=="undefined") {
               if(typeof raeume[raum].TEMP_Minimum == "number") {
                   mi = raeume[raum].TEMP_Minimum;
               }
               if(typeof raeume[raum].FEUCH_Maximum == "number") {
                   xh = raeume[raum].FEUCH_Maximum;
               }
             
               if(typeof raeume[raum].FEUCH_Minimum == "number") {
                   xt = raeume[raum].FEUCH_Minimum;
               }
               
               // Auskühlschutz,  hysMinTemp (Variable) Grad hysMinTemp Hysterese. Tiefer darf die Innentemperatur nicht sinken
               var mih = mi + hysMinTemp;      // Temperaturmindestwert hoch (Mindesttemperatur plus Hysterese)
               var mit = mi;                   // Temperaturmindestwert tief
             
               var idLueften       = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften"].DpName;
               var idLueftenText   = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_Beschreibung"].DpName;
               var idLueftenB1     = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_b1"].DpName;
               var idLueftenB2     = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_b2"].DpName;
               var idLueftenB3     = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_b3"].DpName;
               var idLueftenB4     = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_b4"].DpName;
               var idLueftenHys    = pfad + raumPfad + raum + "." + raumDatenpunkte["lüften_Hysterese"].DpName;
             
               var lueftenText     = "";
             
             
             
               // Lüftungslogik
               // -------------
               // Lüftungsempfehlung steuern mit 0,3 g/kg und 0,5 K Hysterese
               // Bedigungen fürs lüften
               var b1lp = (xa <= (xi - (hysEntfeuchten + 0.1)))    ? true : false;   // Bedingnung 1 lüften positv (Außenluft ist mind. 0,4 trockener als Innen)
               var b2lp = (ta <= (ti - 0.6))                       ? true : false;   // Bedingnung 2 lüften positv (Außentemperatur ist mindestens 0,6 Grad kühler als innen)
               var b3lp = (ti >= mih)                              ? true : false;   // Bedingnung 3 lüften positv (Innentemperatur ist höher als die Minimumtemperatur + Hysterese)
               var b4lp = (rh >= xh)                               ? true : false;   // Bedingnung 4 lüften positv (Relative Raumfeuchte ist höher als die Maximalfeuchtewert)
             
               var b1lpText = "Entfeuchten:    Außenluft ist mind. 0,4 trockener als Innen";
               var b2lpText = "Kühlen:         Außentemperatur ist mindestens 0,6 Grad kühler als innen";
               var b3lpText = "Auskühlschutz:  Innentemperatur ist höher als die Mindesttemperatur";
               var b4lpText = "Raumfeuchte:    Raumfeuchte ist höher als der Maximalfeuchte";
             
               setState(idLueftenB1,b1lp);
               setState(idLueftenB2,b2lp);
               setState(idLueftenB3,b3lp);
               setState(idLueftenB4,b4lp);
             
               // Bedingungen gegen das Lüften
               var b1ln = (xa >= (xi - 0.1))   ? true : false;   // Bedingnung 1 lüften negativ (Außenluft ist zu feucht)
               var b2ln = (ta >= (ti - 0.1))   ? true : false;   // Bedingnung 2 lüften negativ (Außentemperatur zu warm)
               var b3ln = (ti <= mit)          ? true : false;   // Bedingnung 3 lüften negativ (Innentemperatur niedriger als Mindesttemperatur)
               var b4ln = (rh <= xt)           ? true : false;   // Bedingnung 4 lüften negativ (Relative Raumfeuchte ist niedriger als die Mindestfeuchte)
             
               var b1lnText = "Entfeuchten:    Außenluft ist zu feucht";
               var b2lnText = "Kühlen:         Außentemperatur zu warm";
               var b3lnText = "Auskühlschutz:  Innentemperatur niedriger als Mindestraumtemperatur";
               var b4lnText = "Raumfeuchte:    Raumfeuchte ist niedriger als der Mindestfeuchte";
             
               
               // Logik:
               //--------------------------------------------------------------------------
               if (b1lp && b2lp && b3lp && b4lp) {
                   // Lüftungsempfehlung, alle bedingungenen erfüllt
                   lueftenText = "Bedingungen für Entfeuchten, Kühlen und Auskühlschutz erfüllt.";
                   setState(idLueften, true);
                   setState(idLueftenHys,false);
             
                   if (debug) log(raum + ': <span style="color:limegreen;"><b>Lüftungsempfehlung</b></span>');
             
               } else if (b1ln || b2ln || b3ln || b4ln) {
                   // Fenster zu. Ein Ausschlusskriterium reicht für die Empfehlung "Fenster zu".
                   lueftenText = "Fenster zu:<br>";
                   if (b1ln) lueftenText += b1lnText + "<br>";
                   if (b2ln) lueftenText += b2lnText + "<br>";
                   if (b3ln) lueftenText += b3lnText + "<br>";
            	   if (b4ln) lueftenText += b4lnText + "<br>";
                   setState(idLueften, false);
                   setState(idLueftenHys,false);
                   if (debug) log(raum + ': <span style="color:red;"><b>Empfehlung Fenster zu</b></span>');
               } else {
                   // Hysterese. Keine Änderung der bisherigen Empfehlung.
                   if (debug) log(raum + ': <span style="color:orange;"><b>im Bereich der Hysterese</b></span> (keine Änderung der Lüftungsempfehlung');
                   if (getState(idLueften).val === null) setState(idLueften,false); // noch keine Empfehlung vorhanden, "Fenster zu" empfehlen
                   lueftenText = "Hysterese, keine Änderung der Lüftungsempfehlung";
                   setState(idLueftenHys,true);
               }
               setState(idLueftenText, lueftenText);
             
             
               /* Erklärung Lüftungslogik (von Paul53)
                  Ergänzung #4 (von Andy3268)
               Lüften:
               wenn    abs. Aussenfeuchte  <   abs. Innenfeuchte     - Hysterese (Entfeuchten)
               UND     Aussentemperatur    <   Innentemperatur       - Hysterese (Kühlen)
               UND     Innentemperatur     >=  Raumtemperaturminimum + Hysterese (Auskühlschutz)
               UND     Innenfeuchte        >=  Raummaximalfechte
               */
             
               // lüften (und - Alle Bedingungen müssen erfüllt sein):
               // #1 - Entfeuchten:    Außenluft ist mind. (hysEntfeuchten + 0,1) trockener als Innen
               // #2 - Kühlen:         Außentemperatur ist mindestens 0,6 Grad kühler als innen TODO: im Winter auch?
               // #3 - Auskühlschutz:  Innentemperatur ist höher als die Mindesttemperatur
               // #4 - Raumfeuchte:    Innenfeuchte ist höher als die Maximalfeuchte
             
               // nicht lüften (oder):
               // #1 - Außenluft ist zu feucht
               // #2 - Außentemperatur zu warm
               // #3 - Innentemperatur niedriger als Mindestraumtemperatur
               // #4 - Innenfeuchte niedriger als Mindestfeuchte
             
               if (debug) log(raum + ":" + cob(b4ln) + " Raumluft ist trocken genug (b4ln): ");
               if (debug) log(raum + ":" + cob(b3ln) + " Außenluft ist zu feucht (b3ln): ");
               if (debug) log(raum + ":" + cob(b2ln) + " Außentemperatur zu warm (b2ln): ");
               if (debug) log(raum + ":" + cob(b1ln) + " Außenluft ist zu feucht (b1ln): " + ": xa: " + xa + " >= (xi - 0.1) " + (xi - 0.1));
               if (debug) log(raum + ": Fenster zu (ein true reicht):");
               
               //if (debug) log(raum + ": b1lp: " + b1lp+ ", b2lp: " + b2lp+ ", b3lp: " + b3lp, b4lp: " + b4lp);
               if (debug) log(raum + ":" + cob(b4lp) + " Raumfeuchte ist niedriger als der Mindstwert (b4lp): "); 
               if (debug) log(raum + ":" + cob(b3lp) + " Innentemperatur ist höher als die Mindesttemperatur (b3lp): ");
               if (debug) log(raum + ":" + cob(b2lp) + " Außentemperatur ist mindestens 0,6 Grad kühler als innen (b2lp): ");
               if (debug) log(raum + ":" + cob(b1lp) + " Außenluft ist mind. 0,4° trockener als Innen (b1lp):  xa: " + xa + " <= (xi - 0.4) " + (xi - 0.4));
               if (debug) log(raum + ": Lüftungsempfehlung (alle Bedingungen auf true):");
             
               if (debug) log(raum + ", ti:"+ti+", ta: "+ta+", xi:"+xi+", xa: "+xa+", mih:"+mih+", mit:"+mit,"info");
               if (debug) log("<b>------ " + raum + " ------- Aussensensor: " + aussen + " -----------</b>");
            }
             
             
             
             
             
            //eric2905 Erzeuge JSON und setzen Variablen "anyLueften" und "countLueften"
            // -----------------------------------------------------------------------------
            function createJSON() {
               // alle Daten im JSON werden als String abgelegt
               if (debug) log("=========================================================");
               if (debug) log("Erzeugung JSON Start");
               if (debug) log("=========================================================");
             
               var anyLueften          = false;
               var countLueften        = 0;
               var raeumeLueftenListe  = [];
               
               var temppfad = "";
               var tempraum = "";
               var tempVal = "";
               var strJSONfinal = "[";
               var strJSONtemp = "";
             
               for (var raum in raeume) {
                   strJSONtemp = strJSONtemp + "{";
                   strJSONtemp = strJSONtemp + "\"Raum\":\"" + raum + "\",";
             
                   for (var datenpunktID in raumDatenpunkte) {
                       
                       // Aussensensor ja oder nein 
                       var aussensensor = false;
                       if (lueftenDp(datenpunktID)) {
                           if (!raeume[raum].Aussensensor) {
                               aussensensor = true;
                           }
                       }
                       
                       temppfad = pfad + raumPfad + raum + "." + raumDatenpunkte[datenpunktID].DpName;
                       tempraum = pfad + raumPfad + raum;
             
                       tempVal = (!aussensensor ? getState(temppfad).val : "");            // kein Aussensenosr: Lüftungsempfehlung auslesen, Aussensensor: Lüftungsempfehlung freilassen
                       if (tempVal === null) tempVal = "";
                       if(raumDatenpunkte[datenpunktID].DpName != "Lüftungsempfehlung") {
                           tempVal = parseFloat(tempVal);
                           tempVal = tempVal.toFixed(2);
                       } else {
                           if (tempVal === true) {
                               anyLueften = true;
                               countLueften = countLueften + 1;
                               raeumeLueftenListe.push(raum);
                           }
                       }
                       strJSONtemp = strJSONtemp + "\"" + raumDatenpunkte[datenpunktID].DpName + "\":\"" + tempVal + "\",";
                       
                   }
                   strJSONtemp = strJSONtemp.substr(0, strJSONtemp.length - 1);
                   strJSONtemp = strJSONtemp + "},";
             
               }
             
               strJSONtemp = strJSONtemp.substr(0, strJSONtemp.length - 1);
               strJSONfinal = strJSONfinal + strJSONtemp + "]";
               if (debug) log("strJSONfinal = " + strJSONfinal);
               if (debug) log("anyLueften = " + anyLueften + ", Anzahl Lüftungsempfehlungen: " + countLueften);
               
               
               setState(pfad + 'Lüften'                    , anyLueften);
               setState(pfad + 'Lüften_Liste'              , writeJson(raeumeLueftenListe));
               setState(pfad + 'Lüften_Anzahl'             , countLueften);
               setState(pfad + 'JSON'                      , strJSONfinal);
               setState(pfad + 'Aktualsierung'             , formatDate(new Date(), strDatum));
               
               if (debug) log("=========================================================");
               if (debug) log("Erzeugung JSON Ende");
               if (debug) log("=========================================================");
            }
            // eric2905 Ende ---------------------------------------------------------------
             
             
             
             
            function calcDelayed(raum, delay) {
               setTimeout(function () {
                   calc(raum);
               }, delay || 0);
            }
             
            function creatJSONDelayed() {
               setTimeout(function () {
                   createJSON();
               }, 4000); 
            }
             
            // Klimadaten in allen Räumen berechnen 
            function calcAll() {
               for (var raum in raeume) {
                   calcDelayed(raum,delayRooms);       // Räume verzögerd nacheinander abarbeiten
               }
            }
             
             
            // finde anhand der Sensor ID einen zugeordneten Raum
            function findRoom(sensor) {
               for (var raum in raeume) {
                   if (raeume[raum].Sensor_TEMP == sensor) return raum;
                   if (raeume[raum].Sensor_HUM == sensor) return raum;
               }
               return null;
            }
             
            // Änderung eines Sensors (Temperatur oder Luftfeuchtigkeit)
            function valChange(obj) {
               var raumname = findRoom(obj.id);
               if (raumname) {
                   if (debug) log('<span style="color:black;"><b>Änderung:' + raumname + ": " + obj.id + ": " + obj.state.val + '</b></span>');
                   calcDelayed(raumname,delayRooms);
               }
               // eric2905 Aufruf eingebaut zum JSON erzeugen und Datenpunkt befüllen
               // -----------------------------------------------------------------------------
               creatJSONDelayed();
               // eric2905 Ende ---------------------------------------------------------------
            }
             
             
            // Datenpunkte für alle Räume anlegen
            function createOn() {
               var dpId    = "";
             
               // TODO: Im Modus CONTROL über Objekte: Bei Änderung der OFFSETS, Temperatur_Minimum werden die Änderung erst nach Aktualisierung der Messwerte oder nach Zeit erneuert (auf on() reagieren) 
               var i =0;
             
               for (var raum in raeume) {
             
                   if (raeume[raum].Sensor_TEMP) {
                       dpId = raeume[raum].Sensor_TEMP;
                       i++;
                       on({id: dpId ,change:'ne'}, function (obj) {
                           valChange(obj);
                       });
                       if (debug) log("on: " + dpId + " angelegt.");
                   }
             
                   if (raeume[raum].Sensor_HUM) {
                       dpId = raeume[raum].Sensor_HUM;
                       i++;
                       on({id: dpId ,change:'ne'}, function (obj) {
                           valChange(obj)
                       });
                       if (debug) log("on: " + dpId + " angelegt.");
                   }
               }
               log("Subscriptions angelegt: " + i);
            }
             
             
             
            // Schedule
            // =============================================================================
             
            // Nach Zeit alle Räume abfragen
            schedule(cronStr, function () {
               calcAll();
               // eric2905 Aufruf eingebaut zum JSON erzeugen und Datenpunkt befüllen
               creatJSONDelayed();
               // eric2905 Ende ---------------------------------------------------------------
            });
             
             
            // main()
            // =============================================================================
             
            function main() {
               calcAll();
               setTimeout(calcAll,2000);
               // eric2905 Aufruf eingebaut zum JSON erzeugen und Datenpunkt befüllen
               creatJSONDelayed();
               // eric2905 Ende ---------------------------------------------------------------
            }
             
             
            // Skriptstart
            // =============================================================================
             
            createDp();                 // Datenpunkte anlegen
            setTimeout(createOn,2000);  // Subscriptions anlegen
            setTimeout(main,    4000);  // Zum Skriptstart ausführen
             
             
            
            

            Folgende Fehlermeldung:

            javascript.0	2020-09-14 18:50:12.205	error	(15655) at Script.runInContext (vm.js:130:18)
            javascript.0	2020-09-14 18:50:12.205	error	(15655) at script.js.common.Komfort.Lueften:481:15
            javascript.0	2020-09-14 18:50:12.205	error	(15655) TypeError: DP is not a constructor
            javascript.0	2020-09-14 18:50:12.205	error	(15655) ^
            javascript.0	2020-09-14 18:50:12.205	error	(15655) var xdp = new DP(hunn);
            javascript.0	2020-09-14 18:50:12.204	error	(15655) script.js.common.Komfort.Lueften: script.js.common.Komfort.Lueften:481
            javascript.0	2020-09-14 18:50:12.198	error	(15655) at Script.runInContext (vm.js:130:18)
            javascript.0	2020-09-14 18:50:12.198	error	(15655) at script.js.common.Komfort.Lueften:93:13
            javascript.0	2020-09-14 18:50:12.197	error	(15655) at require (/opt/iobroker/node_modules/iobroker.javascript/lib/sandbox.js:390:36)
            javascript.0	2020-09-14 18:50:12.197	error	(15655) at require (internal/modules/cjs/helpers.js:72:18)
            javascript.0	2020-09-14 18:50:12.197	error	(15655) at Module.require (internal/modules/cjs/loader.js:1025:19)
            javascript.0	2020-09-14 18:50:12.196	error	(15655) script.js.common.Komfort.Lueften: Error: Cannot find module '/opt/iobroker/node_modules/iobroker.javascript/lib/../../dewpoint
            
            sigi234 Homoran 2 Replies Last reply Reply Quote 0
            • sigi234
              sigi234 Forum Testing Most Active @Dragon last edited by

              @David-Froebus

              dewpoint in JS Adapter als Modul eintragen

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

                @David-Froebus sagte in Skript läuft nicht... Lüften oder nicht...:

                '/opt/iobroker/node_modules/iobroker.javascript/lib/../../dewpoint

                im zweiten Skript steht es ja drin:

                @David-Froebus sagte in Skript läuft nicht... Lüften oder nicht...:

                benötigt in der Javascript das Modul "dewpoint"

                Master67 1 Reply Last reply Reply Quote 0
                • D
                  Dragon last edited by

                  vielen Dank für die Unterstützung. Zweites Skript läuft! Dann werde ich mal weiterbasteln! Tolle Community hier!

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

                    @Homoran sagte in [gelöst] Skript läuft nicht... Lüften oder nicht...:

                    @David-Froebus sagte in Skript läuft nicht... Lüften oder nicht...:

                    '/opt/iobroker/node_modules/iobroker.javascript/lib/../../dewpoint

                    im zweiten Skript steht es ja drin:

                    @David-Froebus sagte in Skript läuft nicht... Lüften oder nicht...:

                    benötigt in der Javascript das Modul "dewpoint"

                    Ähm..... Hust...... wie bekomme ich denn "dewpoint" genau als Modul eingebunden (naja Linux.. ich.. und Konsolen Befehle wir üben noch...)

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

                      @Master67

                      Instanzen->Javaskript->Zusätzliche NPM-Module

                      Master67 1 Reply Last reply Reply Quote 0
                      • Master67
                        Master67 @Adnim last edited by

                        @Adnim .... Ah... Danke 🙂

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

                        Support us

                        ioBroker
                        Community Adapters
                        Donate

                        377
                        Online

                        32.0k
                        Users

                        80.5k
                        Topics

                        1.3m
                        Posts

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