NEWS
[Skript] Absolute Feuchte berechnen
-
Hallo zusammen.
Ich habe jetzt einen TH-O in der Sauna montiert. Dort kommt es natürlich zu erheblichem Wassergehalt.
Was mich wundert ist nur die doch deutliche Abweichung zu dem Skript das ich auf der CCU (ebenfalls nach @paul53) erstellt habe und dem was hier herauskommt.
Wenn ich es noch richtig im Kopf habe kommt bei der CCU etwa 27g/kg heraus, während mit diesem Skript dann 37g/kg angegeben wird.
Die Daten für rel. Feuchte und Temperatur sind jeweils identisch (20%; 65-70°C)
Was ist jetzt richtig?
Gruß
Rainer `
Die 37g/kg kommen der Realität ziemlich nahe. Siehe dazu folgendes h1+x-Diagramm (leider nur bis 35g/kg): -
ich nehme an, dass das an den limitierten Fähigkeiten der CCU-Skripte liegt, dass sie komplexere Rechenformeln nicht umsetzen können.
Ja, daran liegt es.
Dann muss ich doch mal alles umbauen
Das Skript hatte ich schon vor längerer Zeit durch die Berechnung des CUxD Thermostat Device ersetzt. Man kann ja die Umstellung auf die Sauna beschränken, da das wohl der einzige Raum mit Temperaturen über 34 °C ist; evtl. noch die Außenfeuchte.
EDIT: Oder Du deaktivierst die Skripte in der CCU und beschreibst die Systemvariablen durch ioBroker.
-
Danke für alle Tipps!
@paul53:Oder Du deaktivierst die Skripte in der CCU und beschreibst die Systemvariablen durch ioBroker. `
Das war der Plan!und entsprechend alle OIDs in vis ändern.
Gruß
Rainer
-
Hier nur mal kurz zur Vollständigkeit der aktuelle Screenshot:
Damit man weiß wovon ich rede.
Noch nichts angepasst.
Gruß
Rainer
-
Hallo,
ich habe eben das Skript aus dem ersten Beitrag genommen und meine Datenpunkte angepasst.
Leider erhalte ich ständig Meldungen der Art:````
14:10:20.306 [error] javascript.1 script.js.Luftfeuchte.Lueften: Cannot use sync getState, use callback instead getState("hm-rpc.0.LEQ0594610.1.TEMPERATURE", function (err, state){});
14:10:22.809 [error] javascript.1 script.js.Luftfeuchte.Lueften: Cannot use sync getState, use callback instead getState("Raumklima.Raum.Aussen.Feuchtegehalt_Absolut", function (err, state){});Ich habe zu wenig Ahnung, um zu verstehen was ich dagegen tun kann.
-
Guten morgen zusammen,
danke an alle Beteiligten für das tolle Skript. Ich bewundere immer wieder, was so einige hier können. Es gibt viel viel zu lernen :idea:
Da ich (zufällig) gesehen habe, dass die v.0.6.4 hier im Thread verfügbar ist möchte ich anregen, diese auch vorne im ersten Posting anzupinnen, denn dort steht noch die 0.5.2 als aktueller Stand.
Ich habe bei mir (sicherlich etwas grobschlächtig in der Umsetzung) noch eingebaut, dass die Lüftungsempfehlung nicht nur in Abhängigkeit der absoluten Feuchtigkeit stattfindet sondern in Abhängigkeit der relativen Feuchtigkeit (innen je nach Raum), sofern denn überhaupt gelüftet werden sollte bzw kann, eine Empfehlung zum Lüften in unterschiedlichen Dringlichkeiten dargestellt wird;
d.h.
<40% = grau, es kann gelüftet werden aber ggf Luft unangenehm trocken
40-60% = grün, es kann gelüftet werden
60 <70 = orange, es sollte gelüftet werden
70 = rot, es muss gelüftet werden (Schimmelgefahr).
Zwei Fragen habe ich:
- ich nutze Homematic IP Thermostaten / Feuchtigkeitsmesser. Diese zeigen mir quer durch alle Räume immer ca 5% MEHR rH an, als meine alten TFA Dostmann Dinger.
Im Skript steht ja auch, man solle die Offsets kalibrieren - aber wie? Wie finde ich denn raus, was "echt" ist, um die Kalibrierung vorzunehmen? Habt ihr alle irgendwelche Profigeräte zu Hause, mit denen ihr messt?
- Zum Code des Skript wäre ich dankbar, wenn mich jemand zu diesem Abschnitt erleuchten könnte:
// Klimadaten in allen Räumen berechnen ##### VERSTANDEN function calcAll() { for (var raum in raeume) { calcDelayed(raum,delayRooms); // Räume verzögert nacheinander abarbeiten } } // finde anhand der Sensor ID einen zugeordneten Raum ##### BITTE ERKLÄREN. WIE KANN ICH DIES z.B: IN ANDEREN SKRIPTEN NUTZEN? function findRoom(sensor) { for (var raum in raeume) { if (raeume[raum].Sensor_TEMP == sensor) return raum; ###### IST "sensor" beliebig ersetzbar? if (raeume[raum].Sensor_HUM == sensor) return raum; } return null; } // Änderung eines Sensors (Temperatur oder Luftfeuchtigkeit) ####### GREIFT AUF FINDROOM ZU, VERSTANDEN function valChange(obj) { var raumname = findRoom(obj.id); if (raumname) { if (debug) log('**Änderung:' + raumname + ": " + obj.id + ": " + obj.state.val + '**'); calcDelayed(raumname,delayRooms); }
-
Wie finde ich denn raus, was "echt" ist, um die Kalibrierung vorzunehmen? `
Am einfachsten siehst du es demokratisch und nimmst die Werte der Mehrheit, zumal du ja nur vergleichen willst.Am ehesten den wahren Wert erfährst du indem du dir ein Haarhygrometer besorgst und es entsprechend kalbrierst (Feuchte Kammer)
Gruß
Rainer
-
Hallo Rainer,
danke. Es ist wirklich so, dass meine alten Sensoren grundsätzlich immer 5% weniger rH anzeigen.
Ich tendiere dazu, einfach die HmIP Werte zu glauben, schon aus Vorsichtsgründen (lieber einmal mehr lüften, gerade im Neubau…)
Kannst du mir bei meiner Code Frage auch weiterhelfen?
-
Moin,
was genau möchtest Du denn zu dem Codesegment wissen?
Bzgl. der Version im ersten Posting schaue ich mal nach, wenn ich wieder zu Hause bin (morgen Abend).
Gleiches gilt für den Offset.
Gruß,
Eric
Von unterwegs getippert
-
Hallo eric,
danke für deine Rückmeldung.
Ich hatte probiert, basierend auf dem Feuchtigkeitsscrip ein anderes Skript für Temperaturen zu bauen. Das man das ggf. alles in eins verarbeiten kann ist klar, aber ich lagere Dinge immer gerne erstmal aus, um zu lernen.
! ````
! // **************************************************************************************************
// SKRIPT ZUM AUSWERTEN DER IST-TEMPERATUREN UND SOLL TEMPERATUREN
// SCHREIBT WERTE IN ERZEUGTE DATENPUNKTE; DIESE KÖNNEN DANN IN ANDEREN SKRIPTEN GENUTZT WERDEN
// zusammengeschustert basierend auf dem Feuchtigkeitsskript von Solear, Paul53, ruhr70, eric2905
// **************************************************************************************************
! // ***********************************************************************
// Einstellungen
// ***********************************************************************
var debug = true; // Loggin mit true ein- bzw mit false ausschalten
// ***********************************************************************
// Parameter Variablen
// ***********************************************************************
var defaultMinTemp = 18.00; // Default TEMP_Minimum, wenn im Raum nicht angegeben (Auskühlschutz, tiefer soll eine Raumtemperatur durchs lüften nicht sinken)
var pfad = "Raumklima.Temperatur.Raum.";
var delayRooms = 500; // Zeit in MS Verzögerung zwischen Berechnungen der Räume
! // ***********************************************************************
// Räume anlegen (Räume werden in Array geschrieben und dann für createState genutzt)
// ***********************************************************************
var raeume = { // Keine Leerzeichen (Name wird als Datenpunktname verwendet!)// Erdgeschoss "FlurEG" : { "Temp_Ist" : "hm-rpc.1.000E57098F208F.1.ACTUAL_TEMPERATURE"/*HmIP Temperatur und Feuchtigkeitsmesser Flur HWR:1.ACTUAL_TEMPERATURE*/, "Temp_Soll" : "hm-rpc.1.000E57098F208F.1.SET_POINT_TEMPERATURE"/*HmIP Temperatur und Feuchtigkeitsmesser Flur HWR:1.SET_POINT_TEMPERATURE*/, "TEMP_Minimum" : defaultMinTemp, // oder Zieltemperatur in Form von: 20.00 angeben }, "Wohnzimmer" : { "Temp_Ist" : "hm-rpc.1.000E57098F207E.1.ACTUAL_TEMPERATURE"/*HmIP Temperatur und Feuchtigkeitsmesser Wohn Ess:1.ACTUAL_TEMPERATURE*/, "Temp_Soll" : "hm-rpc.1.000E57098F207E.1.SET_POINT_TEMPERATURE"/*HmIP Temperatur und Feuchtigkeitsmesser Wohn Ess:1.SET_POINT_TEMPERATURE*/, "TEMP_Minimum" : defaultMinTemp, // oder Zieltemperatur in Form von: 20.00 angeben }, "Gaestebad" : { "Temp_Ist" : "hm-rpc.0.NEQ1640752.4.ACTUAL_TEMPERATURE"/*Gaestebad Thermostat:4.ACTUAL_TEMPERATURE*/, "Temp_Soll" : "hm-rpc.0.NEQ1640752.4.SET_TEMPERATURE"/*Gaestebad Thermostat:4.SET_TEMPERATURE*/, "TEMP_Minimum" : defaultMinTemp, // oder Zieltemperatur in Form von: 20.00 angeben }, // Obergeschoss "Schlafzimmer" : { "Temp_Ist" : "hm-rpc.1.000E57098F20A8.1.ACTUAL_TEMPERATURE", "Temp_Soll" : "hm-rpc.1.000E57098F20A8.1.SET_POINT_TEMPERATURE"/*HmIP Temperatur und Feuchtigkeitsmesser Schlafzimmer:1.SET_POINT_TEMPERATURE*/, "TEMP_Minimum" : defaultMinTemp, // oder Zieltemperatur in Form von: 20.00 angeben }, "BadOG" : { "Temp_Ist" : "hm-rpc.1.000393C99A1CDE.1.ACTUAL_TEMPERATURE"/*Badezimmer oben Thermostat Handtuchhalter:1.ACTUAL_TEMPERATURE*/, "Temp_Soll" : "hm-rpc.1.000393C99A1CDE.1.SET_POINT_TEMPERATURE"/*Badezimmer oben Thermostat Handtuchhalter:1.SET_POINT_TEMPERATURE*/, "TEMP_Minimum" : defaultMinTemp, // oder Zieltemperatur in Form von: 20.00 angeben }, };
! // ***********************************************************************
// Raumvariablen --> werden als Unterpunkte in die createState geschrieben
// ***********************************************************************
! var raumDatenpunkte = {
"tempSoll" : {
"DpName" : "Soll_Temperatur",
"init": 0,
"dp": {
"name": 'Soll Temperatur',
"desc": 'manuell eingestellt oder via HM',
"type": 'number',
"role": 'value',
"unit": '°C'
}
},"tempIst" : { "DpName" : "Ist_Temperatur", "init": 0, "dp": { "name": 'Aktuelle Temperatur', "desc": 'fakt', "type": 'number', "role": 'value', "unit": '°C' } }
};
// ***********************************************************************
// Funktionen
// ***********************************************************************// finde anhand der Sensor ID einen zugeordneten Raum <<<<<<<<<<<<<<<<<< HIER DENKE ICH LIEGT MEIN FEHLER!!
function findRoom(sensor) {
for (var raum in raeume) {
if (raeume[raum].Temp_Ist == sensor) return raum;
if (raeume[raum].Temp_Soll == sensor) return raum;
}
return null;
}
// Änderung eines Sensors (Temperatur ISt oder Soll) <<<<<<<<<<<<<<<<< oder hier, aber diese benötigt ja die Function findRoom, d.h. wahrscheinlich dort mein Fehler
function valChange(obj) {
var raumname = findRoom(obj.id);
if (raumname) {
if(debug) log('Änderung:' + raumname + ": " + obj.id + ": " + obj.state.val + '');
calcDelayed(raumname, delayRooms);
}
}
// Berechnung verzögern, damit die Räume nacheinander abgearbeitet werden
function calcDelayed(raum, delay) {
setTimeout(function () {
calc(raum);
}, delay || 0);
}// Berechnung
function calc(raum) {
var ist = getState(raeume[raum].Temp_Ist).val; // Temperatur IST auslesen
var soll = getState(raeume[raum].Temp_Soll).val; // Temperatur SOLL auslesenif(debug) log("Temperatur IST in "+raum +" beträgt "+ist); if(debug) log("Temperatur Soll in "+raum +" beträgt "+soll); var idSoll = pfad + raum + "."+raumDatenpunkte['tempSoll'].DpName; // Pfad für nachfolgendes setState zusammenbauen var idIst = pfad + raum + "."+raumDatenpunkte['tempIst'].DpName; setState(idSoll, soll); setState(idIst, ist);
}
// Berechnung
function calcAll() {
for (var raum in raeume) {
calcDelayed(raum,delayRooms); // Räume verzögert nacheinander abarbeiten
}
}// Anlegen der Datenpunkte --> legt Datenpunkte an mit Raum(name) und den Sub-Dps
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);
if(debug) log("neuer Datenpunkt: " + name);
}
}
}! // START
createDp(); // Datenpunkte anlegen
setTimeout(calcAll, 3000); // Zum Skriptstart ausführen
! ````Leider funktioniert das nicht, es wird nicht auf Änderungen der IST oder Soll Temperatur reagiert. Diese sollten ja eigentlich als Auslöser dienen.
D.h. irgendwie muss ich einen (Verständnis-)fehler in den Funktionen findRoom oder valChange (welche ja findRoom nutzt) haben; aber diesen finde ich nicht.
Mein Problem besteht sicherlich auch darin, dass ich in Sachen Javascript trotz mehrfachem Lesens des "Eloquent JS" Buchs (bzw Teilen davon), die Funktionen wohl noch nicht zu 100% durchsteige.
z.B. oben im "findRoom" steht == sensor <– was ist das "sensor" hier?
Danke für deine Hilfe, ich will das unbedingt verstehen!
-
Moin Leute,
Ich hab Eurer Skript mal bei mir eingesetzt und wunder mich etwas was die Lüftungsempfehlung angeht.
Hab die Daten die letzten Tage beobachtet und hab seid gut einer Woche eine dauerhafte Lüftungsempfehlung.
alle Datenpunkt werden gelistet und die Min/Maxwerte hab ich eingestellt.
Wann emfiehlt er denn nicht zu lüften ??? Irgendwas passt doch da nicht.
Grüße
Thomas
-
Solange die absolute Feuchte draußen niedriger ist wie drinnen, wird es wohl eine Lüftungsempfehlung geben.
Genauere Infos stehen in den Datenpunkten (unter Objekte nachsehen).
Gruß,
Eric
Von unterwegs getippert
-
Moin Leute,
Ich hab Eurer Skript mal bei mir eingesetzt und wunder mich etwas was die Lüftungsempfehlung angeht.
Hab die Daten die letzten Tage beobachtet und hab seid gut einer Woche eine dauerhafte Lüftungsempfehlung.
alle Datenpunkt werden gelistet und die Min/Maxwerte hab ich eingestellt.
Wann emfiehlt er denn nicht zu lüften ??? Irgendwas passt doch da nicht.
Grüße
Thomas `
Wie Eric schon sagt: es geht (korrekt) allein um die absolute Feuchte.
Ich habe daher bei mir das Skript so erweitert, dass auch die relative Feuchte berücksichtigt wird und daraus dann vier Stufen abgeleitet:
-
abs. Feuchte draußen > abs F. innen: VERBOT
-
abs. Feuchte draußen < innen und innen rel. Feuchte innen <58%: kann lüften
-
abs. Feuchte draußen < innen und innen rel. Feuchte innen >58% <70%: SOLLTE lüften
-
abs. Feuchte draußen < innen und innen rel. Feuchte innen >= 70%: MUSS LÜFTEN, Schimmelgefahr
nächster Schritt wäre, die Temperatur zu berücksichtigen und quasi einen "trade off" einzubauen im Sinne von "erlaube Erhöhung der rel. Luftfeuchtigkeit innen im Gegenzug zum Lüften bis zum Stand von XX".
Das steht aber noch auf meiner Todo Liste. Meine Frau ist leider nicht sehr offen für "ich muss da mal was testen"; es soll bitte immer einfach alles funktionieren. :lol: :lol:
-
-
Das klingt nach einer sinnvollen Ergänzung
Es kann natürlich sein, daß die absoluten Werte sich immer passend verändert haben und so Lüften sinnvoll ist.
Gibt es eigentlich eine gute Alternative zum automatischen Fensteröffnen?
Das HM Fensterkippmodul finde ich etwas zu teuer ehrlich gesagt.
-
Hi,
Gibt es eigentlich eine gute Alternative zum automatischen Fensteröffnen? `
Frau? Kinder? :lol: :lol: (Sorry, ich konnte nicht anders)Gruß,
Eric
-
Gibt es eigentlich eine gute Alternative zum automatischen Fensteröffnen?
Das HM Fensterkippmodul finde ich etwas zu teuer ehrlich gesagt. `
Das bringt auch nicht so sehr viel, weil es nur ganz wenig ankippt. Da braucht man dann schon 2 davon für einen Durchzug.
Alternative sind: Entfeuchter (laut und mit 400 - 1000 Watt Verbrauch/h sehr teuer im Unterhalt) oder solche Dachfenstermotoren mit Ketten oder Gewindestange, damit habe ich 2 Kellerfenster ausgestattet und sie werden über HM-Jalousieaktoren angesteuert.
-
@ Eric, Ja der war gut
Die sagen aber leider immer: Mach du mal
Ja, die Kettenantriebe. Muß ich mir mal genauer anschauen.
Aber ich denke, da braucht man auch 2 Stk für den Durchzug von.
-
Alternativ ein Rohrventilator durch die Wand oder Fensterventilatoren.
Ich hatte erst vor, mir zwei größere Fensterventilatoren in den Kellerfenstern einzubauen. Aber einerseits muss man isolierte Fenster dafür einschneiden (oder bei Rohrventilatoren einen Wanddurchbruch machen) und andererseits benötigen die eben auch 10-20 Watt im Betrieb. Vorteile sind aber, man kann auch zwangslüften bei Windstille (gerade in warmen Sommernächten damit man möglichst viel von der kühlen Luft draußen durchziehen lässt) und man hat einen besseren Einbruchsschutz als mit angekippten Fenstern. Nachteile sind die Kosten (im Betrieb) und die Anschaffungskosten, gute, ausreichend große und leise kosten gerne über 200 € das Stück.
-
Also bei Ventilatoren und Co sind wir ja schon fast bei einer Raumlüftungsanlage
Die gibt es zwar auch als Nachrüstung für Altbauten, aber ich denke das rechnet sich dann erst recht nicht mehr.
Und 100er Löcher ins Haus bohren mit einfachen Lüftungsgeräten? Ob das so zielführend ist weis ich nicht.
Da hat man ja auch immer direkt Kältbrücken, da die Ventilatoren nicht 100% schliessen.
Einfache System mit Heizung bzw. Wärmerückgewinnung verbrauchen dann auch wieder ordentlich Strom.
Ist halt ein komplexes Thema.
-
Hi zusammen, ich nutze seit ein paar Tagen das Skript 0.6.4.
nur verstehe ich nicht ganz warum ich in der Garage nicht lüften sollte, der Feuchtegehalt in der Garage ist höher wie Außen, könnte mit da jemand weiterhelfen?
Gruß Christian
! //
! // Raumklima - v0.6.4
! //
! // 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 ... alpie.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/bauph ... luften.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
! // 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 = 40.01; // 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 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.2; // Default 0.5, Hysterese Mindesttemperatur (Auskühlschutz). Innerhalb dieser Deltatemperatur bleibt die alte Lüftungsempfehlung für den Auskühlschutz bestehen.
! var hysEntfeuchten = 0.3; // 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!
! // Beispiel Innensensor:
! /
! "Sensor_TEMP" : "hm-rpc.0.KEQ0175977.1.TEMPERATURE", // Datenpunkt Temperatur für den Raum
! "Sensor_HUM" : "hm-rpc.0.KEQ0175977.1.HUMIDITY", // Datenpunkt Luftfeuchtigkeit für den Raum
! "Sensor_TEMP_OFFSET" : 0.0, // Kalibrierung des Messwertes durch Offset
! "Sensor_HUM_OFFSET" : 0, // Kalibrierung des Messwertes durch Offset
! "TEMP_Minimum" : defaultTemp, // defaultTemp, oder Zieltemperatur in Form von: 20.00 angeben
! "Aussensensor" : "Balkon" // Names des dazugehörigen Außensensors (Name muss in der Schreibweise übereinstimmen)
! }
! /
! // Beispiel Aussensensor:
! /
! "weatherunderground" : {
! "Sensor_TEMP" : "weatherunderground.0.current.temp_c",
! "Sensor_HUM" : "weatherunderground.0.current.relative_humidity",
! "Sensor_TEMP_OFFSET" : 0.0,
! "Sensor_HUM_OFFSET" : 0
! }
! /
! var raeume = { // Keine Leerzeichen (Name wird als Datenpunktname verwendet!)
! // Sensoren Aussen
! "Aussen" : {
! "Sensor_TEMP" : "mihome.0.devices.sensor_ht_158d000171973a.temperature"/Temperature/ /Balkon gr. Klima:1.TEMPERATURE/,
! "Sensor_HUM" : "mihome.0.devices.sensor_ht_158d000171973a.humidity"/Humidity/ /Balkon gr. Klima:1.HUMIDITY/,
! "Sensor_TEMP_OFFSET" : 0.0,
! "Sensor_HUM_OFFSET" : 0
! },
! // Sensoren Innen
! "Garage" : {
! "Sensor_TEMP" : "mqtt.0.Wemo1.Umwelt_Garage.Temperatur"//Wemo1/Umwelt Garage/Temperatur/ /Bad Lana.TEMPERATURE/,
! "Sensor_HUM" : "mqtt.0.Wemo1.Umwelt_Garage.Feuchtigkeit"//Wemo1/Umwelt Garage/Feuchtigkeit/ /Bad Lana.HUMIDITY/,
! "Sensor_TEMP_OFFSET" : 0.0,
! "Sensor_HUM_OFFSET" : 0,
! "TEMP_Minimum" : 5.00, // oder Zieltemperatur in Form von: 20.00 angeben
! "Aussensensor" : "Aussen"
! },
! "HWR" : {
! "Sensor_TEMP" : "mihome.0.devices.sensor_ht_158d000170d50f.temperature"/Temperature/ /Arbeitszimmer Thermostat.TEMPERATURE/,
! "Sensor_HUM" : "mihome.0.devices.sensor_ht_158d000170d50f.humidity"/Humidity/ /Arbeitszimmer Thermostat.HUMIDITY/,
! "Sensor_TEMP_OFFSET" : 0.0,
! "Sensor_HUM_OFFSET" : 0,
! "TEMP_Minimum" : 18.00,
! "Aussensensor" : "Aussen"
! }
! };
! // =============================================================================
! // =============================================================================
! // 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 2 Auskühlschutz erfüllt (Innentemperatur soll nicht unter Minimumteperatur fallen)',
! "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'
! }
! },
! "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_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) ? 'true' : 'false';
! 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,((at)/(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/(kmolK) (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 , t); // Sensor Temperatur inkl. Offset
! setState(idrh , 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("–---- " + raum + " ------- Aussen, keine Lüftungsempfehlung -----------");
! 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)
! //if(typeof raeume[raum].TEMP_Minimum !=="undefined") {
! if(typeof raeume[raum].TEMP_Minimum == "number") {
! mi = raeume[raum].TEMP_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 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 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";
! setState(idLueftenB1,b1lp);
! setState(idLueftenB2,b2lp);
! setState(idLueftenB3,b3lp);
! // 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 b1lnText = "Entfeuchten: Außenluft ist zu feucht";
! var b2lnText = "Kühlen: Außentemperatur zu warm";
! var b3lnText = "Auskühlschutz: Innentemperatur niedriger als Mindestraumtemperatur";
! // Logik:
! //--------------------------------------------------------------------------
! if (b1lp && b2lp && b3lp) {
! // 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 + ': Lüftungsempfehlung');
! } else if (b1ln || b2ln || b3ln) {
! // Fenster zu. Ein Ausschlusskriterium reicht für die Empfehlung "Fenster zu".
! lueftenText = "Fenster zu:
";
! if (b1ln) lueftenText += b1lnText + "
";
! if (b2ln) lueftenText += b2lnText + "
";
! if (b3ln) lueftenText += b3lnText + "
";
! setState(idLueften, false);
! setState(idLueftenHys,false);
! if (debug) log(raum + ': Empfehlung Fenster zu');
! } else {
! // Hysterese. Keine Änderung der bisherigen Empfehlung.
! if (debug) log(raum + ': im Bereich der Hysterese (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)
! Lüften:
! wenn abs. Aussenfeuchte < abs. Innenfeuchte - Hysterese (Entfeuchten)
! UND Aussentemperatur < Innentemperatur - Hysterese (Kühlen)
! UND Innentemperatur >= Raumtemperaturminimum + Hysterese (Auskühlschutz)
! */
! // 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
! // nicht lüften (oder):
! // #1 - Außenluft ist zu feucht
! // #2 - Außentemperatur zu warm
! // #3 - Innentemperatur niedriger als Mindestraumtemperatur
! 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);
! 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("------ " + raum + " ------- Aussensensor: " + aussen + " -----------");
! }
! //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('Änderung:' + raumname + ": " + obj.id + ": " + obj.state.val + '');
! 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
2113_unbenannt.png