98_Verkehrsinfo.pm 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868
  1. # $Id: 98_Verkehrsinfo.pm 13112 2017-01-16 19:18:49Z martins $
  2. ############################################################################
  3. #
  4. # 98_Verkehrsinfo.pm
  5. #
  6. # Copyright (C) 2016 by Martin Schubert
  7. # e-mail: martin@dermschub.de
  8. #
  9. # This program is free software; you can redistribute it and/or modify
  10. # it under the terms of the GNU General Public License as published by
  11. # the Free Software Foundation; either version 2 of the License, or
  12. # (at your option) any later version.
  13. #
  14. # This program is distributed in the hope that it will be useful,
  15. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. # GNU General Public License for more details.
  18. #
  19. # You should have received a copy of the GNU General Public License
  20. # along with fhem. If not, see <http://www.gnu.org/licenses/>.
  21. #
  22. ############################################################################
  23. ############################################################################
  24. #
  25. # Changelog:
  26. # 2017-01-16, v2.2
  27. # Feature: Quelle http://www.radiosaw.de/verkehrsmeldungen added
  28. #
  29. # 2016-12-26, v2.1
  30. # Bugfix: update state with readings update
  31. # CHANGE: remove requirement perl-module json
  32. # CHANGE: update commandref with link to readingFn-Attributen
  33. #
  34. # Changelog:
  35. # 2016-12-13, v2.0RC1
  36. # Bugfix: Bugfix Hessenschau Message
  37. #
  38. # 2016-11-17, v2.0
  39. # CHANGE: Module change to HttpNonBlocking
  40. # CHANGE: remove requirement perl-module json
  41. # CHANGE: update commandref
  42. #
  43. # 2016-11-10, v1.9
  44. # Bugfix: hessenschau.de, messages end with a point
  45. #
  46. # Changelog:
  47. # 2016-10-24, v1.8
  48. # Bugfix: hessenschau.de, messages end with a point
  49. #
  50. # 2016-10-23, v1.7RC2
  51. # CHANGE: Module change to NonBlocking
  52. # CHANGE: Code optimization
  53. # CHANGE: update commandref
  54. # Bugfix: hessenschau.de, Cant call method "as_trimmed_text" fixed
  55. #
  56. # 2016-08-17, v1.7RC1
  57. # Feature: New Reading for humanly readable message
  58. # Feature: New Attribut for Messageformat
  59. # Feature: New Attribut for Sorting added
  60. # CHANGE: State Value
  61. # CHANGE: LWP::Simple replace to HttpUtils
  62. # CHANGE: check if HTML::TreeBuilder::XPath is installed
  63. # CHANGE: update commandref
  64. #
  65. # 2016-08-17, v1.6
  66. # Bugfix: verkehrsinfo.de, Display of the zone has been corrected
  67. # Bugfix: Characterset has been corrected
  68. #
  69. # 2016-08-16, v1.5
  70. # Bugfix: verkehrsinfo.de, URL changed to the new address
  71. #
  72. # 2016-08-04, v1.4
  73. # Bugfix: hessenschau.de, Message Sperrung added
  74. #
  75. # 2016-07-29, v1.3
  76. # Bugfix: hessenschau.de, Message Warnung added
  77. #
  78. # 2016-07-11, v1.2
  79. # Feature: Quelle http://hessenschau.de/verkehr/index.html added
  80. #
  81. # 2016-07-03, v1.1
  82. # Bugfix: Check if a valid URL was passed
  83. # Feature: Include Filterattribut added, regex available, Pipe as delimiter
  84. # Feature: Exclude Filterattribut added, regex available, Pipe as delimiter
  85. #
  86. # 2016-06-29, v1.0
  87. # Initzial Version
  88. #
  89. ############################################################################
  90. package main;
  91. use strict;
  92. use warnings;
  93. #use Encode qw(decode encode);
  94. use HttpUtils;
  95. my $missingModul = "";
  96. eval "use HTML::TreeBuilder::XPath;1" or $missingModul .= "HTML::TreeBuilder::XPath ";
  97. my %Verkehrsinfo_gets = (
  98. "update" => "noArg",
  99. "info" => "noArg"
  100. );
  101. #my $encode = 'UTF-8';
  102. sub Verkehrsinfo_Initialize($) {
  103. my ($hash) = @_;
  104. $hash->{DefFn} = 'Verkehrsinfo_Define';
  105. $hash->{UndefFn} = 'Verkehrsinfo_Undef';
  106. $hash->{SetFn} = 'Verkehrsinfo_Set';
  107. $hash->{GetFn} = 'Verkehrsinfo_Get';
  108. $hash->{AttrFn} = 'Verkehrsinfo_Attr';
  109. $hash->{ReadFn} = 'Verkehrsinfo_Read';
  110. $hash->{AttrList} =
  111. "filter_exclude filter_include orderby "
  112. . "msg_format:road,head,both "
  113. . $readingFnAttributes;
  114. }
  115. sub Verkehrsinfo_Define($$) {
  116. my ($hash, $def) = @_;
  117. my @param = split('[ \t]+', $def);
  118. if(int(@param) < 4) {
  119. return "too few parameters: define <name> Verkehrsinfo <url> <interval>";
  120. }
  121. $hash->{name} = $param[0];
  122. $hash->{url} = $param[2];
  123. $hash->{Interval} = $param[3];
  124. # check Module is installed
  125. if ( $missingModul ) {
  126. my $returnmsg = "Cannot define $hash->{name} device. Perl modul $missingModul is missing.";
  127. Log3 $hash->{name}, 1, $returnmsg;
  128. return $returnmsg;
  129. }
  130. # check if the url is ok
  131. if ($hash->{url} !~ /verkehrsinfo\.de\/httpsmobil\/index\.php/ &&
  132. $hash->{url} !~ /hessenschau\.de\/verkehr\/index\.html/ &&
  133. $hash->{url} !~ /radiosaw/){
  134. my $returnmsg = "Diese URL wird nicht unterstützt. Bitte schauen Sie in die Modulbeschreibung.";
  135. Log3 $hash->{name}, 1, $returnmsg;
  136. return $returnmsg;
  137. }
  138. # get Zone name
  139. if ($hash->{url} =~ /verkehrsinfo.de/i){
  140. my $param = {
  141. url => "https://www.verkehrsinfo.de/httpsmobil/index.php?c=1&lat=&lon=",
  142. timeout => 5,
  143. hash => $hash,
  144. callback => \&Verkehrsinfo_HttpNbDefineZone
  145. };
  146. HttpUtils_NonblockingGet($param);
  147. }
  148. elsif ($hash->{url} =~ /hessenschau.de/i) {
  149. readingsSingleUpdate( $hash, "zone", "Hessen", 1 );
  150. }
  151. elsif ($hash->{url} =~ /radiosaw/i) {
  152. readingsSingleUpdate( $hash, "zone", "SAW", 1 );
  153. $hash->{url} = 'http://www.radiosaw.de/verkehrsmeldungen';
  154. }
  155. InternalTimer(gettimeofday()+4, "Verkehrsinfo_GetUpdate", $hash, 0);
  156. readingsSingleUpdate($hash, "state", 'initialized',1 );
  157. return undef;
  158. }
  159. # recieve zone from www http nonblocking
  160. sub Verkehrsinfo_HttpNbDefineZone($) {
  161. my ($param, $err, $content) = @_;
  162. my $hash = $param->{hash};
  163. my $name = $hash->{NAME};
  164. Log3 $hash, 4, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbDefineZone start";
  165. if($err ne "") {
  166. Log3 $name, 3, "error while requesting ".$param->{url}." - $err";
  167. readingsSingleUpdate($hash, "state", 'ERROR Update Zone ' . FmtDateTime(time()), 1);
  168. }
  169. elsif($content ne "")
  170. {
  171. my $tree = HTML::TreeBuilder->new;
  172. my @arrzone = split(/[=&]/, $hash->{url});
  173. my $zone = '';
  174. if ($arrzone[2] eq "bl")
  175. {
  176. # prepare HTML Code
  177. $content =~ s/getLocation\(\)\;//g;
  178. #####################
  179. $tree->parse($content); #<-- Testen !
  180. #$tree->parse(encode($encode, $content));
  181. $zone = $tree->findnodes('//button[contains(@onclick, "'.$arrzone[3].'")]')->[0]->as_trimmed_text;
  182. $zone =~ s/\s\[.*\]//;
  183. }
  184. else {
  185. $zone = $arrzone[3];
  186. }
  187. readingsSingleUpdate( $hash, "zone", $zone, 1 );
  188. }
  189. Log3 $hash, 4, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbDefineZone done";
  190. }
  191. sub Verkehrsinfo_Undef($$) {
  192. my ($hash, $arg) = @_;
  193. RemoveInternalTimer($hash);
  194. return undef;
  195. }
  196. sub Verkehrsinfo_Get($@) {
  197. my ($hash, @param) = @_;
  198. return '"get Verkehrsinfo" needs at least one argument' if (int(@param) < 2);
  199. my $name = shift @param;
  200. my $opt = shift @param;
  201. if(!$Verkehrsinfo_gets{$opt}) {
  202. return "Unknown argument $opt, choose one of info:noArg";
  203. }
  204. if ($opt eq "info"){
  205. return Verkehrsinfo_GetData($hash->{NAME});
  206. }
  207. }
  208. sub Verkehrsinfo_Set($@) {
  209. my ($hash, @param) = @_;
  210. return '"set Verkehrsinfo" needs at least one argument' if (int(@param) < 2);
  211. my $name = shift @param;
  212. my $opt = shift @param;
  213. my $value = join("", @param);
  214. if(!defined($Verkehrsinfo_gets{$opt})) {
  215. return "Unknown argument $opt, choose one of update:noArg";
  216. }
  217. if ($opt eq "update"){
  218. Verkehrsinfo_GetUpdate($hash);
  219. #return "Update is runing";
  220. return undef;
  221. }
  222. }
  223. sub Verkehrsinfo_Attr(@) {
  224. my ($cmd,$name,$attr_name,$attr_value) = @_;
  225. if($cmd eq "set") {
  226. if($attr_name eq "filter_exclude" || $attr_name eq "filter_include") {
  227. eval { qr/$attr_value/ };
  228. if ($@) {
  229. my $err = "Verkehrsinfo: Ungültiger Filter in attr $name $attr_name $attr_value: $@";
  230. Log3 $name, 3, $err;
  231. return $err;
  232. }
  233. }
  234. elsif($attr_name eq "msg_format" && $attr_value !~ "road|head|both") {
  235. my $err = "Verkehrsinfo: Ungültiges Message Format in attr $name $attr_name $attr_value: $@";
  236. Log3 $name, 3, $err;
  237. return $err;
  238. }
  239. elsif($attr_name eq "msg_format" && InternalVal($name, 'url', '') =~ 'hessenschau.de/verkehr') {
  240. my $err = "Verkehrsinfo: Message Format ist für " . InternalVal($name, 'url', '') . " nicht Verfügbar";
  241. Log3 $name, 3, $err;
  242. return $err;
  243. }
  244. }
  245. return undef;
  246. }
  247. # recieve data from www http nonblocking
  248. sub Verkehrsinfo_HttpNbUpdateData ($) {
  249. my ($param, $err, $content) = @_;
  250. my $hash = $param->{hash};
  251. my $name = $hash->{NAME};
  252. my $headline;
  253. my @toc;
  254. my @toc2;
  255. my $message = '';
  256. my $message_zone = '';
  257. my $message_head = '';
  258. my $dataarray;
  259. Log3 $hash, 4, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbUpdateData start";
  260. if($err ne "") {
  261. Log3 $name, 3, "error while requesting ".$param->{url}." - $err";
  262. readingsSingleUpdate($hash, "state", 'ERROR Update Readings ' . FmtDateTime(time()), 1);
  263. }
  264. elsif($content ne "")
  265. {
  266. # read attribut filter
  267. my $filterexclude = AttrVal($name,"filter_exclude","");
  268. my $filterinclude = AttrVal($name,"filter_include",".*");
  269. my $orderby = AttrVal($name,"orderby","");
  270. my $tree = HTML::TreeBuilder->new;
  271. my $i = 1;
  272. ##################
  273. # verkehrsinfo.de
  274. ##################
  275. if ($hash->{url} =~ /verkehrsinfo.de/i){
  276. # prepare HTML Code
  277. $content =~ s/getLocation\(\)\;//g;
  278. $tree->parse($content);
  279. @toc = $tree->findnodes('//div[contains(@class, "panel-body")]/ul/li');
  280. shift(@toc); # delete advertising
  281. for my $el ( Verkehrsinfo_hf_orderby($orderby, @toc) ) {
  282. if (grep(!/$filterexclude/i, $el->as_trimmed_text) && grep(/$filterinclude/i, $el->as_trimmed_text)){
  283. if (exists $el->findnodes('div/div')->[0] && exists $el->findnodes('div')->[1]->findnodes('span')->[0] && exists $el->findnodes('div')->[1]->findnodes('span')->[1]){
  284. $dataarray->{"e_".$i."_road"} = $el->findnodes('div/div')->[0]->as_trimmed_text;
  285. $dataarray->{"e_".$i."_head"} = $el->findnodes('div')->[1]->findnodes('span')->[0]->as_trimmed_text;
  286. $dataarray->{"e_".$i."_msg"} = $el->findnodes('div')->[1]->findnodes('span')->[1]->as_trimmed_text;
  287. $message .= (AttrVal($name,"msg_format","") =~ "road|both") ? $el->findnodes('div/div')->[0]->as_trimmed_text .', ' : '';
  288. $message .= (AttrVal($name,"msg_format","") =~ "head|both") ? $el->findnodes('div')->[1]->findnodes('span')->[1]->as_trimmed_text .', ' : '';
  289. $message .= $el->findnodes('div')->[1]->findnodes('span')->[1]->as_trimmed_text .'. ' ;
  290. $i++;
  291. }
  292. else{
  293. Log3 $hash, 3, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbUpdateData DataNodeElements not found";
  294. }
  295. }
  296. }
  297. $message =~ s/ \(.*?\)//g; # Remove number of exits
  298. $message_zone = (ReadingsVal($name, 'zone', '') =~ /[0-9]/i) ? ' für die ' : ' für ';
  299. $message_zone .= ReadingsVal($name, 'zone', '');
  300. }
  301. ##################
  302. # hessenschau.de
  303. ##################
  304. elsif ($hash->{url} =~ /hessenschau.de/i) {
  305. # prepare HTML Code
  306. $content =~ s/<\/title>/<\/span>/g;
  307. $content =~ s/<title id="iconTitle--[0-9]+">/<span>/g;
  308. $content =~ s/<text/<p/g;
  309. $content =~ s/<\/text>/<\/p>/g;
  310. $content =~ s/<use.*traffic-arrow.*>.*<\/use>/Richtung/g;
  311. #$tree->parse_content(encode($encode, $content));
  312. $tree->parse_content($content);
  313. $message_zone = ' für Hessen';
  314. @toc = $tree->findnodes('//li[contains(@class, "trafficInfo__item")]');
  315. for my $el ( Verkehrsinfo_hf_orderby($orderby, @toc) ) {
  316. if (grep(!/$filterexclude/i, $el->as_trimmed_text) && grep(/$filterinclude/i, $el->as_trimmed_text)){
  317. if (exists $el->findnodes('div/p')->[0] && exists $el->findnodes('div/span')->[0]){
  318. # check message if it's road or information
  319. if ($el->findnodes('div/p')->[0]->as_trimmed_text =~ /^[0-9]/){
  320. $dataarray->{"e_".$i."_road"} = substr($el->findnodes('div/span')->[0]->as_trimmed_text, 0 ,1).
  321. $el->findnodes('div/p')->[0]->as_trimmed_text;
  322. $dataarray->{"e_".$i."_head"} = (exists $el->findnodes('div/strong')->[0]) ? $el->findnodes('div/strong')->[0]->as_trimmed_text : '';
  323. $dataarray->{"e_".$i."_msg"} = (exists $el->findnodes('div/p')->[1]) ? $el->findnodes('div/p')->[1]->as_trimmed_text : '';
  324. $message .= (exists $el->findnodes('div/p')->[1]) ? $el->findnodes('div/p')->[1]->as_trimmed_text : '';
  325. $message .= ($el->findnodes('div/p')->[1]->as_trimmed_text =~ /\.$/) ? ' ' : '. ';
  326. }
  327. else {
  328. $dataarray->{"e_".$i."_road"} = '-';
  329. $dataarray->{"e_".$i."_head"} = $el->findnodes('div/span')->[0]->as_trimmed_text;
  330. $dataarray->{"e_".$i."_msg"} = $el->findnodes('div/p')->[0]->as_trimmed_text;
  331. $message .= (AttrVal($name,"msg_format","") =~ "head|both") ? $el->findnodes('div/span')->[0]->as_trimmed_text .', ' : '';
  332. $message .= $el->findnodes('div/p')->[0]->as_trimmed_text;
  333. $message .= ($el->findnodes('div/p')->[0]->as_trimmed_text =~ /\.$/) ? ' ' : '. ' ;
  334. }
  335. $i++;
  336. }
  337. else{
  338. Log3 $hash, 3, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbUpdateData DataNodeElements not found";
  339. }
  340. }
  341. }
  342. }
  343. ##################
  344. # radiosaw.de
  345. ##################
  346. elsif ($hash->{url} =~ /radiosaw.de/i) {
  347. $message_zone = ' für SAW';
  348. # part one meldungen
  349. $tree->parse_content($content);
  350. @toc = $tree->findnodes('//div[contains(@id, "block-system-main")]/div/div[2]/div[2]/div/div/div');
  351. # part two baustellen
  352. @toc2 = $tree->findnodes('//div[contains(@id, "block-system-main")]/div/div[3]/div[1]/div/div[1]/div/div[2]/div/div/div/ul/li');
  353. push (@toc, @toc2);
  354. for my $el ( Verkehrsinfo_hf_orderby($orderby, @toc) ) {
  355. if (grep(!/$filterexclude/i, $el->as_trimmed_text) && grep(/$filterinclude/i, $el->as_trimmed_text)){
  356. if ($el->as_trimmed_text =~ /^Aktuelle Baustelle/){
  357. my $tmp = $el->as_HTML;
  358. $tmp =~ s/<br \/>-+<br \/>/\|/g;
  359. $tmp =~ s/<br \/>/###/g;
  360. my $subtree = HTML::TreeBuilder->new;
  361. $subtree->parse_content($tmp);
  362. my @tmp2 = split(/\|/, $subtree->as_trimmed_text);
  363. shift @tmp2;
  364. for my $sel ( @tmp2 ){
  365. $dataarray->{"e_".$i."_road"} = ((split(/\s/, $sel))[0] =~ /^.[0-9]+/) ? (split(/\s/, $sel))[0] : '-';
  366. $dataarray->{"e_".$i."_head"} = (split(/\###/, $sel))[0];
  367. $dataarray->{"e_".$i."_msg"} = $sel;
  368. $dataarray->{"e_".$i."_msg"} =~ s/^.*?###//;
  369. $dataarray->{"e_".$i."_msg"} =~ s/###/ /g;
  370. $message .= (AttrVal($name,"msg_format","") =~ "road|both") ? $dataarray->{"e_".$i."_road"} .', ' : '';
  371. $message .= (AttrVal($name,"msg_format","") =~ "head|both") ? $dataarray->{"e_".$i."_head"} .', ' : '';
  372. $message .= $dataarray->{"e_".$i."_msg"};
  373. $message .= ($dataarray->{"e_".$i."_msg"} =~ /\.$/) ? ' ' : '. ' ;
  374. $i++;
  375. }
  376. $i--;
  377. }
  378. elsif ($el->findnodes('../li')->[0]){
  379. if (exists $el->findnodes('strong')->[0]){
  380. $dataarray->{"e_".$i."_road"} = ((split(/\s/, $el->findnodes('strong')->[0]->as_trimmed_text .' -'))[1] =~ /^[0-9]+/) ? (split(/\s/, $el->findnodes('strong')->[0]->as_trimmed_text))[0] . ' ' . (split(/\s/, $el->findnodes('strong')->[0]->as_trimmed_text))[1] : '-';
  381. $dataarray->{"e_".$i."_head"} = $el->findnodes('strong')->[0]->as_trimmed_text;
  382. $dataarray->{"e_".$i."_msg"} = $el->as_trimmed_text;
  383. $dataarray->{"e_".$i."_msg"} =~ s/\(/_ko_/g;
  384. $dataarray->{"e_".$i."_msg"} =~ s/\)/_kc_/g;
  385. my $tmp = $el->findnodes('strong')->[0]->as_trimmed_text;
  386. $tmp =~ s/\(/_ko_/g;
  387. $tmp =~ s/\)/_kc_/g;
  388. $dataarray->{"e_".$i."_msg"} =~ s/$tmp//;
  389. }
  390. else{
  391. $dataarray->{"e_".$i."_road"} = '-';
  392. $dataarray->{"e_".$i."_head"} = '-';
  393. $dataarray->{"e_".$i."_msg"} = $el->as_trimmed_text;
  394. }
  395. $dataarray->{"e_".$i."_msg"} =~ s/_ko_/\(/g;
  396. $dataarray->{"e_".$i."_msg"} =~ s/_kc_/\)/g;
  397. $message .= (AttrVal($name,"msg_format","") =~ "road|both") ? $dataarray->{"e_".$i."_road"} .', ' : '';
  398. $message .= (AttrVal($name,"msg_format","") =~ "head|both") ? $dataarray->{"e_".$i."_head"} .', ' : '';
  399. $message .= $dataarray->{"e_".$i."_msg"};
  400. $message .= ($dataarray->{"e_".$i."_msg"} =~ /\.$/) ? ' ' : '. ' ;
  401. $i++;
  402. }
  403. else {
  404. my $tmp = $el->as_HTML;
  405. $tmp =~ s/<br \/>/###/g;
  406. my $subtree = HTML::TreeBuilder->new;
  407. $subtree->parse_content($tmp);
  408. my $anz = scalar(split(/\###/, $subtree->as_trimmed_text));
  409. $dataarray->{"e_".$i."_road"} = ((split(/\s/, $subtree->as_trimmed_text))[0] =~ /^.[0-9]+/) ? (split(/\s/, $subtree->as_trimmed_text))[0] : '-';
  410. $dataarray->{"e_".$i."_head"} = ($anz == 2) ? (split(/\###/, $subtree->as_trimmed_text))[0] : '-';
  411. $dataarray->{"e_".$i."_msg"} = ($anz == 2) ? (split(/\###/, $subtree->as_trimmed_text))[1] : $subtree->as_trimmed_text;
  412. $message .= (AttrVal($name,"msg_format","") =~ "road|both") ? $dataarray->{"e_".$i."_road"} .', ' : '';
  413. $message .= (AttrVal($name,"msg_format","") =~ "head|both") ? $dataarray->{"e_".$i."_head"} .', ' : '';
  414. $message .= $dataarray->{"e_".$i."_msg"};
  415. $message .= ($dataarray->{"e_".$i."_msg"} =~ /\.$/) ? ' ' : '. ' ;
  416. $i++;
  417. }
  418. # else{
  419. # Log3 $hash, 3, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbUpdateData DataNodeElements not found";
  420. # }
  421. }
  422. }
  423. }
  424. if ($i - 1 == 0){
  425. $message_head = "Es liegen um " . strftime('%H:%M', localtime) . $message_zone . " keine Staumeldungen vor.";
  426. }
  427. elsif ($i - 1 == 1){
  428. $message_head = "Es liegt um " . strftime('%H:%M', localtime) . $message_zone . " eine Staumeldung vor:\n";
  429. }
  430. else{
  431. my $anz_msg = $i - 1;
  432. $message_head = "Es liegen um " . strftime('%H:%M', localtime) . $message_zone . ', ' . $anz_msg ." Staumeldungen vor:\n";
  433. }
  434. $dataarray->{'message'} = $message_head . ' ' . $message;
  435. $dataarray->{'message'} =~ s/\<pre\>//;
  436. $dataarray->{'message'} =~ s/\<\/pre\>//;
  437. $dataarray->{'count'} = $i - 1;
  438. # delete old readings
  439. Log3 $hash, 4, "Verkehrsinfo: ($name) Delete old Readings";
  440. CommandDeleteReading(undef, "$hash->{NAME} e_.*_.*");
  441. Log3 $hash, 4, "Verkehrsinfo: ($name) Create new Readings";
  442. # update readings
  443. readingsBeginUpdate($hash);
  444. readingsBulkUpdate($hash, "date_time", FmtDateTime(time()) );
  445. foreach my $readingName (keys %{$dataarray}){
  446. Log3 $hash, 4, "Verkehrsinfo: ($name) ReadingsUpdate: $readingName - ".$dataarray->{$readingName};
  447. readingsBulkUpdate($hash,$readingName,$dataarray->{$readingName});
  448. }
  449. readingsBulkUpdate($hash, "state", 'update ' . FmtDateTime(time()));
  450. readingsEndUpdate($hash, 1);
  451. }
  452. Log3 $hash, 4, "Verkehrsinfo: ($name) Verkehrsinfo_HttpNbUpdateData done";
  453. }
  454. sub Verkehrsinfo_GetUpdate($) {
  455. my ($hash) = @_;
  456. my $name = $hash->{NAME};
  457. if ( $hash->{Interval}) {
  458. RemoveInternalTimer ($hash);
  459. }
  460. InternalTimer(gettimeofday()+$hash->{Interval}, "Verkehrsinfo_GetUpdate", $hash, 1);
  461. Log3 $hash, 4, "Verkehrsinfo: ($name) internal interval timer set to call GetUpdate again in " . int($hash->{Interval}). " seconds";
  462. my $param = {
  463. url => $hash->{url},
  464. timeout => 5,
  465. hash => $hash,
  466. callback => \&Verkehrsinfo_HttpNbUpdateData
  467. };
  468. HttpUtils_NonblockingGet($param);
  469. }
  470. # give back all traffic information as formated text
  471. sub Verkehrsinfo_GetData($){
  472. my ($device) = @_;
  473. my $hash = $defs{$device};
  474. if (!defined $device){
  475. Log3 $hash, 1, "Verkehrsinfo: ($device) Device not found";
  476. return "Device not found";
  477. }
  478. my $msg = '';
  479. my $i = 1;
  480. $msg = ReadingsVal($device, 'count', '') . " Meldungen für ". ReadingsVal($device, 'zone', '') .":\n\n";
  481. for ($i=1; $i <= ReadingsVal($device, 'count', ''); $i++){
  482. $msg = $msg . ReadingsVal($device, 'e_'.$i.'_road', '') . " - ";
  483. $msg = $msg . ReadingsVal($device, 'e_'.$i.'_head', '') . "\n";
  484. $msg = $msg . ReadingsVal($device, 'e_'.$i.'_msg', '') . "\n\n";
  485. }
  486. return $msg;
  487. }
  488. ##################
  489. # helper function
  490. ##################
  491. # sort messages
  492. sub Verkehrsinfo_hf_orderby ($@) {
  493. my ($order, @inp) = @_;
  494. my @res;
  495. my %diff;
  496. for my $oel (split(/\|/, $order)) {
  497. for my $ael ( @inp ) {
  498. push(@res, $ael) if (grep (/$oel/i, $ael->as_trimmed_text));
  499. }
  500. }
  501. @diff{ @inp } = @inp;
  502. delete @diff{ @res };
  503. return (@res, values %diff);
  504. }
  505. 1;
  506. =pod
  507. =item device
  508. =item summary read trafficinformation from various sources
  509. =item summary_DE Verkehrsinformationen von verschiedenen Quellen auslesen.
  510. =begin html
  511. <a name="Verkehrsinfo"></a>
  512. <h3>Verkehrsinfo</h3>
  513. <ul>
  514. <i>Verkehrsinfo</i> can read trafficinformation from various source.
  515. <br><br>
  516. <ul>
  517. <li>Verkehrsinfo.de</li>
  518. For receiving the traffic informationen, following website https://www.verkehrsinfo.de/httpsmobil will be called on.<br>
  519. There you can select streets or federal states. Afterwards the URL will be committed as a parameter.
  520. <br><br>
  521. <li>Hessenschau.de</li>
  522. Here is no configuration necessary, the URL http://hessenschau.de/verkehr/index.html will be used as a parameter.
  523. <br><br>
  524. <li>RadioSAW.de</li>
  525. Here is no configuration necessary, the keyword radiosaw will be used as a parameter.
  526. </ul>
  527. <br><br>
  528. <b>Requirement:</b>
  529. <ul><br>
  530. For this module, following perl-modules are required:<br>
  531. <li>HTML::TreeBuilder::XPath<br>
  532. <code>sudo apt-get install libxml-treebuilder-perl libhtml-treebuilder-xpath-perl</code>
  533. </li>
  534. </ul>
  535. <br><br>
  536. <a name="Verkehrsinfodefine"></a>
  537. <b>Define</b>
  538. <ul>
  539. <code>define &lt;name&gt; Verkehrsinfo &lt;url&gt; &lt;interval&gt;</code>
  540. <br><br>
  541. example: <code>define A8 Verkehrsinfo https://www.verkehrsinfo.de/httpsmobil/index.php?c=staulist&street=A8&lat=&lon= 3600 </code>
  542. <br><br>
  543. Options:
  544. <ul>
  545. <li><i>url</i><br>
  546. URL regarding the traffic information</li>
  547. <li><i>interval</i><br>
  548. How often the data will be updated in seconds</li>
  549. </ul>
  550. </ul>
  551. <br>
  552. <a name="Verkehrsinfoset"></a>
  553. <b>Set</b><br>
  554. <ul>
  555. <code>set &lt;name&gt; &lt;option&gt;</code>
  556. <br><br>
  557. Options:
  558. <ul>
  559. <li><i>update</i><br>
  560. update will be executed right away</li>
  561. </ul>
  562. </ul>
  563. <br>
  564. <a name="Verkehrsinfoget"></a>
  565. <b>Get</b><br>
  566. <ul>
  567. <code>get &lt;name&gt; &lt;option&gt;</code>
  568. <br><br>
  569. Options:
  570. <ul>
  571. <li><i>info</i><br>
  572. output currently traffic information</li>
  573. </ul>
  574. </ul>
  575. <br>
  576. <a name="Verkehrsinfoattr"></a>
  577. <b>Attributes</b><br>
  578. <ul>
  579. <code>attr &lt;name&gt; &lt;option&gt; &lt;value&gt;</code>
  580. <br><br>
  581. Options:
  582. <ul>
  583. <li><i>filter_exclude</i><br>
  584. This is an exclusion filter. Traffic information containing these words, will not be displayed.<br>
  585. The filter supports regular expressions. Attention: regex control character, for example brackets have to be masked with a backslash "\".<br>
  586. Multiple searching keywords can be seperated with the pipe "|".<br><br></li>
  587. <li><i>filter_include</i><br>
  588. This is an inclusion filter. Traffic information containing these words, will be displayed.<br>
  589. The filter supports regular expressions. Attention: regex control character, for example brackets have to be masked with a backslash "\".<br>
  590. Multiple searching keywords can be seperated with the pipe "|".<br><br></li>
  591. <li>Hint: Both filters can be used at the same time, or optional just one.<br>
  592. The filters are linked with a logical and. That means, for example, when something is excluded, it can be reincluded with the other filter.<br><br></li>
  593. <li><i>orderby</i><br>
  594. Messages will be sorted by relevance by reference to the string.<br>
  595. The sort supports regular expressions.<br>
  596. Multiple searching keywords can be seperated with the pipe "|".<br><br></li>
  597. <li><i>msg_format [ road | head | both ]</i> (only Verkehrsinfo.de and RadioSAW.de)<br>
  598. Using this parameter you can format the output, regarding streets, direction or both.<br><br></li>
  599. <li><i><a href="#readingFnAttributes">readingFnAttributes</a></i><br><br></li>
  600. </ul>
  601. </ul>
  602. <br>
  603. <a name="Verkehrsinforeading"></a>
  604. <b>Readings</b>
  605. <ul>
  606. <br>
  607. <li><b>e_</b><i>0|1|2|3...|9</i><b>_...</b> - aktiv message</li>
  608. <li><b>count</b> - number of aktiv messages</li>
  609. <li><b>e_</b><i>0</i><b>_road</b> - street</li>
  610. <li><b>e_</b><i>0</i><b>_head</b> - direction</li>
  611. <li><b>e_</b><i>0</i><b>_msg</b> - message</li>
  612. </ul>
  613. <br>
  614. <a name="Verkehrsinfofunktion"></a>
  615. <b>Funktion</b>
  616. <ul>
  617. <code>Verkehrsinfo_GetData(&lt;devicename&gt;)</code>
  618. <br><br>
  619. The function can be accessed anywhere in FHEM.
  620. The output of this function is the same as get <name> info and the string can be used for further forwarding.
  621. <br><br>
  622. example: <code>my $result = Verkehrsinfo_GetData('A8')</code>
  623. </ul>
  624. <br>
  625. </ul>
  626. =end html
  627. =begin html_DE
  628. <a name="Verkehrsinfo"></a>
  629. <h3>Verkehrsinfo</h3>
  630. <ul>
  631. <i>Verkehrsinfo</i> kann die aktuellen Verkehrsinformationen von verschiedenen Quellen auslesen.
  632. <br><br>
  633. <ul>
  634. <li>Verkehrsinfo.de</li>
  635. Um die gewünschten Verkehrsinformation zu erhalten wird die Webseite https://www.verkehrsinfo.de/httpsmobil besucht.
  636. Hier können Sie dann entweder Straßen oder Bundesländer auswählen. Anschließend wird die URL als Parameter übergeben.
  637. <br><br>
  638. <li>Hessenschau.de</li>
  639. Hier ist keine Konfiguration notwendig, man verwendet die URL http://hessenschau.de/verkehr/index.html als Parameter.
  640. <br><br>
  641. <li>RadioSAW.de</li>
  642. Hier ist keine Konfiguration notwendig, man verwendet als Parameter radiosaw.
  643. </ul>
  644. <br><br>
  645. <b>Voraussetzung:</b>
  646. <ul><br>
  647. Für dieses Modul werden folgende Perlmodule benötigt:<br>
  648. <li>HTML::TreeBuilder::XPath<br>
  649. <code>sudo apt-get install libxml-treebuilder-perl libhtml-treebuilder-xpath-perl</code>
  650. </li>
  651. <li>JSON<br>
  652. <code>sudo apt-get install libjson-perl</code>
  653. </li>
  654. </ul>
  655. <br><br>
  656. <a name="Verkehrsinfodefine"></a>
  657. <b>Define</b>
  658. <ul>
  659. <code>define &lt;name&gt; Verkehrsinfo &lt;url&gt; &lt;interval&gt;</code>
  660. <br><br>
  661. Beispiel: <code>define A8 Verkehrsinfo https://www.verkehrsinfo.de/httpsmobil/index.php?c=staulist&street=A8&lat=&lon= 3600 </code>
  662. <br><br>
  663. Options:
  664. <ul>
  665. <li><i>url</i><br>
  666. URL der auszulesenden Verkehrsinformationen</li>
  667. <li><i>interval</i><br>
  668. Alle wieviel Sekunden die Daten aktualisiert werden</li>
  669. </ul>
  670. </ul>
  671. <br>
  672. <a name="Verkehrsinfoset"></a>
  673. <b>Set</b><br>
  674. <ul>
  675. <code>set &lt;name&gt; &lt;option&gt;</code>
  676. <br><br>
  677. Options:
  678. <ul>
  679. <li><i>update</i><br>
  680. Update wird sofort ausgeführt</li>
  681. </ul>
  682. </ul>
  683. <br>
  684. <a name="Verkehrsinfoget"></a>
  685. <b>Get</b><br>
  686. <ul>
  687. <code>get &lt;name&gt; &lt;option&gt;</code>
  688. <br><br>
  689. Options:
  690. <ul>
  691. <li><i>info</i><br>
  692. Ausgeben der aktuellen Verkehrsinformationen</li>
  693. </ul>
  694. </ul>
  695. <br>
  696. <a name="Verkehrsinfoattr"></a>
  697. <b>Attributes</b><br>
  698. <ul>
  699. <code>attr &lt;name&gt; &lt;option&gt; &lt;value&gt;</code>
  700. <br><br>
  701. Options:
  702. <ul>
  703. <li><i>filter_exclude</i><br>
  704. Dies ist ein Ausschlussfilter. Verkehrsmeldung die eines der Wörter enthalten, werden nicht angezeigt.<br>
  705. Der Filter unterstütz Regulärer Ausdrücke. Achtung: Regex Steuerzeichen, z.B. Klammern müssen mit einem Backslash "\" maskiert werden.<br>
  706. Mehrer Suchbegriffe können mit einer Pipe "|" getrennt werden.<br><br></li>
  707. <li><i>filter_include</i><br>
  708. Dies ist ein Einschlussfilter. Es werden nur Verkehrsmeldung angezeigt die eines der Wörter enthalten.<br>
  709. Der Filter unterstütz Regulärer Ausdrücke. Achtung: Regex Steuerzeichen, z.B. Klammern müssen mit einem Backslash "\" maskiert werden.<br>
  710. Mehrer Suchbegriffe können mit einer Pipe "|" getrennt werden.<br><br></li>
  711. <li>Hinweis: Beide Filter können gleichzeitig benutzt werden, aber es kann auch wahlweise nur einer verwendet werden.<br>
  712. Die Filter sind mit einem Logischen UND verknüpft. Das heist z.B.: wenn etwas ausgeschlossen wurde, kann es nicht mit dem Einschlussfilter wiedergeholt werden.<br><br></li>
  713. <li><i>orderby</i><br>
  714. Anhand von Zeichefolgen wird eine Sortierung der Meldungen nach Relevanz vorgenommen.<br>
  715. Die Sortierung unterstützt Regulärer Ausdrücke.<br>
  716. Mehrer Suchbegriffe können mit einer Pipe "|" getrennt werden.<br><br></li>
  717. <li><i>msg_format [ road | head | both ]</i> (Nur Verkehrsinfo.de und RadioSAW.de)<br>
  718. Über diesen Parameter kann die Meldung formatiert werden nach Strasse, Richtung oder beides<br><br></li>
  719. <li><i><a href="#readingFnAttributes">readingFnAttributes</a></i><br><br></li>
  720. </ul>
  721. </ul>
  722. <br>
  723. <a name="Verkehrsinforeading"></a>
  724. <b>Readings</b>
  725. <ul>
  726. <br>
  727. <li><b>e_</b><i>0|1|2|3...|9</i><b>_...</b> - aktive Meldungen</li>
  728. <li><b>count</b> - Anzahl der aktiven Meldungen</li>
  729. <li><b>e_</b><i>0</i><b>_road</b> - Straße</li>
  730. <li><b>e_</b><i>0</i><b>_head</b> - Fahrtrichtung</li>
  731. <li><b>e_</b><i>0</i><b>_msg</b> - Meldung</li>
  732. </ul>
  733. <br>
  734. <a name="Verkehrsinfofunktion"></a>
  735. <b>Funktion</b>
  736. <ul>
  737. <code>Verkehrsinfo_GetData(&lt;devicename&gt;)</code>
  738. <br><br>
  739. Die Funktion kann überall in FHEM aufgerufen werden und liefert als Rückgabewert das gleiche Ergebnis wie der get &lt;name&gt; info Aufruf.
  740. Der Rückgabewert als Text, kann dann für weiteres verwendet werden.
  741. <br><br>
  742. Beispiel: <code>my $result = Verkehrsinfo_GetData('A8')</code>
  743. </ul>
  744. <br>
  745. </ul>
  746. =end html_DE
  747. =cut