NEWS
Set State ( Timestamp ) Zeitverzögert [gelöst].
-
Servus,
bin der neue und habe gleich mal eine frage zum Java Script. Folgendes Problem:
Ich habe ein Script was alle 3 min durch schedule gestartet wird. Das Script ist für eine Dosieranlage.
Ich möchte in dem Script einen Aktor zeitverzögert auswerten. Sprich ich möchte das eine Pumpe ( Aktor ) erst mindestestens 5 Minuten läuft bei einmaligen einschalten bevor die Dosierung überhaupt anläuft. Das Problem dabei ist das das Script alle 3 Minuten neu startet.
Wie mache ich das am besten ?
Grüße Timo
-
Wo ist der Script code ?
@kasperfunsurfer:Ich habe ein Script was alle 3 min durch schedule gestartet wird. `
Weshalb ? -
// IDS der Elemente IOBroker var idChlorAcidAlternative = "hm-rega.0.Dosierung_Umschaltung_Chlor/Acid"; var idPHWert = "parser.0.Pool-ph"; var idORPWert = "parser.0.Pool-orp"; var idPumpeChlor = "hm-rpc.0.MEQ0454385.1.STATE"; var idPoolPumpe = "hm-rpc.1.NEQ1810472.13.STATE"; var idDosingAuto = "hm-rega.0.25168"; // Zielwerte var idPHZielwert = "7,1"; var orpZielwert = "480"; // Script alle 2 Minuten Starten schedule ("*/2 * * * *", function() { // interne Variablen var dosingOK = true; // Steuerung Chlor Acid im Wechsel var aktuellerZustand = getState(idChlorAcidAlternative).val; if ( aktuellerZustand === "chlor") { setState(idChlorAcidAlternative, "acid"); } else { setState(idChlorAcidAlternative, "chlor"); } // Check, ob die Pumpe aktuell läuft if (getState(idPoolPumpe).val === false) { log('Pool Pumpe läuft nicht -> keine Dosierung'); dosingOK = false; else { dosingOK = true; } // Check, ob Dosierung im Automatikmodus läuft if (getState(idDosingAuto).val === false) { log('Dosierung nicht im Automatik Modus -> keine Dosierung'); dosingOK = false; } else { dosingOK = true; } // --------------- // Steuerung Chlor // --------------- if (aktuellerZustand === "chlor" && dosingOK === true) { var orpIstWert = getState(idORPWert).val; var dosingTime = 0; // DosingTime abhängig von ORP-Differenz festlegen if (orpIstWert < orpZielwert) { if (orpZielwert - orpIstWert > 100) { dosingTime=80; } else if (orpZielwert - orpIstWert < 5) { dosingTime=10; } else if (orpZielwert - orpIstWert < 10) { dosingTime=20; } else if (orpZielwert - orpIstWert < 20) { dosingTime=35; } else if (orpZielwert - orpIstWert < 40) { dosingTime=45; } else if (orpZielwert - orpIstWert < 60) { dosingTime=50; } else if (orpZielwert - orpIstWert < 100) { dosingTime=60; } } // Chlor-Dosierung starten if (dosingTime > 0 && getState(idPumpeChlor).val === false) { // log('Dosierzeit',dosingTime); //chlor_time+=$dosing_time; //canister_chlor_time+=$dosing_time; setState(idPumpeChlor, true); setTimeout(function(){setState(idPumpeChlor, false)},dosingTime*1000); } } });
-
Servus,
vielleicht so verständlicher. Möchte das die Pumpe erst 5 Minuten läuft bevor die Dosierung startet. Wie? aber nicht wie ich das realisieren soll.
Bin leider nicht der Java Profi.
Grüße
-
Wie ich es sehe, soll die Dosierung nur laufen, wenn der Datenpunkt "hm-rega.0.25168" (Auto) auf wahr ist ? Dann sollte auf diesen Datenpunkt getriggert werden.
function Dosierung() { // Hier Dosierung ausführen } on(idDosingAuto, function(dp) { if(dp.state.val) { setState(idPoolPumpe, true); // Poolpumpe ein } else { setState(idPumpeChlor, false); setState(idPoolPumpe, false); } }); on({id: idPoolPumpe, val: true, ack: true}, function() { // Poolpumpe hat eingeschaltet setTimeout(Dosierung, 300000); // Funktion Dosierung() wird nach 5 Minuten aufgerufen });
Für das abwechselnde Dosieren Chlor / Acid sollte man innerhalb der Funktion Dosierung() dann setInterval() verwenden.
Bitte Javascriptcode in Code tags posten !
-
Ahh Ok danke werde ich mal ausprobieren. Danke schön
Und weil es so leicht war noch eine frage
Wie Prüfe ich in dem Script ob die Werte die ich mit dem Parser Adapter abrufe nicht älter als 3 Minuten sind. Sprich der Timestamp darf nicht äler als 3 Minuten sein.
Grüße
-
der Timestamp darf nicht äler als 3 Minuten sein. `
if(Date.now() - getState(idPHWert).ts < 180000)
-
Danke schön……
Für das abwechselnde Dosieren Chlor / Acid sollte man innerhalb der Funktion Dosierung() dann setInterval() verwenden. `
Wie meinst du das ?
-
Wie meinst du das ? `
Hast Du nicht deshalb das Schedule verwendet, um abwechselnd alle 2 Minuten zwischen Chlor und Acid umzuschalten ? Innerhalb einer Funktion verwendet man besser https://www.a-coding-project.de/ratgeber/javascript/setinterval.var acid = false; setState(idChlorAcidAlternative, "chlor"); timer = setInterval(function() { acid = !acid; if(acid) setState(idChlorAcidAlternative, "acid"); else setState(idChlorAcidAlternative, "chlor"); }, 120000); // alle 2 Minuten
Allerdings sollte man den Intervall-Timer zum Ende der Dosierung stoppen.
clearInterval(timer);
-
Wie ich es sehe, soll die Dosierung nur laufen, wenn der Datenpunkt "hm-rega.0.25168" (Auto) auf wahr ist ? Dann sollte auf diesen Datenpunkt getriggert werden.
function Dosierung() { // Hier Dosierung ausführen } on(idDosingAuto, function(dp) { if(dp.state.val) { setState(idPoolPumpe, true); // Poolpumpe ein } else { setState(idPumpeChlor, false); setState(idPoolPumpe, false); } }); on({id: idPoolPumpe, val: true, ack: true}, function() { // Poolpumpe hat eingeschaltet setTimeout(Dosierung, 300000); // Funktion Dosierung() wird nach 5 Minuten aufgerufen });
Für das abwechselnde Dosieren Chlor / Acid sollte man innerhalb der Funktion Dosierung() dann setInterval() verwenden.
Bitte Javascriptcode in Code tags posten ! `
Hallo nochmal,
erstmal vielen Danke für die Hilfe.
So wie es oben dargestellt ist läuft es nicht. Ich möchte das die Pool Pumpe 5 Minuten in Betrieb ist und dann erst Dosierung stattfindet. Also die Dosierung soll erst nach ablauf der 5 Minuten anfangen.
So wie es jetzt ist Starte ich Dosierung und Pumpe läuft an. Das wäre falsch. Da die Pumpe nicht nur für die Dosierung genutzt wird.
// Check, ob die Pumpe aktuell läuft
if (getState(idPoolPumpe).val === false) {
log('Pool Pumpe läuft nicht -> keine Dosierung');
dosingOK = false;
else {
dosingOK = true;
// Außerdem muss die Pumpe schon mindestens 5Minuten laufen
??
??
??
Grüße
-
So wie es jetzt ist Starte ich Dosierung und Pumpe läuft an. Das wäre falsch. Da die Pumpe nicht nur für die Dosierung genutzt wird. `
Das wusste ich nicht. Dann sollte es so funktionieren (nicht getestet):// IDS der Elemente IOBroker const idChlorAcidAlternative = "hm-rega.0.Dosierung_Umschaltung_Chlor/Acid"; const idPHWert = "parser.0.Pool-ph"; const idORPWert = "parser.0.Pool-orp"; const idPumpeChlor = "hm-rpc.0.MEQ0454385.1.STATE"; const idPoolPumpe = "hm-rpc.1.NEQ1810472.13.STATE"; const idDosingAuto = "hm-rega.0.25168"; // Zielwerte const idPHZielwert = "7,1"; const orpZielwert = 480; // interne Variablen var timer; var acid; var dosingOK = false; setTimeout(function() { dosingOK = getState(idPoolPumpe).val; }, 300000); function Dosierung() { acid = false; setState(idChlorAcidAlternative, "chlor"); timer = setInterval(function() { // Umschaltung Chlor / Acid acid = !acid; if(acid) setState(idChlorAcidAlternative, "acid"); else setState(idChlorAcidAlternative, "chlor"); // DosingTime abhängig von ORP-Differenz festlegen var orpIstWert = parseFloat(getState(idORPWert).val); // parseFloat, falls ORPWert ein String ist var dosingTime = 80; if (orpZielwert - orpIstWert < 100) dosingTime =60; if (orpZielwert - orpIstWert < 60) dosingTime = 50; if (orpZielwert - orpIstWert < 40) dosingTime = 45; if (orpZielwert - orpIstWert < 20) dosingTime = 35; if (orpZielwert - orpIstWert < 10) dosingTime = 20; if (orpZielwert - orpIstWert < 5) dosingTime = 10; if (orpZielwert <= orpIstWert) dosingTime = 0; // Chlor-Dosierung starten if (dosingOK && dosingTime > 0 && !getState(idPumpeChlor).val) { // log('Dosierzeit',dosingTime); //chlor_time+=$dosing_time; //canister_chlor_time+=$dosing_time; setState(idPumpeChlor, true); setTimeout(function(){ setState(idPumpeChlor, false); }, dosingTime * 1000); } }, 120000); // alle 2 Minuten } // Skriptstart if(getState(idDosingAuto).val) Dosierung(); on(idDosingAuto, function(dp) { if(dp.state.val) { Dosierung(); } else { if(timer) clearInterval(timer); setState(idPumpeChlor, false); } }); on(idPoolPumpe, function(dp) { if(dp.state.val) { setTimeout(function() { dosingOK = true; }, 300000); // Dosierung wird nach 5 Minuten freigegeben } else { dosingOK = false; setState(idPumpeChlor, false); } });
EDIT: Korrektur in Zeile 65: {
-
Servus,
habe gerade das Script probiert. Leider mit folgender Fehlermeldung: javascript.0 script.js.Skript1 compile failed: at script.js.Skript1:68
Wie würde das Script aussehen wenn ich jetzt noch eine Abfrage vom Ph Wert mache das der übertragene Wert vom Parser nicht älter als 2 Minuten ist sowie eine Abfrage das der der PH Wert im Bereich größer 6,5 und kleiner 8,0 liegt ?
-
compile failed: at script.js.Skript1:68 `
Habe es im Skript korrigiert: Es fehlte eine öffnende geschweifte Klammer in Zeile 65.if(dp.state.val) { setTimeout(function() {
-
eine Abfrage vom Ph Wert mache das der übertragene Wert vom Parser nicht älter als 2 Minuten ist sowie eine Abfrage das der der PH Wert im Bereich größer 6,5 und kleiner 8,0 liegt ? `
Was soll passieren, wenn beide Bedingungen erfüllt sind ?Was soll passieren, wenn eine der beiden Bedingungen nicht erfüllt ist ?
-
// IDS der Elemente IOBroker var idChlorAcidAlternative = "hm-rega.0.Dosierung_Umschaltung_Chlor/Acid"; // Umschaltung PH/Chlor var idPHWert = "parser.0.Pool-ph"; // PH Wert von Parser Adapter var idORPWert = "parser.0.Pool-orp"; // Chlor Wert von Parser Adapter var idPumpeChlor = "hm-rpc.1.NEQ1810472.19.STATE"; // Pumpe Chlor Aktor var idPumpePH = "hm-rpc.1.NEQ1810472.18.STATE"; // Pumpe PH Aktor var idDayTimeChlor = "hm-rega.0.Chor_Zeit_Tagesmenge"; // Tagesmenge Chlor var idDayTimePH = "hm-rega.0.Ph_Zeit_Tagesmenge"; // Tagesmenge PH var idDayTimeChlorOver = "hm-rega.0.25721"; // Maximale Tagesmenge Chlor ereicht var idDayTimePHOver = "hm-rega.0.25720"; // Maximale Tagesmenge PH ereicht var idPoolPumpe = "hm-rpc.1.NEQ1810472.13.STATE"; // Abfrage Pool Pumpe var idPoolPumpeAuto = "hm-rega.0.19785"; // Abfrage Pool Auto/Hand var idDosingAuto = "hm-rega.0.25168"; // Abfrage Dosierung Auto/Hand var idStoerung = "hm-rega.0.25719"; // Störung // Zielwerte var PHZielwert = "7.05"; // Ph Zielwert var orpZielwert = "440"; // Chlor Zielwert var maxTimeChlor = "80"; // Chlor Tagesmenge in sec. var maxTimePH = "80"; // PH Tagesmenge in sec. // Script alle 3 Minuten Starten schedule ("*/1 * * * *", function() { // interne Variablen var dosingOK = true; var dosingTime = 0; // Steuerung Chlor/PH im Wechsel var aktuellerZustand = getState(idChlorAcidAlternative).val; if ( aktuellerZustand === "chlor") { setState(idChlorAcidAlternative, "ph"); } else { setState(idChlorAcidAlternative, "chlor"); } //Dosierung ist noch nicht abgeschaltet??? if ((getState(idPumpeChlor).val === true) || (getState(idPumpePH).val === true)) { log('Dosierung läuft noch?'); if (getState(idStoerung).val === false) { setState(idStoerung, true); } return; } // Check, ob Pumpe im Automatik Betrieb if (getState(idPoolPumpeAuto).val === false) { log('PoolPumpe nicht im Automatik Betrieb -> keine Dosierung'); dosingOK = false; return; } // Check, ob die Pumpe aktuell läuft if (getState(idPoolPumpe).val === false) { log('Pool Pumpe läuft nicht -> keine Dosierung'); dosingOK = false; return; } // Außerdem muss die Pumpe schon mindestens 5Minuten laufen getState(idPoolPumpe).lc; // Check, ob Dosierung im Automatikmodus läuft if (getState(idDosingAuto).val === false) { log('Dosierung nicht im Automatik Modus -> keine Dosierung'); dosingOK = false; return; } // ***************** // // *Steuerung Chlor* // // ***************** // if (aktuellerZustand === "chlor" && dosingOK === true) { var orpIstWert = getState(idORPWert).val; // Check, ob die Tagesdosierung Chlor noch nicht überschritten if (getState(idDayTimeChlor).val >= maxTimeChlor) { log('Tagesmenge Chlor erreicht -> keine Dosierung'); dosingOK = false; } // Ist der Zielwert bereits überschritten? if (orpIstWert > orpZielwert) { log('Dosierung Target ORP erreicht ->keine Dosierung'); dosingOK = false; // in die globale Variable speichern if (getState(idDayTimeChlorOver).val === false) { setState(idDayTimeChlorOver, true); } } // DosingTime abhängig von ORP-Differenz festlegen if (orpIstWert < orpZielwert) { if (orpZielwert - orpIstWert > 100) { dosingTime=80; } else if (orpZielwert - orpIstWert < 5) { dosingTime=10; } else if (orpZielwert - orpIstWert < 10) { dosingTime=20; } else if (orpZielwert - orpIstWert < 20) { dosingTime=35; } else if (orpZielwert - orpIstWert < 40) { dosingTime=45; } else if (orpZielwert - orpIstWert < 60) { dosingTime=50; } else if (orpZielwert - orpIstWert < 100) { dosingTime=60; } } // Chlor-Dosierung starten if (dosingTime > 0 && getState(idPumpeChlor).val === false && dosingOK) { // Fortschreibung Tagesdosierung Chlor var chlor_time = getState(idDayTimeChlor).val; chlor_time += dosingTime; setState(idDayTimeChlor, chlor_time); // Dosierung starten und Timer setzen setState(idPumpeChlor, true); setTimeout(function(){setState(idPumpeChlor, false)},dosingTime*1000); } } // ************** // // *Steuerung PH* // // ************** // if (aktuellerZustand === "ph" && dosingOK === true) { var phIstWert = getState(idPHWert).val; // Check, ob die Tagesdosierung PH noch nicht überschritten if (getState(idDayTimePH).val >= maxTimePH) { log('Tagesmenge PH erreicht -> keine Dosierung'); dosingOK = false; } // Ist der Zielwert bereits überschritten? if (phIstWert > PHZielwert) { log('Dosierung Target PH erreicht ->keine Dosierung'); dosingOK = false; // in die globale Variable speichern if (getState(idDayTimePHOver).val === false) { setState(idDayTimePHOver, true); } } // DosingTime abhängig von PH Differenz festlegen if (phIstWert < PHZielwert) { if (PHZielwert - phIstWert > 0.1) { dosingTime = 35; } else if (PHZielwert - phIstWert < 0.03) { dosingTime = 10; } else if (PHZielwert - phIstWert < 0.05) { dosingTime = 20; } else if (PHZielwert - phIstWert < 0.1) { dosingTime = 25; } } // PH-Dosierung starten if (dosingTime > 0 && getState(idPumpePH).val === false && dosingOK) { // Fortschreibung Tagesdosierung PH var ph_time = getState(idDayTimePH).val; ph_time+=dosingTime; setState(idDayTimePH, ph_time); // Dosierung starten und Timer setzen setState(idPumpePH, true); setTimeout(function(){setState(idPumpePH, false)},dosingTime*1000); } } }); // Reset der TagesCounter bei einem Datumswechsel schedule ("30 0 0 * * *", function() { var chlor_time = 0; setState(idDayTimeChlor, chlor_time); var ph_time = 0; setState(idDayTimePH, ph_time); // Reset Kennzeichen Tagesmenge überschritten setState(idDayTimeChlorOver, false); setState(idDayTimePHOver, false); });
Habe mir das quasi so gedacht das wenn die Werte nicht in dem Bereich liegen das die Variable Störung ausgelöst wird und nicht mehr weiter dosiert wird.
Was mir noch aufgefallen ist. Das der Timer von der Pumpe also die 5 Minuten nicht wieder zurückgesetzt wird sobald die Pumpe ausgeht. So müßte es sein. Setzt sich erst zurück wenn ich Dosierung Automatik auf Hand setzte. Abfragen sind eigentlich immer auf Automatik und werden nur zum Pool säubern auf Hand geschaltet sonst nicht.
Nochmal vielen Danke für deine Mühe !!!!!
-
// Zielwerte var PHZielwert = "7.05"; // Ph Zielwert var orpZielwert = "440"; // Chlor Zielwert var maxTimeChlor = "80"; // Chlor Tagesmenge in sec. var maxTimePH = "80"; // PH Tagesmenge in sec. ```` `
Du weist den Werten Strings zu, verwendest sie aber für Berechnungen (Subtraktion / Vergleich mit Zahl). Das funktioniert nicht !!
-
Guten Morgen,
dann lass uns bei dem Script bleiben was du geändert hast. Das andere übersteigt meinen Horizont.
// IDS der Elemente IOBroker const idChlorAcidAlternative = "hm-rega.0.Dosierung_Umschaltung_Chlor/Acid"; const idPHWert = "parser.0.Pool-ph"; const idORPWert = "parser.0.Pool-orp"; const idPumpeChlor = "hm-rpc.0.MEQ0454385.1.STATE"; const idPoolPumpe = "hm-rpc.1.NEQ1810472.13.STATE"; const idDosingAuto = "hm-rega.0.25168"; const idStörung = ?????????? // Zielwerte const idPHZielwert = "8,1"; const orpZielwert = 480; // interne Variablen var timer; var acid; var dosingOK = false; setTimeout(function() { dosingOK = getState(idPoolPumpe).val; }, 300000); function Dosierung() { acid = false; setState(idChlorAcidAlternative, "chlor"); timer = setInterval(function() { // Umschaltung Chlor / Acid acid = !acid; if(acid) setState(idChlorAcidAlternative, "acid"); else setState(idChlorAcidAlternative, "chlor"); // DosingTime abhängig von ORP-Differenz festlegen var orpIstWert = parseFloat(getState(idORPWert).val); // parseFloat, falls ORPWert ein String ist var dosingTime = 80; if (orpZielwert - orpIstWert > 100) dosingTime =60; if (orpZielwert - orpIstWert > 60) dosingTime = 50; if (orpZielwert - orpIstWert > 40) dosingTime = 45; if (orpZielwert - orpIstWert > 20) dosingTime = 35; if (orpZielwert - orpIstWert > 10) dosingTime = 20; if (orpZielwert - orpIstWert > 5) dosingTime = 10; if (orpZielwert >= orpIstWert) dosingTime = 0; // Chlor-Dosierung starten if (dosingOK && dosingTime > 0 && !getState(idPumpeChlor).val) { setState(idPumpeChlor, true); setTimeout(function(){ setState(idPumpeChlor, false); }, dosingTime * 1000); } }, 120000); // alle 2 Minuten } // Skriptstart if(getState(idDosingAuto).val) Dosierung(); on(idDosingAuto, function(dp) { if(dp.state.val) { Dosierung(); } else { if(timer) clearInterval(timer); setState(idPumpeChlor, false); } }); on(idPoolPumpe, function(dp) { if(dp.state.val) { setTimeout(function() { dosingOK = true; }, 300000); // Dosierung wird nach 5 Minuten freigegeben } else { dosingOK = false; setState(idPumpeChlor, false); } });
Dosierung läuft schonmal die Klammern waren falsch rum. Supi :mrgreen:
-Jetzt noch das Problem mit dem Timer. Der Timer müsste von der Pumpe zurückgesetzt werden. Sprich Pumpe aus Timer zurücksetzen.
-Was auch noch Interessant wäre Die Abfrage vom Orp Wert <500 und >800. Sowie die Abfrage des Timestamp vom Parser Adapter Wert nicht älter als 3 Minuten. Wenn das nicht der fall wäre Dosierung Stoppen und Variable const idStörung .
-
Servus,
komme leider nicht weiter dafür reichen nicht ansatzweise meine Java Kenntnisse. Paul kannst du noch mal weiter helfen oder sonst wer ?
Grüße Timo
-
Der Timer müsste von der Pumpe zurückgesetzt werden. Sprich Pumpe aus Timer zurücksetzen. `
Welche Timer meinst Du ? Den, der die Freigabe nach 5 Minuten Laufzeit der Poolpumpe macht ?// interne Variablen var timer; var acid; var dosingOK = false; var timerPool = setTimeout(function() { dosingOK = getState(idPoolPumpe).val; }, 300000); ... on(idPoolPumpe, function(dp) { clearTimeout(timerPool); if(dp.state.val) { timerPool = setTimeout(function() { dosingOK = true; }, 300000); // Dosierung wird nach 5 Minuten freigegeben } else { dosingOK = false; setState(idPumpeChlor, false); } });
-
Servus Paul,
läuft doch alles mit dem Timer von der Pumpe. Komme schon durcheinander das 2 Script gleichzeitig laufen und sich beeinflussen. Mit der Klammer < war übrigens doch richtig.
Jetzt wäre noch die geschichte mit der Abfrage von vom Orp Wert <500 und >800. Sowie die Abfrage des Timestamp vom Parser Adapter Wert nicht älter als 3 Minuten. Wenn das nicht der fall wäre Dosierung Stoppen und Variable const idStörung .
Damit komme ich gar nicht klar.