Bewässerungssteuerung
Todo: Diese Seite sollte an die im Forum beschriebene aktuelle Version der Steuerung angepasst werden. |
Diese Seite beschreibt, wie man intelligent seine Bewässerung steuern kann wenn man zb. Bodenfeuchtesensoren installiert hat.
Hinweis: Der Autor hat dieses hier in einem Modul bereitgestellt
Es wird von folgenden Komponenten ausgegangen:
- Gardena Magnetventile, angesprochen über einen FS20-Aktor (zb. S8M)
- Bodenfeuchtesensoren, zb. FS20 BF oder andere (zb. an einem AVR-NET-IO)
allgemeine Einstellungen
Bitte die Prozedur "on-for-timer-offcheck" anlegen: On-for-timer_zurücksetzen
Anlegen zusätzlicher Attribute
Es müssen folgende neue globale Attribute angelegt werden:
- schwellwert
- sensor
Anlegen der Bodenfeuchtesensoren
Für einen FS20 Sensor: FS20BF
define Bodenfeuchtesensor FS20 34f2 00 attr Bodenfeuchtesensor room Bewässerung
Für eine 1wire Version: einen analogen Sensor (zb. Vegetronix), angeschlossen an einen 4fach A/D Converter DS2450
define Bodenfeuchte OWAD DS2450 50310C000000 60 attr Bodenfeuchte AFunction VA*33.9 attr Bodenfeuchte AUnit percent|% attr Bodenfeuchte BFunction VB*33.9 attr Bodenfeuchte BUnit percent|% attr Bodenfeuchte CFunction VC*33.9 attr Bodenfeuchte CUnit percent|% attr Bodenfeuchte event-on-change-reading state attr Bodenfeuchte model DS2450 attr Bodenfeuchte room Bewässerung
Alternativ auch andere, siehe AVR-NET-IO
zentrale Schaltvariable
Erstellung einer Schaltvariable um im Frontend die Bewässerung zentral zu steuern
- Auto => Je nach Angabe des Bodenfeuchtesensors wird bewässert oder nicht
- On => Es wird immer bewässert
- Off => Es wird nicht bewässert
define Bewaesserung_Active dummy attr Bewaesserung_Active room Bewässerung attr Bewaesserung_Active setList Auto On Off attr Bewaesserung_Active webCmd Auto:On:Off
Steuerung mittels 1wire
wird die Bewässerungssteuerung mittels 1Wire betrieben, so verfügt 1 Device (i.d.R DS2406/DS2408/DS1413) über 2 oder 8 Ports. Wie diese einerseits komfortabel per FHEM-Weboberfläche zu steuern als auch eine automatisierte Steuerung zu realisieren ist soll dieser Abschnitt aufzeigen.
Anlegen des Schalt-Devices
An folgendem Device hängen die physischen Bewässerungsventile. Dieses muss natürlich an die eigenen Gegebenheiten angepasst werden :)
define Schalter_links OWSWITCH DS2408 F4D210000000 attr Schalter_links event-on-change-reading A,B,C,D,E,F,G,H attr Schalter_links model DS2408 attr Schalter_links room OWX
Anlegen der DummyDevices
Über diese "Schalter" werden die einzelnen Bewässerungsstränge gesteuert.
define Bewaesserung_Tomaten dummy attr Bewaesserung_Tomaten alias Bewässerung Tomaten attr Bewaesserung_Tomaten eventMap /on-for-timer 30:Anschalten/off:Ausschalten/ attr Bewaesserung_Tomaten group Bewässerung attr Bewaesserung_Tomaten room OWX attr Bewaesserung_Tomaten schwellwert 58 attr Bewaesserung_Tomaten sensor Schalter_links:A Bodenfeuchte:A attr Bewaesserung_Tomaten webCmd Anschalten:Ausschalten
Auf der Weboberfläche erscheinen die Schalter "Anschalten" und Ausschalten". Diese werden mittels eventmap auf "on-for-timer 30" und "off" intern umgesetzt und später an den 1wireBaustein weitergegeben. Über das Attribut "sensor" wird eine Zuordnung zum tatsächlichen 1wireDevice als auch zum dazugehörigen Sensor vorgenommen und wird folgendermaßen definiert:
attr Bewaesserung_Tomaten sensor Device[:Port] [Bodenfeuchtesensor[:Port]]
Im Beispiel ist das 1wireDevice ein DS2408 und heißt "Schalter_links" und die Bewässerung der Tomaten hängt am Port A. Der zugehörige Bodenfeuchtesensor in den Tomaten hängt an einem DS2450 mit dem Namen "Bodenfeuchte" ebenfalls am Port A. Über den Schwellwert wird eingestellt, ab welcher Bodenfeuchte tatsächlich bewässert werden soll.
Anlegen des Notify´s für die DummySchalter
Folgendes Notify reagiert auf die Events der Dummyschalter. Es wird hier im Beispiel das Event "Anschalten" auf "on-for-timer 30" umgesetzt sowie das Sensor- und schwellwertattribut abgerufen. Das Sensorattribut wird in den Device- und Sensorabschnitt zergliedert. Anschließend wird mit den so ermittelten Parametern die zentrale Bewässerungsprozedur aufgerufen.
define Bewaesserung_Macro notify Bewaesserung.* {\ my $device=$NAME;;\ my @a = (undef, $EVENT);;\ @a = ReplaceEventMap($device, \@a, 0);;\ shift @a;;\ my $status = join(" ", @a);;\ my @temp = split(" ", AttrVal($device, "sensor", " "));;\ my $schwellwert = AttrVal($device, "schwellwert", "");;\ \ Log 5, "Prozeduraufruf: Bewaesserung($temp[0],$temp[1],$schwellwert,$status)";;\ Bewaesserung($temp[0],$temp[1],$schwellwert,$status, $device);;\ } attr Bewaesserung_Macro group Bewässerung attr Bewaesserung_Macro room OWX
zentrale Bewässerungsprozedur
Die zentrale Prozedur wird als UTIL Datei im FHEM Verzeichnis gespeichert. Hier wird die Entscheidung getroffen, ob bewässert werden soll oder nicht. Es wird die aktuelle Bodenfeuchte im Zusammenhang mit dem Schwellwert ausgewertet sowie die globale Einstellung "Auto"/"On"/"Off"
############################################## # $Id:$ package main; use strict; use warnings; use POSIX; use URI::Escape; sub Utils_Bewaesserung_Initialize($$) { my ($hash) = @_; } ############################################## # Prozedur zur Bewässerungssteuerung # Parameter: # 1. Device:Port (zb. Schalter:A) # 2. Sensor:Port (zb. MySens:C) # 3. Schwellwert (0-100 in % - zb. 55) # 4. Timer Befehl (zb. on-for-timer 30) # 5. Dummy-Schalter (optional) # # Bsp: Bewaesserung("Schalter_links:A","Bodenfeuchte:A",56,"on-for-timer 30") ############################################## sub Bewaesserung($$$$$) { my ($device,$sensor,$schwellwert,$command,$dummyschalter) = @_; my ($deviceport, $sensorport); my $auto = ReadingsVal("Bewaesserung_Active", "state", "Off"); my $bodenfeuchte; my (@tempd, @temps); #es muss immer ein Command übergeben werden if($command !~ m/(on|on-for-timer|off)/) { Log 3, "Abbruch: Fehlende Angabe des Commands"; return; } # falsche Schwellwertangabe if(defined($schwellwert) && $schwellwert !~ m/^(\d+)$/) { Log 3, "Schwellwert definiert, aber nicht nummerisch: $schwellwert"; return; } @tempd = split(":", $device) if($device =~ m/:/); $device = $tempd[0]; $deviceport = $tempd[1] if($tempd[1]); @temps = split(":", $sensor) if($sensor =~ m/:/); $sensor = $temps[0]; $sensorport = $temps[1] if($temps[1]); if(!$defs{$device}) { Log 3, "Fehlerhafte Angabe des Devices: $device"; return; } if(defined($sensor) && !$defs{$sensor}) { Log 3, "Fehlerhafte Angabe des Sensors: $sensor"; return; } if (($auto eq "Auto") && (!defined($sensor))) { Log 3, "Automatikmodus angefordert aber kein Sensor definiert."; return; } elsif (($auto eq "Auto") && (!defined($schwellwert))) { Log 3, "Automatikmodus angefordert aber kein Schwellwert definiert."; return; } elsif(($auto eq "Auto") && (defined($sensor))) { # Status des Sensors abfragen if (defined($sensorport)) { $bodenfeuchte = ReadingsVal($sensor, $sensorport, undef) if (!defined($bodenfeuchte)); } else { $bodenfeuchte = ReadingsVal($sensor, "state", undef) if (!defined($bodenfeuchte)); $bodenfeuchte = ReadingsVal($sensor, "status", undef) if (!defined($bodenfeuchte)); } # im Fehlerfall setze Bodenfeuchte auf 100% if(!defined($bodenfeuchte)) { Log 3, "Kann Bodenfeuchte nicht ermitteln. Breche Vorgang ab."; $bodenfeuchte = 999; } else { #Bodenfeuchte ist kein Messwert sondern nur ein on/off Reading $bodenfeuchte = 0 if(lc($bodenfeuchte) =~ m/^(on)/); $bodenfeuchte = 101 if(lc($bodenfeuchte) =~ m/^(off)/); } } else { # kein sensor definiert und kein Automatikmodus, setze Bodenfeuchte auf 0% und # Schwellwert auf 100% damit bewässert wird $bodenfeuchte = 0; $schwellwert = 100; } if(($auto eq "Auto") || ($auto eq "On")) { Log 3, "Modus: $auto -> Device: $device:$deviceport, Bodenfeuchte: $bodenfeuchte%, Schwellwert: $schwellwert%, DummySchalter: $dummyschalter"; if ($schwellwert >= $bodenfeuchte) { Log 3, "fhem: set $device output $deviceport $command"; fhem "set $device output $deviceport $command"; if (defined($dummyschalter)) { Log 3, "fhem: setState $dummyschalter $command"; Log 3, "fhem: on_for_timer_offcheck($dummyschalter, $command)"; fhem "setState $dummyschalter $command"; on_for_timer_offcheck($dummyschalter, $command); } } else { Log 3, "fhem: set $device output $deviceport off"; fhem "set $device output $deviceport off"; if (defined($dummyschalter)) { Log 3, "fhem: setState $dummyschalter off"; fhem "setState $dummyschalter off"; } } } #elsif Device ne off -> set Device off } 1;
Steuerung mittels FS20/einzelner Devices
Falls für jedes Magnetventil ein eigenes Device existiert. Zb. bei Nutzung von eines SM8/SM4/FS20ST
Bewässerungsventile definieren
Als erstes werden die Bewässerungsventile in FHEM definiert, zb.:
define Bewaesserung_Ventil1 FS20 2305 51 attr Bewaesserung_Ventil1 model fs20st attr Bewaesserung_Ventil1 room Bewässerung
Zuweisung eines Sensors einem oder mehrerer Bewässerungsventile
Hier der Fall eines FS20BF Bodenfeuchtesensors. Dieser übermittelt nur im state ein "on" oder "off"
attr Bewaesserung_Ventil1 sensor Bodenfeuchtesensor
Im Falle der 1wire Version wird das Ventil 1 mit dem Bodenfeuchtesensor "gepaired" der am Port A des Sensors "Bodenfeuchte" hängt. Dieser übermittelt einen echten Dezimalwert.
attr Bewaesserung_Ventil1 sensor Bodenfeuchte:A
zentrale Prozedur zur Bewässerung
Jetzt wird die zentrale Prozedur zur Steuerung angelegt:
define BewNotify notify BewNotify { my @@args = split(" ",'%EVENT'); my $device = "%EVTPART0"; my $duration = "%EVTPART1"; my $sensor = AttrVal($device, "sensor", undef); my @@sArr = split(":", $sensor) if($sensor =~ m/:/); my $auto = ReadingsVal("Bewaesserung_Active", "state", "Off"); my $sensorstate = "on"; my $schwellwert = 0; if(($auto eq "Auto") || ($auto eq "On")) { if((defined ($sensor)) && ($auto eq "Auto")) { if (defined($sArr[0])) { $sensorstate = ReadingsVal($sArr[0], $sArr[1], undef); $schwellwert = AttrVal($sArr[0], "schwellwert", undef); $schwellwert = AttrVal($sArr[0], $sArr[1]."Low", 0) if (!defined($schwellwert)); } else { $schwellwert = AttrVal($sensor, "schwellwert", undef); $sensorstate = ReadingsVal($sensor, "state", undef) if (!defined($sensorstate)); $sensorstate = ReadingsVal($sensor, "status", undef) if (!defined($sensorstate)); } $sensorstate = "off" if (!defined($sensorstate)); } if($sensorstate eq "on") { fhem "set $device on-for-timer $duration"; } elsif ($sensorstate =~ m/^(\d+)/) { Log 5, "Sensorstate: $sensorstate"; Log 5, "Schwellwert: $schwellwert"; if ($schwellwert >= $sensorstate) { fhem "set $device on-for-timer $duration"; } } elsif (ReadingsVal($device, "state", "") ne "off" ) { fhem "set $device off"; } } } attr BewNotify comment Prozedur um aufgrund Umweltzustände die Bewässerung freizugeben oder nicht. Bsp: BewNotify {Device} {Dauer in sek} {Schaltvariable} attr BewNotify room Bewässerung
Das geübte Auge wird feststellen, das für die Bodenfeuchtesensoren sowohl das Reading "state" als auch das Reading "status" abgefragt wird. Das liegt daran, das die FS20 BF ein Reading "state" haben, die analogen Bodenfeuchtesensoren über die ECMDDevice classdef-Definition ein "status" Reading (Das Reading "state" ist intern verwendet und darf nicht doppelt vergeben werden).
Zeitsteuerung
6. Abschließend wird die Bewässerung in die Zeitsteuerung übergeben
define Timer_Ventil1 at *00:00:00 trigger BewNotify Bewaesserung_Ventil1 1920 attr Timer_Ventil1 room Bewässerung
Hiermit wird täglich um 0Uhr die Bewässerungsprozedur aufgerufen bei dem Ventil1 für 30min geöffnet wird. Abhängig natürlich von der Einstellung [Auto|On|Off] der Schaltariable "Bewaesserung_Active"