91_watchdog.pm 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. ##############################################
  2. # $Id: 91_watchdog.pm 14888 2017-08-13 12:07:12Z rudolfkoenig $
  3. package main;
  4. use strict;
  5. use warnings;
  6. use Time::HiRes qw(gettimeofday);
  7. #####################################
  8. sub
  9. watchdog_Initialize($)
  10. {
  11. my ($hash) = @_;
  12. $hash->{DefFn} = "watchdog_Define";
  13. $hash->{UndefFn} = "watchdog_Undef";
  14. $hash->{AttrFn} = "watchdog_Attr";
  15. $hash->{SetFn} = "watchdog_Set";
  16. $hash->{NotifyFn} = "watchdog_Notify";
  17. $hash->{AttrList} = "disable:0,1 disabledForIntervals execOnReactivate ".
  18. "regexp1WontReactivate:0,1 addStateEvent:0,1 ".
  19. "autoRestart:0,1";
  20. }
  21. #####################################
  22. # defined watchme watchdog reg1 timeout reg2 command
  23. sub
  24. watchdog_Define($$)
  25. {
  26. my ($watchdog, $def) = @_;
  27. my ($name, $type, $re1, $to, $re2, $cmd) = split("[ \t]+", $def, 6);
  28. if(defined($watchdog->{TO})) { # modify
  29. $re1 = $watchdog->{RE1} if(!defined($re1));
  30. $to = $watchdog->{TO} if(!defined($to));
  31. $re2 = $watchdog->{RE2} if(!defined($re2));
  32. $cmd = $watchdog->{CMD} if(!defined($cmd));
  33. $watchdog->{DEF} = "$re1 $to $re2 $cmd";
  34. } else {
  35. return "Usage: define <name> watchdog <re1> <timeout> <re2> <command>"
  36. if(!$cmd);
  37. }
  38. # Checking for misleading regexps
  39. eval { "Hallo" =~ m/^$re1$/ };
  40. return "Bad regexp 1: $@" if($@);
  41. $re2 = $re1 if($re2 eq "SAME");
  42. eval { "Hallo" =~ m/^$re2$/ };
  43. return "Bad regexp 2: $@" if($@);
  44. return "Wrong timespec, must be HH:MM[:SS]"
  45. if($to !~ m/^(\d\d):(\d\d)(:\d\d)?$/);
  46. $to = $1*3600+$2*60+($3 ? substr($3,1) : 0);
  47. $watchdog->{RE1} = $re1;
  48. $watchdog->{RE2} = $re2;
  49. $watchdog->{TO} = $to;
  50. $watchdog->{CMD} = $cmd;
  51. if($re1 eq ".") {
  52. watchdog_Activate($watchdog)
  53. } else {
  54. $watchdog->{STATE} = "defined";
  55. }
  56. InternalTimer(1, sub($){
  57. my $re = ($re1 eq "." ? "" : $re1).
  58. (($re2 eq "SAME" || $re2 eq $re1) ? "" : "|$re2");
  59. $re .= "|$name"; # for trigger w .
  60. notifyRegexpChanged($watchdog, $re);
  61. }, $watchdog, 0);
  62. return undef;
  63. }
  64. #####################################
  65. sub
  66. watchdog_Notify($$)
  67. {
  68. my ($watchdog, $dev) = @_;
  69. my $ln = $watchdog->{NAME};
  70. return "" if(IsDisabled($ln) || $watchdog->{STATE} eq "inactive");
  71. my $dontReAct = AttrVal($ln, "regexp1WontReactivate", 0);
  72. my $n = $dev->{NAME};
  73. my $re1 = $watchdog->{RE1};
  74. my $re2 = $watchdog->{RE2};
  75. my $events = deviceEvents($dev, AttrVal($ln, "addStateEvent", 0));
  76. my $max = int(@{$events});
  77. for (my $i = 0; $i < $max; $i++) {
  78. my $s = $events->[$i];
  79. $s = "" if(!defined($s));
  80. my $dotTrigger = ($ln eq $n && $s eq "."); # trigger w .
  81. if($watchdog->{STATE} =~ m/Next:/) {
  82. if($dotTrigger) {
  83. RemoveInternalTimer($watchdog);
  84. $watchdog->{STATE} = "defined";
  85. return;
  86. }
  87. if($n =~ m/^$re2$/ || "$n:$s" =~ m/^$re2$/) {
  88. my $isRe1 = ($re1 eq $re2 || $re1 eq ".");
  89. return if($isRe1 && $dontReAct); # 60414
  90. RemoveInternalTimer($watchdog);
  91. if($re1 eq $re2 || $re1 eq ".") {
  92. watchdog_Activate($watchdog);
  93. return "";
  94. } else {
  95. $watchdog->{STATE} = "defined";
  96. }
  97. } elsif($n =~ m/^$re1$/ || "$n:$s" =~ m/^$re1$/) {
  98. watchdog_Activate($watchdog) if(!$dontReAct);
  99. }
  100. } elsif($watchdog->{STATE} eq "defined") {
  101. if($dotTrigger || ($n =~ m/^$re1$/ || "$n:$s" =~ m/^$re1$/)) {
  102. watchdog_Activate($watchdog)
  103. }
  104. } elsif($dotTrigger) {
  105. $watchdog->{STATE} = "defined"; # trigger w .
  106. }
  107. }
  108. return "";
  109. }
  110. sub
  111. watchdog_Trigger($)
  112. {
  113. my ($watchdog) = @_;
  114. my $name = $watchdog->{NAME};
  115. if(IsDisabled($name) || $watchdog->{STATE} eq "inactive") {
  116. $watchdog->{STATE} = "defined";
  117. return "";
  118. }
  119. Log3 $name, 3, "Watchdog $name triggered";
  120. my $exec = SemicolonEscape($watchdog->{CMD});
  121. $watchdog->{STATE} = "triggered";
  122. setReadingsVal($watchdog, "Triggered", "triggered", TimeNow());
  123. my $ret = AnalyzeCommandChain(undef, $exec);
  124. Log3 $name, 3, $ret if($ret);
  125. if(AttrVal($name, "autoRestart", 0)) {
  126. $watchdog->{STATE} = "defined"; # auto trigger w .
  127. }
  128. }
  129. sub
  130. watchdog_Activate($)
  131. {
  132. my ($watchdog) = @_;
  133. my $nt = gettimeofday() + $watchdog->{TO};
  134. $watchdog->{STATE} = "Next: " . FmtTime($nt);
  135. RemoveInternalTimer($watchdog);
  136. InternalTimer($nt, "watchdog_Trigger", $watchdog, 0);
  137. my $eor = AttrVal($watchdog->{NAME}, "execOnReactivate", undef);
  138. if($eor) {
  139. my $wName = $watchdog->{NAME};
  140. my $aTime = ReadingsTimestamp($wName, "Activated", "");
  141. my $tTime = ReadingsTimestamp($wName, "Triggered", "");
  142. $eor = undef if(!$aTime || !$tTime || $aTime ge $tTime)
  143. }
  144. setReadingsVal($watchdog, "Activated", "activated", TimeNow());
  145. AnalyzeCommandChain(undef, SemicolonEscape($eor)) if($eor);
  146. }
  147. sub
  148. watchdog_Undef($$)
  149. {
  150. my ($hash, $name) = @_;
  151. RemoveInternalTimer($hash);
  152. return undef;
  153. }
  154. sub
  155. watchdog_Attr(@)
  156. {
  157. my ($cmd, $name, $attrName, $attrVal) = @_;
  158. my $do = 0;
  159. my $hash = $defs{$name};
  160. if($cmd eq "set" && $attrName eq "disable") {
  161. $do = (!defined($attrVal) || $attrVal) ? 1 : 2;
  162. }
  163. $do = 2 if($cmd eq "del" && (!$attrName || $attrName eq "disable"));
  164. return if(!$do);
  165. $hash->{STATE} = ($do == 1 ? "disabled" : "defined");
  166. return undef;
  167. }
  168. sub
  169. watchdog_Set($@)
  170. {
  171. my ($hash, @a) = @_;
  172. my $me = $hash->{NAME};
  173. return "no set argument specified" if(int(@a) < 2);
  174. my %sets = (inactive=>0, active=>0);
  175. my $cmd = $a[1];
  176. return "Unknown argument $cmd, choose one of ".join(" ", sort keys %sets)
  177. if(!defined($sets{$cmd}));
  178. return "$cmd needs $sets{$cmd} parameter(s)" if(@a-$sets{$cmd} != 2);
  179. if($cmd eq "inactive") {
  180. readingsSingleUpdate($hash, "state", "inactive", 1);
  181. }
  182. elsif($cmd eq "active") {
  183. readingsSingleUpdate($hash, "state", "defined", 1)
  184. if(!AttrVal($me, "disable", undef));
  185. }
  186. return undef;
  187. }
  188. 1;
  189. =pod
  190. =item helper
  191. =item summary execute a command, if no event is received within timeout
  192. =item summary_DE f&uuml;hrt Befehl aus, falls innerhalb des Timeouts kein Event empfangen wurde
  193. =begin html
  194. <a name="watchdog"></a>
  195. <h3>watchdog</h3>
  196. <ul>
  197. <br>
  198. <a name="watchdogdefine"></a>
  199. <b>Define</b>
  200. <ul>
  201. <code>define &lt;name&gt; watchdog &lt;regexp1&gt; &lt;timespec&gt; &lt;regexp2&gt; &lt;command&gt;</code><br>
  202. <br>
  203. Start an arbitrary FHEM command if after &lt;timespec&gt; receiving an
  204. event matching &lt;regexp1&gt; no event matching &lt;regexp2&gt; is
  205. received.<br>
  206. The syntax for &lt;regexp1&gt; and &lt;regexp2&gt; is the same as the
  207. regexp for <a href="#notify">notify</a>.<br>
  208. &lt;timespec&gt; is HH:MM[:SS]<br>
  209. &lt;command&gt; is a usual fhem command like used in the <a
  210. href="#at">at</a> or <a href="#notify">notify</a>
  211. <br><br>
  212. Examples:
  213. <code><ul>
  214. # Request data from the FHT80 _once_ if we do not receive any message for<br>
  215. # 15 Minutes.<br>
  216. define w watchdog FHT80 00:15:00 SAME set FHT80 date<br>
  217. # Request data from the FHT80 _each_ time we do not receive any message for
  218. <br>
  219. # 15 Minutes, i.e. reactivate the watchdog after it triggered. Might be<br>
  220. # dangerous, as it can trigger in a loop.<br>
  221. define w watchdog FHT80 00:15:00 SAME set FHT80 date;; trigger w .<br>
  222. # Shout once if the HMS100-FIT is not alive<br>
  223. define w watchdog HMS100-FIT 01:00:00 SAME "alarm-fit.sh"<br>
  224. # Send mail if the window is left open<br>
  225. define w watchdog contact1:open 00:15 contact1:closed
  226. "mail_me close window1"<br>
  227. attr w regexp1WontReactivate<br>
  228. </ul></code>
  229. Notes:<br>
  230. <ul>
  231. <li>if &lt;regexp1&gt; is . (dot), then activate the watchdog at
  232. definition time. Else it will be activated when the first matching
  233. event is received.</li>
  234. <li>&lt;regexp1&gt; resets the timer of a running watchdog, to avoid it
  235. use the regexp1WontReactivate attribute.</li>
  236. <li>if &lt;regexp2&gt; is SAME, then it will be the same as the first
  237. regexp, and it will be reactivated, when it is received.
  238. </li>
  239. <li>trigger &lt;watchdogname&gt; . will activate the trigger if its state
  240. is defined, and set it into state defined if its state is active
  241. (Next:...) or triggered. You always have to reactivate the watchdog
  242. with this command once it has triggered (unless you restart
  243. fhem)</li>
  244. <li>a generic watchdog (one watchdog responsible for more devices) is
  245. currently not possible.</li>
  246. <li>with modify all parameters are optional, and will not be changed if
  247. not specified.</li>
  248. </ul>
  249. <br>
  250. </ul>
  251. <a name="watchdogset"></a>
  252. <b>Set </b>
  253. <ul>
  254. <li>inactive<br>
  255. Inactivates the current device. Note the slight difference to the
  256. disable attribute: using set inactive the state is automatically saved
  257. to the statefile on shutdown, there is no explicit save necesary.<br>
  258. This command is intended to be used by scripts to temporarily
  259. deactivate the notify.<br>
  260. The concurrent setting of the disable attribute is not recommended.</li>
  261. <li>active<br>
  262. Activates the current device (see inactive).</li>
  263. </ul>
  264. <br>
  265. <a name="watchdogget"></a>
  266. <b>Get</b> <ul>N/A</ul><br>
  267. <a name="watchdogattr"></a>
  268. <b>Attributes</b>
  269. <ul>
  270. <li><a href="#disable">disable</a></li>
  271. <li><a href="#disabledForIntervals">disabledForIntervals</a></li>
  272. <li><a name="regexp1WontReactivate">regexp1WontReactivate</a><br>
  273. When a watchdog is active, a second event matching regexp1 will
  274. normally reset the timeout. Set this attribute to prevents this.</li>
  275. <li><a href="#execOnReactivate">execOnActivate</a>
  276. If set, its value will be executed as a FHEM command when the watchdog is
  277. reactivated (after triggering) by receiving an event matching regexp1.
  278. </li>
  279. <li><a href="#autoRestart">autoRestart</a>
  280. When the watchdog has triggered it will be automatically re-set to state
  281. defined again (waiting for regexp1) if this attribute is set to 1.
  282. </li>
  283. </ul>
  284. <br>
  285. </ul>
  286. =end html
  287. =begin html_DE
  288. <a name="watchdog"></a>
  289. <h3>watchdog</h3>
  290. <ul>
  291. <br>
  292. <a name="watchdogdefine"></a>
  293. <b>Define</b>
  294. <ul>
  295. <code>define &lt;name&gt; watchdog &lt;regexp1&gt; &lt;timespec&gt;
  296. &lt;regexp2&gt; &lt;command&gt;</code><br>
  297. <br>
  298. Startet einen beliebigen FHEM Befehl wenn nach dem Empfang des
  299. Ereignisses &lt;regexp1&gt; nicht innerhalb von &lt;timespec&gt; ein
  300. &lt;regexp2&gt; Ereignis empfangen wird.<br>
  301. Der Syntax f&uuml;r &lt;regexp1&gt; und &lt;regexp2&gt; ist der gleiche wie
  302. regexp f&uuml;r <a href="#notify">notify</a>.<br>
  303. &lt;timespec&gt; ist HH:MM[:SS]<br>
  304. &lt;command&gt; ist ein gew&ouml;hnlicher fhem Befehl wie z.B. in <a
  305. href="#at">at</a> oderr <a href="#notify">notify</a>
  306. <br><br>
  307. Beispiele:
  308. <code><ul>
  309. # Frage Daten vom FHT80 _einmalig_ ab, wenn wir keine Nachricht f&uuml;r<br>
  310. # 15 Minuten erhalten haben.<br>
  311. define w watchdog FHT80 00:15:00 SAME set FHT80 date<br><br>
  312. # Frage Daten vom FHT80 jedes Mal ab, wenn keine Nachricht f&uuml;r<br>
  313. # 15 Minuten emfpangen wurde, d.h. reaktiviere den Watchdog nachdem er
  314. getriggert wurde.<br>
  315. # Kann gef&auml;hrlich sein, da er so in einer Schleife getriggert werden
  316. kann.<br>
  317. define w watchdog FHT80 00:15:00 SAME set FHT80 date;; trigger w .<br><br>
  318. # Alarmiere einmalig wenn vom HMS100-FIT f&uuml;r eine Stunde keine
  319. Nachricht empfangen wurde.<br>
  320. define w watchdog HMS100-FIT 01:00 SAME "alarm-fit.sh"<br><br>
  321. # Sende eine Mail wenn das Fenster offen gelassen wurde<br>
  322. define w watchdog contact1:open 00:15 contact1:closed "mail_me close
  323. window1"<br>
  324. attr w regexp1WontReactivate<br><br>
  325. </ul></code>
  326. Hinweise:<br>
  327. <ul>
  328. <li>Wenn &lt;regexp1&gt; . (Punkt) ist, dann aktiviere den Watchdog zur
  329. definierten Zeit. Sonst wird er durch den Empfang des ersten passenden
  330. Events aktiviert.</li>
  331. <li>&lt;regexp1&gt; Resetet den Timer eines laufenden Watchdogs. Um das
  332. zu verhindern wird das regexp1WontReactivate Attribut gesetzt.</li>
  333. <li>Wenn &lt;regexp2&gt; SAME ist , dann ist es das gleiche wie das erste
  334. regexp, und wird reaktiviert wenn es empfangen wird. </li>
  335. <li>trigger &lt;watchdogname&gt; . aktiviert den Trigger wenn dessen
  336. Status defined ist und setzt ihn in den Status defined wenn sein status
  337. triggered oder aktiviert (Next:...) ist.<br>
  338. Der Watchdog musst immer mit diesem Befehl reaktiviert werden wenn er
  339. getriggert wurde.</li>
  340. <li>Ein generischer Watchdog (ein Watchdog, verantwortlich f&uuml;r
  341. mehrere Devices) ist derzeit nicht m&ouml;glich.</li>
  342. <li>Bei modify sind alle Parameter optional, und werden nicht geaendert,
  343. falls nicht spezifiziert.</li>
  344. </ul>
  345. <br>
  346. </ul>
  347. <a name="watchdogset"></a>
  348. <b>Set </b>
  349. <ul>
  350. <li>inactive<br>
  351. Deaktiviert das entsprechende Ger&auml;t. Beachte den leichten
  352. semantischen Unterschied zum disable Attribut: "set inactive"
  353. wird bei einem shutdown automatisch in fhem.state gespeichert, es ist
  354. kein save notwendig.<br>
  355. Der Einsatzzweck sind Skripte, um das notify tempor&auml;r zu
  356. deaktivieren.<br>
  357. Das gleichzeitige Verwenden des disable Attributes wird nicht empfohlen.
  358. </li>
  359. <li>active<br>
  360. Aktiviert das entsprechende Ger&auml;t, siehe inactive.
  361. </li>
  362. </ul>
  363. <br>
  364. <a name="watchdogget"></a>
  365. <b>Get</b> <ul>N/A</ul><br>
  366. <a name="watchdogattr"></a>
  367. <b>Attribute</b>
  368. <ul>
  369. <li><a href="#addStateEvent">addStateEvent</a></li>
  370. <li><a href="#disable">disable</a></li>
  371. <li><a href="#disabledForIntervals">disabledForIntervals</a></li>
  372. <li><a name="regexp1WontReactivate">regexp1WontReactivate</a><br>
  373. Wenn ein Watchdog aktiv ist, wird ein zweites Ereignis das auf regexp1
  374. passt normalerweise den Timer zur&uuml;cksetzen. Dieses Attribut wird
  375. das verhindern.</li>
  376. <li><a href="#execOnReactivate">execOnActivate</a>
  377. Falls gesetzt, wird der Wert des Attributes als FHEM Befehl
  378. ausgef&uuml;hrt, wenn ein regexp1 Ereignis den Watchdog
  379. aktiviert nachdem er ausgel&ouml;st wurde.</li>
  380. <li><a href="#autoRestart">autoRestart</a>
  381. Wenn dieses Attribut gesetzt ist, wird der Watchdog nach dem er
  382. getriggert wurde, automatisch wieder in den Zustand defined
  383. gesetzt (Wartet also wieder auf Aktivierung durch regexp1)</li>
  384. </ul>
  385. <br>
  386. </ul>
  387. =end html_DE
  388. =cut