Spritpreismonitor
Die hier vorgestellten Geräte-Definitionen ermöglichen es, mit Hilfe von HTTPMOD bei clever-tanken.de die Kraftstoffpreise von gewünschten Tankstellen zu protokollieren. Die Credits dafür gehen an Forum-Benutzer Phil__ . Die Original Diskussion im Forum ist hier.
Alternativ dazu können die Kraftstoffpreise auch über die Webseite tankerkönig.de bezogen werden. Die Vorgehensweise dafür ist im Detail in diesem Blog beschrieben.
Einleitung
Im Folgenden wird mit Hilfe des Moduls HTTPMOD der Kraftstoffpreis von ausgewählten Tankstellen abgerufen. Die Daten werden gefiltert, gespeichert und als Diagramm aufbereitet.
Zum Nachbau ist es erforderlich, die persönlich gewünschten Tankstellen bei Clever-Tanken zu suchen. Jede Tankstelle hat eine eigene Nummer:
/tankstelle_details/0815
Die 600 in den Beispielen gibt das Abrufintervall in Sekunden an.
Konfiguration
Die erforderlichen Definitionen in der FHEM Konfiguration bestehen aus mehreren Teilen. Zunächst die erforderlichen Einstellungen um die Daten der gewünschten Tankstellen abzuholen.
Beispielkonfiguration für eine Shell Tankstelle inkl. Sondersorten:
define Tankstelle_Shell HTTPMOD http://www.clever-tanken.de/tankstelle_details/4871 600 attr Tankstelle_Shell alignTime 00:00 attr Tankstelle_Shell enableControlSet 1 attr Tankstelle_Shell reading01Name Diesel attr Tankstelle_Shell reading01Regex "price-type-name">Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell reading02Name SuperE5 attr Tankstelle_Shell reading02Regex "price-type-name">Super.E5<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell reading03Name SuperE10 attr Tankstelle_Shell reading03Regex "price-type-name">Super.E10<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell reading04Name SuperPlus attr Tankstelle_Shell reading04Regex "price-type-name">SuperPlus<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell reading05Name ShellVPowerRacing attr Tankstelle_Shell reading05Regex "price-type-name">Shell.V-Power.Racing<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell reading06Name ShellVPowerDiesel attr Tankstelle_Shell reading06Regex "price-type-name">Shell.V-Power.Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Shell stateFormat E5: SuperE5, E10: SuperE10, D: Diesel, SP: SuperPlus attr Tankstelle_Shell timeout 10
Beispielkonfiguration für eine Total Tankstelle inkl. Sondersorten:
define Tankstelle_Total HTTPMOD http://www.clever-tanken.de/tankstelle_details/4870 600 attr Tankstelle_Total alignTime 00:00 attr Tankstelle_Total enableControlSet 1 attr Tankstelle_Total reading01Name Diesel attr Tankstelle_Total reading01Regex "price-type-name">Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading02Name SuperE5 attr Tankstelle_Total reading02Regex "price-type-name">Super.E5<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading03Name SuperE10 attr Tankstelle_Total reading03Regex "price-type-name">Super.E10<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading04Name SuperPlus attr Tankstelle_Total reading04Regex "price-type-name">SuperPlus<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading05Name LKWDiesel attr Tankstelle_Total reading05Regex "price-type-name">LKW-Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading06Name TESuperPlus attr Tankstelle_Total reading06Regex "price-type-name">Total.Excellium.Super.Plus<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading07Name TEDiesel attr Tankstelle_Total reading07Regex "price-type-name">Total.Excellium.Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading08Name Autogas attr Tankstelle_Total reading08Regex "price-type-name">Autogas<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading09Name Erdgas attr Tankstelle_Total reading09Regex "price-type-name">Erdgas<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total reading10Name Wasserstoff attr Tankstelle_Total reading10Regex "price-type-name">Wasserstoff<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Total stateFormat E5: SuperE5, E10: SuperE10, D: Diesel, SP: SuperPlus attr Tankstelle_Total timeout 10
Beispielkonfiguration für eine Aral Tankstelle inkl. Sondersorten:
define Tankstelle_Aral HTTPMOD http://www.clever-tanken.de/tankstelle_details/4869 600 attr Tankstelle_Aral alignTime 00:00 attr Tankstelle_Aral enableControlSet 1 attr Tankstelle_Aral reading01Name Diesel attr Tankstelle_Aral reading01Regex "price-type-name">Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading02Name SuperE5 attr Tankstelle_Aral reading02Regex "price-type-name">Super.E5<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading03Name SuperE10 attr Tankstelle_Aral reading03Regex "price-type-name">Super.E10<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading04Name AralSuperPlus attr Tankstelle_Aral reading04Regex "price-type-name">ARAL.Superplus<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading05Name Autogas attr Tankstelle_Aral reading05Regex "price-type-name">Autogas<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading06Name AralUltimate102 attr Tankstelle_Aral reading06Regex "price-type-name">ARAL.Ultimate.102<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral reading07Name ARALUltimateDiesel attr Tankstelle_Aral reading07Regex "price-type-name">ARAL.Ultimate.Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Aral stateFormat E5: SuperE5, E10: SuperE10, D: Diesel, SP: AralSuperPlus attr Tankstelle_Aral timeout 10
Beispielkonfiguration für eine Esso Tankstelle inkl. Sondersorten:
define Tankstelle_Esso HTTPMOD http://www.clever-tanken.de/tankstelle_details/17413 600 attr Tankstelle_Esso alignTime 00:00 attr Tankstelle_Esso enableControlSet 1 attr Tankstelle_Esso reading01Name Diesel attr Tankstelle_Esso reading01Regex "price-type-name">Diesel<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Esso reading02Name SuperE5 attr Tankstelle_Esso reading02Regex "price-type-name">Super.E5<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Esso reading03Name SuperE10 attr Tankstelle_Esso reading03Regex "price-type-name">Super.E10<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Esso reading04Name EssoSuperPlus attr Tankstelle_Esso reading04Regex "price-type-name">Esso.Superplus<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Esso reading05Name Autogas attr Tankstelle_Esso reading05Regex "price-type-name">Autogas<[\d\D]{600,900}"current-price-\d">(\d.\d\d) attr Tankstelle_Esso stateFormat E5: SuperE5, E10: SuperE10, D: Diesel, SP: EssoSuperPlus attr Tankstelle_Esso timeout 10
Wer die Anzeige der Kraftstoffpreise inklusive der obligatorischen abschließenden "9" möchte, muss zu jedem readingXXName eine entsprechende Zusatzzeile hinzufügen. Diese addiert zum Wert des Readings (hier Diesel) den Wert 0.009:
attr Tankstelle_Shell reading01OExpr $val+0.009
Diese Zeile fügt dem Dieselpreis der Shell Tankstelle bei einem Literpreis von 1.29 die (normalerweise hochgestellte) Ziffer "9" hinzu und ergibt somit 1.299.
Dieses starre Konstrukt kann man allerdings auch dynamisch erzeugen.
Dazu wird beim RegEx eine zweite Capture Gruppe definiert. Dadurch entstehen immer zwei Readings mit dem jeweiligen Kraftstoffnamen und den Suffixen "-1" und "-2". Der neu dazu gekommene (2.) Wert wird dann noch mit dem Attribut reading0X-2OExpr durch 1000 dividiert. In entsprechenden userReadings (mit dem ursprünglichen Kraftstoffnamen - also ohne die Suffixe) werden dann beide Werte summiert.
Dazu hier mal eine Beispielkonfiguration für eine Agip Tankstelle, bei der auch die Straße, die Postleitzahl und die Stadt mit ausgelesen wird:
define Tankstelle_Agip HTTPMOD http://www.clever-tanken.de/tankstelle_details/34459 600 attr Tankstelle_Agip alignTime 00:00 attr Tankstelle_Agip enableControlSet 1 attr Tankstelle_Agip event-on-change-reading SuperE5.*,SuperE10.*,Diesel.*,SuperPlus.* attr Tankstelle_Agip reading01-2OExpr $val/1000 attr Tankstelle_Agip reading01Name SuperE5 attr Tankstelle_Agip reading01Regex "price-type-name">Super.E5<[\d\D]{200,900}"current-price-\d">(\d.\d\d)<[\d\D]{40,60}"suffix-price-\d">(\d) attr Tankstelle_Agip reading02-2OExpr $val/1000 attr Tankstelle_Agip reading02Name SuperE10 attr Tankstelle_Agip reading02Regex "price-type-name">Super.E10<[\d\D]{700,900}"current-price-\d">(\d.\d\d)<[\d\D]{40,60}"suffix-price-\d">(\d) attr Tankstelle_Agip reading03-2OExpr $val/1000 attr Tankstelle_Agip reading03Name Diesel attr Tankstelle_Agip reading03Regex "price-type-name">Diesel<[\d\D]{700,900}"current-price-\d">(\d.\d\d)<[\d\D]{40,60}"suffix-price-\d">(\d) attr Tankstelle_Agip reading04-2OExpr $val/1000 attr Tankstelle_Agip reading04Name SuperPlus attr Tankstelle_Agip reading04Regex "price-type-name">SuperPlus<[\d\D]{200,900}"current-price-\d">(\d.\d\d)<[\d\D]{40,60}"suffix-price-\d">(\d) attr Tankstelle_Agip reading05Name street attr Tankstelle_Agip reading05Regex "streetAddress">(.*)< attr Tankstelle_Agip reading06Name zip attr Tankstelle_Agip reading06Regex "http://schema.org/postalCode">(.*)< attr Tankstelle_Agip reading07Name city attr Tankstelle_Agip reading07Regex "http://schema.org/addressCountry">(.*)< attr Tankstelle_Agip stateFormat E5: SuperE5, E10: SuperE10, SP: SuperPlus, D: Diesel attr Tankstelle_Agip timeout 10 attr Tankstelle_Agip userReadings SuperE5:SuperE5-.* {ReadingsNum($name,"SuperE5-1",0)+ReadingsNum($name,"SuperE5-2",0)},\ SuperE10:SuperE10-.* {ReadingsNum($name,"SuperE10-1",0)+ReadingsNum($name,"SuperE10-2",0)},\ SuperPlus:SuperPlus-.* {ReadingsNum($name,"SuperPlus-1",0)+ReadingsNum($name,"SuperPlus-2",0)},\ Diesel:Diesel-.* {ReadingsNum($name,"Diesel-1",0)+ReadingsNum($name,"Diesel-2",0)}
Eine passende readingsGroup mit farblicher Hervorhebung der Kraftstoffpreise:
define Kraftstoffpreise readingsGroup <Tankstelle>,<Diesel>,<Super E5>,<Super E10>,<Super Plus> Tankstelle_.*:Diesel,SuperE5,SuperE10,SuperPlus attr Kraftstoffpreise cellStyle {"r:1" => 'style="font-weight:bold"'} attr Kraftstoffpreise valueFormat {'%.2f €'} attr Kraftstoffpreise valueStyle {KraftstoffWerte($READING,$VALUE)}
Eine etwas andere readingsGroup mit farblicher Hervorhebung des Tiefstpreises der jeweiligen Kraftstoffsorte und hochgestellter letzter Ziffer beim Kraftstoffpreis:
define rg_Kraftstoffpreise readingsGroup <Tankstelle>,<Diesel>,<Super E5>,<Super E10>,<Super Plus>,<PLZ>,<Ort>,<Straße> Tankstelle_.*:Diesel,SuperE5,SuperE10,SuperPlus,zip,city,street attr rg_Kraftstoffpreise alias Kraftstoffpreisübersicht attr rg_Kraftstoffpreise cellStyle {"r:1" => 'style="font-weight:bold"'} attr rg_Kraftstoffpreise valueFormat {"$1<sup>$2</sup> €" if ($READING=~/^Super|Diesel/ && $VALUE=~/^(\d\.\d\d)(\d)$/)} attr rg_Kraftstoffpreise valueStyle {'style="color:green;;font-weight:bold"' if ($READING=~/^Super|Diesel/ && $VALUE eq KraftstoffLP($READING))}
Hier noch eine mögliche Definition für das FileLog. Es werden von allen definierten Tankstellen (Namen beginnend mit "Tankstelle_") die Werte für Super und Diesel gespeichert:
define FileLog_Kraftstoffpreise FileLog ./log/Kraftstoffpreise-%Y-%m.log Tankstelle_.*:(Super|Diesel).*
Und eine Plotdefinition
define SVG_FileLog_Kraftstoffpreise_1 SVG FileLog_Kraftstoffpreise:SVG_FileLog_Kraftstoffpreise_1:CURRENT
Farbliche Hervorhebung
Folgender Code kommt in 99_myUtils.pm; die Werte können natürlich entsprechend eigener Wünsche und Vorlieben angepasst werden.
######################################################## ### Kraftstoffpreisübersicht - Farbsortierung ### ######################################################## sub KraftstoffWerte($$) { my ($name,$wert) = @_; # Log(3,"$name $wert"); if ($name eq "Diesel") { return 'style="color:red"' if($wert >= 1.39); return 'style="color:blue"' if(($wert >= 1.33) && ($wert < 1.39)); return 'style="color:green;;font-weight:bold"' if($wert <= 1.32); } elsif ($name eq "SuperE10") { return 'style="color:crimson"' if($wert >= 1.70); return 'style="color:yellow"' if(($wert >= 1.55) && ($wert < 1.70)); return 'style="color:lightgreen;;font-weight:bold"' if($wert < 1.55); } elsif ($name eq "SuperE5") { return 'style="color:red"' if($wert >= 1.59); return 'style="color:blue"' if(($wert >= 1.49) && ($wert < 1.59)); return 'style="color:green;;font-weight:bold"' if($wert <= 1.48); } } ########################################################################################### ### Funktion für Ermittlung vom niedrigsten Preis für readingsGroup rg_Kraftstoffpreise ### ########################################################################################### # damit die Funktion richtig funktioniert müssen alle Tankstellennamen mit "Tankstelle_" # beginnen oder entsprechend devspec2array auf die eigenen Namen anpassen sub KraftstoffLP($) { my ($r) = @_; return (sort map {ReadingsNum($_,$r,999)} devspec2array("Tankstelle_.*"))[0]; }
Die zugehörige .gplot Datei
SVG_FileLog_Kraftstoffpreise_1.gplot
# Created by FHEM/98_SVG.pm, 2014-10-22 12:57:34 set terminal png transparent size <SIZE> crop set output '<OUT>.png' set xdata time set timefmt "%Y-%m-%d_%H:%M:%S" set xlabel " " set title 'Kraftstoffpreise' set ytics set y2tics set grid set ylabel "Preise (€/Liter)" set y2label "Preise (€/Liter)" #FileLog 4:Aral.Diesel\x3a:1: #FileLog 4:Shell.Diesel\x3a:1: #FileLog 4:Total.Diesel\x3a:1: #FileLog 4:Aral.SuperE5\x3a:1: #FileLog 4:Shell.SuperE5\x3a:1: #FileLog 4:Total.SuperE5\x3a:1: plot "<IN>" using 1:2 axes x1y2 title 'Aral Diesel' ls l2 lw 1 with lines,\ "<IN>" using 1:2 axes x1y2 title 'Shell Diesel' ls l1 lw 1 with lines,\ "<IN>" using 1:2 axes x1y2 title 'Total Diesel' ls l0 lw 1 with lines,\ "<IN>" using 1:2 axes x1y2 title 'Aral Super' ls l3 lw 1 with lines,\ "<IN>" using 1:2 axes x1y2 title 'Shell Super' ls l4 lw 1 with lines,\ "<IN>" using 1:2 axes x1y2 title 'Total Super' ls l5 lw 1 with lines
Tricks für HTTPMOD
Je nach Internetverbindung und Geschwindigkeit des FHEM Rechners kann das Modul manchmal Probleme haben, die Daten schnell genug zu verarbeiten.
Eine Lösung dazu wäre, die Daten vorher mit wget auf einem eigenen Webserver zu speichern (z.B. ein lokaler Apache) und HTTPMOD auf diesen zugreifen zu lassen.
Links
- Vorstellung und Diskussion im FHEM Forum
- Der "Datenlieferant": Clever Tanken
- Eine Anleitung, wie man das mit wget macht, gibt es hier (ist aber noch nicht ganz fertig)
- Umsetzung für Tankerkönig API