| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621 |
- ##############################################
- # $Id: 98_IF.pm 12944 2017-01-03 12:56:17Z Damian $
- package main;
- use strict;
- use warnings;
- use POSIX;
- sub CommandIF($$);
- sub GetBlockIf ($$);
- sub CmdIf($);
- sub ReplaceReadingsIf($);
- sub ReplaceAllReadingsIf($$);
- sub ParseCommandsIf($);
- sub EvalAllIf($);
- sub InternalIf($$$);
- sub ReadingValIf($$$);
- #####################################
- sub
- IF_Initialize($$)
- {
- my %lhash = ( Fn=>"CommandIF",
- Hlp=>"(<condition>) (<FHEM commands>) ELSE (<FHEM commands>), executes FHEM commands depending on the condition");
- $cmds{IF} = \%lhash;
- }
- sub
- GetBlockIf ($$)
- {
- my ($cmd,$match) = @_;
- my $count=0;
- my $first_pos=0;
- my $last_pos=0;
- my $err="";
- while($cmd =~ /$match/g) {
- if (substr($cmd,pos($cmd)-1,1) eq substr($match,2,1)) {
- $count++;
- $first_pos=pos($cmd) if ($count == 1);
- } elsif (substr($cmd,pos($cmd)-1,1) eq substr($match,4,1)) {
- $count--;
-
- }
- if ($count < 0)
- {
- $err="right bracket without left bracket";
- return ("",substr($cmd,pos($cmd)-1),$err,"");
- }
-
- if ($count == 0) {
- $last_pos=pos($cmd);
- last;
- }
- }
- if ($count > 0) {
- $err="no right bracket";
- return ("",substr($cmd,$first_pos-1),$err);
- }
- if ($first_pos) {
- return (substr($cmd,0,$first_pos-1),substr($cmd,$first_pos,$last_pos-$first_pos-1),"",substr($cmd,$last_pos));
- } else {
- return ($cmd,"","","");
- }
- }
- sub
- InternalIf($$$)
- {
- my ($name,$internal,$regExp)=@_;
- my $r="";
- my $element;
- $r=$defs{$name}{$internal};
- if ($regExp) {
- $element = ($r =~ /$regExp/) ? $1 : "";
- } else {
- $element=$r;
- }
- return($element);
- }
- sub
- ReadingValIf($$$)
- {
- my ($name,$reading,$regExp)=@_;
- my $r="";
- my $element;
- $r=$defs{$name}{READINGS}{$reading}{VAL};
- if ($regExp) {
- $element = ($r =~ /$regExp/) ? $1 : "";
- } else {
- $element=$r;
- }
- return($element);
- }
- sub
- ReplaceReadingIf($)
- {
- my ($element) = @_;
- my $beginning;
- my $tailBlock;
- my $err;
- my $regExp="";
- my ($name,$reading,$format)=split(":",$element);
- my $internal="";
- if ($name) {
- return ($name,"unknown Device") if(!$defs{$name});
- if ($reading) {
- if (substr($reading,0,1) eq "\&") {
- $internal = substr($reading,1);
- return ($name.":".$internal,"unknown internal") if(!$defs{$name}{$internal});
- } else {
- return ($name.":".$reading,"unknown reading") if(!$defs{$name}{READINGS}{$reading});
- }
- if ($format) {
- if ($format eq "d") {
- $regExp = '(-?\d+(\.\d+)?)';
- } elsif (substr($format,0,1) eq '[') {
- ($beginning,$regExp,$err,$tailBlock)=GetBlockIf($format,'[\[\]]');
- return ($regExp,$err) if ($err);
- return ($regExp,"no round brackets in regular expression") if ($regExp !~ /.*\(.*\)/);
- } else {
- return($format,"unknown expression format");
- }
- }
- if ($internal) {
- return("InternalIf('$name','$internal','$regExp')","");
- } else {
- return("ReadingValIf('$name','$reading','$regExp')","");
- }
- } else {
- return("InternalIf('$name','STATE','$regExp')","");
- }
- }
- }
- sub
- ReplaceAllReadingsIf($$)
- {
- my ($tailBlock,$evalFlag)= @_;
- my $block="";
- my $beginning;
- my $err;
- my $cmd="";
- my $ret="";
- while ($tailBlock ne "") {
- ($beginning,$block,$err,$tailBlock)=GetBlockIf($tailBlock,'[\[\]]');
- return ($block,$err) if ($err);
- if ($block ne "") {
- if ($block =~ /:/ or ($block =~ /[a-z]/i and $block =~ /^[a-z0-9._]*$/i))
- {
- ($block,$err)=ReplaceReadingIf($block);
- return ($block,$err) if ($err);
- if ($evalFlag) {
- my $ret = eval $block;
- return($block." ",$@) if ($@);
- # return($eval,"no reading value") if (!$ret);
- $ret =~ s/'/\\'/g;
- $block=$ret;
- }
- } else {
- $block="[".$block."]";
- }
- }
- $cmd.=$beginning.$block;
- }
- return ($cmd,"");
- }
- sub
- EvalAllIf($)
- {
- my ($tailBlock)= @_;
- my $eval="";
- my $beginning;
- my $err;
- my $cmd="";
- my $ret="";
-
- while ($tailBlock ne "") {
- ($beginning,$eval,$err,$tailBlock)=GetBlockIf($tailBlock,'[\{\}]');
- return ($eval,$err) if ($err);
- if ($eval) {
- if (substr($eval,0,1) eq "(") {
- my $ret = eval $eval;
- return($eval." ",$@) if ($@);
- $eval=$ret;
- } else {
- $eval="{".$eval."}";
- }
- }
- $cmd.=$beginning.$eval;
- }
- return ($cmd,"");
- }
- sub
- ParseCommandsIf($)
- {
- my($tailBlock) = @_;
- my $currentBlock="";
- my $beginning="";
- my $err="";
- my $parsedCmd="";
- my $pos=0;
- $tailBlock =~ s/;/;;/g;
- my $sleep;
- while ($tailBlock ne "") {
- if ($tailBlock=~ /^\s*\{/) { # perl block
- ($beginning,$currentBlock,$err,$tailBlock)=GetBlockIf($tailBlock,'[\{\}]');
- return ($currentBlock,$err) if ($err);
- $parsedCmd.=$currentBlock;
- }
- if ($tailBlock =~ /^\s*IF/) {
- ($beginning,$currentBlock,$err,$tailBlock)=GetBlockIf($tailBlock,'[\(\)]'); #condition
- return ($currentBlock,$err) if ($err);
- $parsedCmd.="fhem('".$beginning."(".$currentBlock.")";
- ($beginning,$currentBlock,$err,$tailBlock)=GetBlockIf($tailBlock,'[\(\)]'); #if case
- return ($currentBlock,$err) if ($err);
- $currentBlock =~ s/'/\\'/g;
- $currentBlock =~ s/;/;;/g;
- $parsedCmd.=$beginning."(".$currentBlock.")";
- if ($tailBlock =~ /^\s*ELSE/) {
- ($beginning,$currentBlock,$err,$tailBlock)=GetBlockIf($tailBlock,'[\(\)]'); #else case
- return ($currentBlock,$err) if ($err);
- $currentBlock =~ s/'/\\'/g;
- $currentBlock =~ s/;/;;/g;
- $parsedCmd.=$beginning."(".$currentBlock.")";
- }
- $parsedCmd.="')";
- } else { #replace Readings if no IF command
- if ($tailBlock =~ /^\s*\(/) { # remove bracket
- ($beginning,$currentBlock,$err,$tailBlock)=GetBlockIf($tailBlock,'[\(\)]');
- return ($currentBlock,$err) if ($err);
- $tailBlock=substr($tailBlock,pos($tailBlock)) if ($tailBlock =~ /^\s*,/g);
- } elsif ($tailBlock =~ /,/g) {
- $pos=pos($tailBlock)-1;
- $currentBlock=substr($tailBlock,0,$pos);
- $tailBlock=substr($tailBlock,$pos+1);
- } else {
- $currentBlock=$tailBlock;
- $tailBlock="";
- }
- if ($currentBlock =~ /[^\s]/g) {
- $currentBlock =~ s/'/\\'/g;
- ($currentBlock,$err)=ReplaceAllReadingsIf($currentBlock,1);
- return ($currentBlock,$err) if ($err);
- ($currentBlock,$err)=EvalAllIf($currentBlock);
- $currentBlock =~ s/;/;;/g;
- return ($currentBlock,$err) if ($err);
- if ($sleep) {
- $parsedCmd.=$currentBlock;
- if ($tailBlock) {
- $parsedCmd.=";;"
- } else {
- $parsedCmd.="')"
- }
- } elsif ($currentBlock =~ /^\s*sleep/) {
- $sleep=1;
- $parsedCmd.="fhem('".$currentBlock.";;";
- $parsedCmd.="')" if !($tailBlock);
- } else {
- $parsedCmd.="fhem('".$currentBlock."')";
- $parsedCmd.=";;" if ($tailBlock);
- }
- } else {
- $parsedCmd.=";;" if ($tailBlock);
- }
- }
- }
-
- return($parsedCmd,"");
- }
- sub
- CmdIf($)
- {
- my($cmd) = @_;
- my $cond="";
- my $err="";
- my $if_cmd="";
- my $else_cmd="";
- my $tail;
- my $tailBlock;
- my $eval="";
- my $beginning;
-
- $cmd =~ s/\n//g;
- return($cmd, "no left bracket") if ($cmd !~ /^ *\(/);
- ($beginning,$cond,$err,$tail)=GetBlockIf($cmd,'[\(\)]');
- return ($cond,$err) if ($err);
- ($cond,$err)=ReplaceAllReadingsIf($cond,0);
- return ($cond,$err) if ($err);
- return ($cmd,"no condition") if ($cond eq "");
- if ($tail =~ /^\s*\(/) {
- ($beginning,$if_cmd,$err,$tail)=GetBlockIf($tail,'[\(\)]');
- return ($if_cmd,$err) if ($err);
- ($if_cmd,$err)=ParseCommandsIf($if_cmd);
- return ($if_cmd,$err) if ($err);
- return ($cmd,"no commands") if ($if_cmd eq "");
- } else {
- return($tail, "no left bracket");
- }
- return ($if_cmd,$err) if ($err);
- if (length($tail)) {
- $tail =~ /^\s*ELSE/g;
- if (pos($tail)) {
- $tail=substr($tail,pos($tail));
- if (!length($tail)) {
- return ($tail,"no else block");
- }
- } else {
- return ($tail,"expected ELSE");
- }
- if ($tail =~ /^\s*\(/) {
- ($beginning,$else_cmd,$err,$tail)=GetBlockIf($tail,'[\(\)]');
- return ($else_cmd,$err) if ($err);
- ($else_cmd,$err)=ParseCommandsIf($else_cmd);
- return ($else_cmd,$err) if ($err);
- } else {
- return($tail, "no left bracket");
- }
- return ($else_cmd,$err) if ($err);
- }
- my $perl_cmd="{if(".$cond.")";
- $perl_cmd .="{".$if_cmd."}";
- $perl_cmd .= "else{".$else_cmd."}" if ($else_cmd);
- $perl_cmd.="}";
- return($perl_cmd,"");
- }
- sub
- CommandIF($$)
- {
- my ($cl, $param) = @_;
- return "Usage: IF (<condition>) (<FHEM commands>) ELSE (<FHEM commands>)\n" if (!$param);
- my $ret;
- #print ("vor IF:$param\n");
- my ($cmd,$err)=CmdIf($param);
- #print ("nach IF:$cmd\n");
- if ($err ne "") {
- $ret="IF: $err: $cmd";
- } else {
- $ret = AnalyzeCommandChain(undef,$cmd);
- use strict "refs";
- }
- return $ret;
- }
- 1;
- =pod
- =item summary FHEM IF-command
- =item summary_DE FHEM IF-Befehl
- =begin html
- <a name="IF"></a>
- <h3>IF</h3>
- <ul>
- <code>IF (<condition>) (<FHEM commands1>) ELSE (<FHEM commands2>)</code><br>
- <br>
- Executes <FHEM commands1> if <condition> is true, else <FHEM commands2> are executed.<br>
- <br>
- IF can be used anywhere where FHEM commands can be used.<br>
- <br>
- The ELSE-case is optional.<br>
- <br>
- The <condition> is the same as in perl-if.<br>
- <br>
- In addition, readings can be specified in the form:<br>
- <br>
- [<device>:<reading>:<format>|[<regular expression>]]<br>
- <br>
- In addition, internals can be specified with & in the form:<br>
- <br>
- [<device>:&<internal>:<format>|[<regular expression>]]<br>
- <br>
- <format> and [<regular expression>] are filter options und are optional.<br>
- <br>
- possible <format>:<br>
- <br>
- 'd' for decimal number<br>
- <br>
- If only the state of a device is to be used, then only the device can be specified:<br>
- <br>
- <code>[<device>]</code> corresponsed to <code>[<device>:&STATE]</code><br>
- <br>
- <b>Examples:</b><br>
- <br>
- IF in combination with at-module, Reading specified in the condition:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 70) (set switch1 off) ELSE (set switch1 on)<br></code>
- <br>
- IF state query of the device "outdoor" in the condition:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor] eq "open") (set switch1 on)<br></code>
- <br>
- corresponds with details of the internal:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:&STATE] eq "open") (set switch1 on)<br></code>
- <br>
- If the reading "state" to be queried, then the name of reading is specified without &:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:state] eq "open") (set switch1 on)<br></code>
- <br>
- Nested IF commands (It can be entered in the DEF input on multiple lines with indentation for better representation):<br>
- <br>
- <code>define test notify lamp <br>
- IF ([lampe] eq "on") (<br>
- <ol>
- IF ([outdoor:humidity] < 70)<br>
- <ol>(set lamp off)</ol>
- ELSE<br>
- <ol>(set lamp on)</ol>
- </ol>
- ) ELSE<br>
- <ol>(set switch on)</ol><br>
- </code>
- Filter by numbers in Reading "temperature":<br>
- <br>
- <code>define settemp at 22:00 IF ([tempsens:temperature:d] >= 10) (set heating on)<br></code>
- <br>
- Filter by "on" and "off" in the status of the device "move":<br>
- <br>
- <code>define activity notify move IF ([move:&STATE:[(on|off)]] eq "on" and $we) (set lamp off)<br></code>
- <br>
- Example of the use of Readings in the then-case:<br>
- <br>
- <code>define temp at 18:00 IF ([outdoor:temperature] > 10) (set lampe [dummy])<br></code>
- <br>
- If an expression is to be evaluated first in a FHEM command, then it must be enclosed in brackets.<br>
- For example, if at 18:00 clock the outside temperature is higher than 10 degrees, the desired temperature is increased by 1 degree:<br>
- <br>
- <code>define temp at 18:00 IF ([outdoor:temperature] > 10) (set thermostat desired-temp {([thermostat:desired-temp:d]+1)})<br></code>
- <br>
- Multiple commands are separated by a comma instead of a semicolon, thus eliminating the doubling, quadrupling, etc. of the semicolon:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 10) (set switch1 off,set switch2 on) ELSE (set switch1 on,set switch2 off)<br></code>
- <br>
- If a comma in FHEM expression occurs, this must be additionally bracketed so that the comma is not recognized as a delimiter:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 10) ((set switch1,switch2 off))<br></code>
- <br>
- IF in combination with a define at multiple set commands:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (define a_test at +00:10 set lampe1 on;;set lampe2 off;;set temp desired 20)<br></code>
- <br>
- The comma can be combined as a separator between the FHEM commands with double semicolon, eg:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (set lamp1 on,define a_test at +00:10 set lampe2 on;;set lampe3 off;;set temp desired 20)<br></code>
- <br>
- sleep can be used with comma, it is not blocking:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (sleep 2,set lampe1 on,sleep 3,set lampe2 on)</code><br>
- <br>
- Time-dependent switch: In the period 20:00 to 22:00 clock the light should go off when it was on and I leave the room:<br>
- <br>
- <code>define n_lamp_off notify sensor IF ($hms gt "20:00" and $hms lt "22:00" and [sensor] eq "absent") (set lamp:FILTER=STATE!=off off)<br></code>
- <br>
- Combination of Perl and FHEM commands ($NAME and $EVENT can also be used):<br>
- <br>
- <code>define mail notify door:open IF ([alarm] eq "on")({system("wmail $NAME:$EVENT")},set alarm_signal on)<br></code>
- </ul>
- =end html
- =begin html_DE
- <a name="IF"></a>
- <h3>IF</h3>
- <ul>
- <code>IF (<Bedingung>) (<FHEM-Kommandos1>) ELSE (<FHEM-Kommandos2>)</code><br>
- <br>
- Es werden <code><FHEM-Kommandos1></code> ausgeführt, wenn <code><Bedingung></code> erfüllt ist, sonst werden <code><FHEM-Kommanodos2></code> ausgeführt.<br>
- <br>
- Beim IF-Befehl (IF in Großbuchstaben) handelt es sich um einen FHEM-Befehl. Der Befehl kann überall dort genutzt werden, wo FHEM-Befehle vorkommen dürfen.
- Im Gegensatz zu Perl-if (if in Kleinbuchstaben) bleibt man auf der FHEM-Ebene und muss nicht auf die Perl-Ebene, um FHEM-Befehle mit Hilfe der fhem-Funktion auszuführen.<br>
- <br>
- IF ist kein eigenständig arbeitendes Modul, sondern ein FHEM-Befehl, der nur in Kombination mit anderen Modulen, wie z. B. notify oder at, sinnvoll eingesetzt werden kann.
- Es gibt inzwischen ein neueres <a href="http://fhem.de/commandref_DE.html#DOIF">DOIF</a>-Modul, welches auf der Syntax vom IF-Befehl aufbaut.
- Es arbeitet im Gegensatz zu IF als Modul selbstständig ereignis- und zeitgesteuert ohne notify bzw. at. Damit lassen sich viele Problemlösungen eleganter, jeweils mit einem einzigen Modul, realisieren.<br>
- <br>
- In der Bedingung des IF-Befehls wird die vollständige Syntax des Perl-if unterstützt. Stati und Readings von Devices werden in eckigen Klammern angegeben.<br>
- <br>
- <br>
- <b>Beispiele:</b><br>
- <br>
- IF in Kombination mit at-Modul, Readingangabe [<Device>:<Reading>] in der Bedingung:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 70) (set switch1 off) ELSE (set switch1 on)<br></code>
- <br>
- IF Statusabfrage des Devices "outdoor" in der Bedingung:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor] eq "open") (set switch1 on)<br></code>
- <br>
- entspricht mit Angabe des Internals mit &:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:&STATE] eq "open") (set switch1 on)<br></code>
- <br>
- Wenn der Reading "state" abgefragt werden soll, dann wird der Readingname ohne & angegeben:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:state] eq "open") (set switch1 on)<br></code>
- <br>
- Geschachtelte Angabe von mehreren IF-Befehlen kann in mehreren Zeilen mit Einrückungen zwecks übersichtlicher
- Darstellung über FHEM-Weboberfläche in der DEF-Eingabe eingegeben werden.<br>
- Die erste Zeile "define test notify lamp " muss mit einem Leerzeichen enden, bevor die Zeile mit Enter umgebrochen wird - das ist eine Eigenschaft von notify und nicht von IF:<br>
- <br>
- <code>define test notify lamp <br>
- IF ([lamp] eq "on") (<br>
- <ol>
- IF ([outdoor:humidity] < 70)<br>
- <ol>
- (set lamp off)<br>
- </ol>
- ELSE<br>
- <ol>
- (set lamp on)<br>
- </ol>
- </ol>
- ) ELSE<br>
- <ol>
- (set switch on)<br>
- </ol>
- <br></code>
- Mehrzeilige Eingaben in der cfg-Datei müssen dagegen jeweils am Zeilenende mit \ verknüpft werden (das ist eine Eigenschaft von FHEM und nicht von IF):<br>
- <br>
- <code>define test notify lamp \<br>
- IF ([lamp] eq "on") (\<br>
- <ol>
- IF ([outdoor:humidity] < 70)\<br>
- <ol>
- (set lamp off)\<br>
- </ol>
- ELSE\<br>
- <ol>
- (set lamp on)\<br>
- </ol>
- </ol>
- ) ELSE\<br>
- <ol>
- (set switch on)<br>
- </ol>
- <br></code>
- Filtern nach Zahlen im Reading "temperature":<br>
- <br>
- <code>define settemp at 22:00 IF ([tempsens:temperature:d] >= 10) (set heating on)<br></code>
- <br>
- Filtern nach "on" und "off" im Status des Devices "move":<br>
- <br>
- <code>define activity notify move IF ([move:&STATE:[(on|off)]] eq "on" and $we) (set lamp off)<br></code>
- <br>
- Beispiel für die Nutzung des Status eines Devices im Ausführungsteil. Hier: "lamp1" wird mit dem Status von "lamp2" geschaltet:<br>
- <br>
- <code>define temp at 18:00 IF ([outdoor:temperature] > 10) (set lamp1 [lamp2])<br></code>
- <br>
- Falls bei einem FHEM-Befehl ein Perl-Ausdruck mit Readings zuvor ausgewertet werden soll, so muss er in geschweifte und runde Klammern gesetzt werden.<br>
- Beispiel: Wenn um 18:00 Uhr die Außentemperatur höher ist als 10 Grad, dann wird die Solltemperatur um 1 Grad erhöht.<br>
- <br>
- <code>define temp at 18:00 IF ([outdoor:temperature] > 10) (set thermostat desired-temp {([thermostat:desired-temp:d]+1)})<br></code>
- <br>
- Mehrerer Befehle werden durch ein Komma statt durch ein Semikolon getrennt, dadurch entfällt das Doppeln, Vervierfachen usw. des Semikolons:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 10) (set switch1 off,set switch2 on) ELSE (set switch1 on,set switch2 off)<br></code>
- <br>
- Falls ein Komma im FHEM-Ausdruck vorkommt, muss dieser zusätzlich geklammert werden, damit das Komma nicht als Trennzeichen erkannt wird:<br>
- <br>
- <code>define check at +00:10 IF ([outdoor:humidity] > 10) ((set switch1,switch2 off))<br></code>
- <br>
- IF in Kombination mit einem define at mit mehreren set-Befehlen (Eingabe muss wegen der Semikolons im DEF-Editor erfolgen,
- einfaches Semikolon ist nicht erlaubt - es würde vom FHEM-Parser "geschluckt" werden und beim IF nicht mehr ankommen):<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (define a_test at +00:10 set lampe1 on;;set lampe2 off;;set temp desired 20)<br></code>
- <br>
- Man kann die Problematik des Doppelns von Semikolons wie folgt umgehen:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (define a_test at +00:10 IF (1) (set lampe1 on,set lampe2 off,set temp desired 20))<br></code>
- <br>
- Das Komma als Trennzeichen zwischen den FHEM-Befehlen lässt sich mit ;; kombinieren, z. B.:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (set lamp1 on,define a_test at +00:10 set lampe2 on;;set lampe3 off;;set temp desired 20)<br></code>
- <br>
- sleep kann mit Komma verwendet werden, dabei wirkt das sleep nicht blockierend:<br>
- <br>
- <code>define check at *10:00 IF ([indoor] eq "on") (sleep 2,set lampe1 on,sleep 3,set lampe2 on)</code><br>
- <br>
- Zeitabhängig schalten: In der Zeit zwischen 20:00 und 22:00 Uhr soll das Licht ausgehen, wenn es an war und ich den Raum verlasse:<br>
- <br>
- <code>define n_lamp_off notify sensor IF ($hms gt "20:00" and $hms lt "22:00" and [sensor] eq "absent") (set lamp:FILTER=STATE!=off off)<br></code>
- <br>
- Kombination von Perl und FHEM-Befehlen ($NAME sowie $EVENT können ebenso benutzt werden):<br>
- <br>
- <code>define mail notify door:open IF ([alarm] eq "on")({system("wmail $NAME:$EVENT")},set alarm_signal on)<br></code>
- <br>
- Der IF-Befehl dient in erster Linie zur Vereinfachung der Schreibweise in Kombination mit anderen FHEM-Modulen wie at, notify oder DOIF.
- Intern wird der IF-Befehl zur Ausführung in einen Perl if-Befehl umgesetzt. Das soll anhand von Beispielen verdeutlicht werden:<br>
- <br>
- <code>IF ([switch] eq "off") (set lamp on)</code><br>
- <br>
- entspricht:<br>
- <br>
- <code>{if (Value('switch') eq "off"){fhem('set lamp on')}}</code><br>
- <br>
- <br>
- <code>IF ([living_room:temperature] > 12) (set lamp on, set lamp2 off)</code><br>
- <br>
- entspricht:<br>
- <br>
- <code>{if (ReadingVal('living_room','temperature','') > 12) {fhem('set lamp on');;fhem('set lamp2 off')}}</code><br>
- <br>
- <br>
- <code>IF ([bathroom:humidity] > 70) (set led red) ELSE (set led green)</code><br>
- <br>
- entspricht:<br>
- <br>
- <code>{if (ReadingsVal('bathroom','humidity','') > 70) {fhem('set led red')} else {fhem('set led green')}}</code><br>
- <br>
- <br>
- </ul>
- =end html_DE
- =cut
|