Gaszähler auslesen

Nachdem meine Versuche den M-BUS Wärmemengen- und Wasserzähler auszulesen vorerst beendet sind und ich auf den bestellten M-BUS-Pegelwandler warte, habe ich mich dem Auslesen des Gaszählers gewidmet. Das sollte erfolgsversprechender sein…

Als ich dann endlich den vor langer Zeit bestellten Reed-Kontakt mk 471 b (z.B. bei Reichelt) wieder gefunden habe, konnte es losgehen.

Der Gaszähler hat an der letzten Stelle des Zählwerkes bei der “0” einen Magneten wodurch der Reedkontakt bei jeder vollen Umdrehung einmal geschlossen wird. Diese Impulse gilt es mit einem Raspberry über einen GPIO zu zählen.

IMG_7735

Den Reedkontakt habe ich natürlich wieder professionell mit Klebeband fixiert…hält!

IMG_7734

Angeschlossen ist der Reedkontakt über eine Zwillingslitze direkt an einen GPIO und Ground des Raspberry PI. Pull-Up braucht es keinen. Dieser wird per Software aktiviert.

Den GPIO lese ich mit einem Python-Script aus welches in einer Endlosschleife ausgeführt wird und den GPIO (im Beispiel Pin “21”) jede Sekunde abfragt. Sekundenweise sollte reichen, da ich nicht hoffe das unser Gaszähler so schnell läuft. Immer wenn der Reed geschlossen wird, wird eine “1” und der Zeitstempel in die Spalten “timestamp” und “zaehlerstand” einer MySQL-Tabelle geschrieben. Das passiert immer nur beim Wechsel des Reed von “Offen” nach “Geschlossen”. Sollte der Zähler genau mit dem Magnet am Reed stehen bleiben, passiert nichts.

[cc lang=”python”]

import RPi.GPIO as GPIO
import time
import MySQLdb

GPIO.setmode(GPIO.BOARD)
GPIO.setwarnings(False)

# GPIO definieren
REED_gas = 21
# definierten GPIO als Eingang setzen
GPIO.setup(REED_gas, GPIO.IN, pull_up_down=GPIO.PUD_UP)

status_alt=1
while True:
status_aktuell = GPIO.input(REED_gas)
# REEDKONTAKT geoeffnet
if status_aktuell == 1:
#print “Kontakt offen”
status_alt=GPIO.input(REED_gas)
# REEDKONTAKT geschlossen
elif status_aktuell==0:
#print “Kontakt geschlossen”
if status_alt!=status_aktuell:
status_alt=GPIO.input(REED_gas)
# Datenbankverbindung
db = MySQLdb.connect(host=”“, user=”“, passwd=”“, db=”“)
# Impuls in Datenbank eintragen
cursor = db.cursor()
cursor.execute(“””INSERT INTO gaszaehler (timestamp,zaehlerstand) VALUES (CURRENT_TIMESTAMP,1)”””)
db.commit()
cursor.close()
time.sleep(1)
[/cc]

IMG_7737

Da das Python-Script sich ab und an mal verabschiedet, starte ich es per Cronjob wieder neu falls der Prozess nicht mehr vorhanden ist. Ich sollte vielleicht lieber das Problem im Python-Script suchen und das ein oder andere Try&Catch einbauen, aber es geht auch erstmal so…

#!/bin/sh
if ps -ef | grep -v grep | grep gaszaehler.py ; then
    exit 0
else
    sudo python /usr/local/bin/gaszaehler.py &
    logger "gaszaehler.py neu gestartet"
    exit 0
fi

Zur Auswertung benutze ich dann folgende SQL-Querys:

SELECT sum(zaehlerstand) FROM gaszaehler

Hiermit werden alle gespeicherten “Ticks” summiert. Dazu muss noch ein Startwert addiert werden da der Gaszähler ja schon eine zeitlang lief.
Dieser Startwert muss ggf. ab und an mal korrigiert werden falls der Raspi mal einen Umlauf des Zählers nicht mitbekommt weil z.B. der unwahrscheinliche Fall eintritt, dass das Klebeband nicht gehalten hat oder wahrscheinlicher, das Python-Script bzw. der ganze Raspi nicht mehr läuft. Den Startwert gebe ich als eine Zahl inkl. der drei Nachkommastellen an und addiere das Ergebnis der SQL-Query dazu. Um die richtige “Einheit” kümmere ich mich in der Ausgabe auf der PHP-Seite mit folgender PHP-Zeile:

[cc lang=”php”]
$sql = “SELECT sum(zaehlerstand) FROM gaszaehler”;
$query = mysql_query($sql) or die(“Anfrage nicht erfolgreich”);
while ($wert = mysql_fetch_array($query)) {
$gesamtwert=($wert[0]*10)+$gas_startwert;
}
$gesamtwert=substr($gesamtwert,0,-3).”.”.substr($gesamtwert,-3,2);
[/cc]

Der Faktor 10 muss hier beachtet werden, da ja nur alle 0,01m³ Impulse gezählt werden und ich den Startwert mit allen drei Nachkommastellen angegeben habe. Die zweite Zeile setzt das Komma in der Ausgabe an die richtige Stelle.

Eine tägliche Auswertung funktioniert z.B. mit folgender SQL-Query:

SELECT CASE DATE_FORMAT(timestamp,'%w')
          WHEN 0 THEN 'Sonntag'
          WHEN 1 THEN 'Montag'
          WHEN 2 THEN 'Dienstag'
          WHEN 3 THEN 'Mittwoch'
          WHEN 4 THEN 'Donnerstag'
          WHEN 5 THEN 'Freitag'
          WHEN 6 THEN 'Samstag'
          ELSE 'fehler' END,
          sum(zaehlerstand)
FROM gaszaehler
WHERE DATE(timestamp)>=DATE_SUB(NOW(),INTERVAL 7 DAY)
GROUP BY day(timestamp)
ORDER BY timestamp

Auch hier bekommt man den korrekten Wert in m³ durch einfache Kommaverschieberei in der Ausgabe. Den Faktor 10 benötigt man bei der täglichen Auswertung nicht:

[cc lang=”php”]substr(($wert[1]),0,-2).”.”.substr(($wert[1]),-2)[/cc]

In der Webseite sieht das folgendermaßen aus:

gaszaehler_screenshot

Das alles läuft erst drei Tage und es gibt bestimmt noch Verbesserungspotential aber ein Anfang ist gemacht.

Gruß
Chris

19 Gedanken zu „Gaszähler auslesen

  • 25. November 2014 um 21:30 Uhr
    Permalink

    Hi,
    super beschrieben, genau das Projekt was ich auch “versuche” umzusetzen. Gibt es da auch die Möglichkeit die ausgelesenen daten in eine RRD Datenbank abzulegen. zur Zeit habe ich Temperatur Daten meiner Heizungsanlage über OneWire in den RRDs, würde gerne auch Gaszaehleraten dort unterbringen. Wie ich jedoch statt mysql RRD einsetze weiss ich noch nicht ??
    Gruß ….. Gerd

    Antwort
    • 28. November 2014 um 18:09 Uhr
      Permalink

      Hallo Gerd,

      ich kenne mich mit rrd leider nicht so gut aus, denke aber das Du dort einen absoluten Wert benötigst (z.B. Tagesverbrauch) anstatt der 0,01m³-Impulse.
      Ein erster Ansatz wäre evtl. die einzelnen Impulse des Python-Scripts mit Zeitstempel in einer Textdatei zeilenweise zu speichern. Diese Daten aus der Textdatei könntest Du dann mit einem zweiten Script einmal täglich anhand des Zeitstempels aufsummieren und das Ergebnis dann in die rrd schreiben (und evtl. die Textdatei wieder löschen um immer nur einen Tag darin zu speichern).

      Vielleicht hilft Dir das ja schon mal weiter!

      Gruß
      Chris

      Antwort
  • 12. Dezember 2014 um 21:58 Uhr
    Permalink

    Hallo Chris,
    habe nun endlich meine SQL Datenbank eingerichtet, jedoch tun sich diverse Fehler oder Fragen auf. Vielleicht kannst Du helfen.
    1. habe das py script angepasst
    import RPi.GPIO as GPIO
    import time
    #import MySQLdb
    import gaszaehler

    GPIO.setmode(GPIO.BOARD)
    GPIO.setwarnings(False)

    # GPIO definieren
    REED_gas = 10
    # definierten GPIO als Eingang setzen
    GPIO.setup(REED_gas, GPIO.IN, pull_up_down=GPIO.PUD_UP)

    # Datenbankverbindung
    db = gaszaehler(host=”localhost”, user=”pi”, passwd=”XXXXXXX”, db=”gaszaehler”)

    status_alt=1
    while True:
    status_aktuell = GPIO.input(REED_gas)
    # REEDKONTAKT geoeffnet
    if status_aktuell == 1:
    #print “Kontakt offen”
    status_alt=GPIO.input(REED_gas)
    # REEDKONTAKT geschlossen
    elif status_aktuell==0:
    #print “Kontakt geschlossen”
    if status_alt!=status_aktuell:
    status_alt=GPIO.input(REED_gas)
    # Impuls in Datenbank eintragen
    cursor = db.cursor()
    cursor.execute(“””INSERT INTO gaszaehler (timestamp,zaehlerstand) VALUES (CURRENT_TIMESTAMP,1)”””)
    db.commit()
    cursor.close()
    time.sleep(1)
    ————
    pi@vdr ~/temperatur/gaszaehler $ sudo python gaszaehler.py
    Traceback (most recent call last):
    File “gaszaehler.py”, line 4, in
    import gaszaehler
    File “/home/pi/temperatur/gaszaehler/gaszaehler.py”, line 15, in
    db = gaszaehler(host=”localhost”, user=”pi”, passwd=”XXXXXXX”, db=”gaszaehler”)
    TypeError: ‘module’ object is not callable
    ————
    1. nach dem start sollte sich eigentlich die db füllen? und dann?
    2. Was bedeutet die Fehlermeldung , hab ich da was in der DB falsch eingerichtet?

    Antwort
  • 14. Januar 2015 um 10:44 Uhr
    Permalink

    Hallo,
    vielen Dank für die Ideen.
    Bekommst du auch immer 2 Impulse pro Umdrehung? Ich bekomme eine Impuls, wenn ich die 1 und die 9 im Sichtfenster sehe.
    Bei mir registriert eine Interrupt Service Routine die Impulse und schreibt dann in die DB.
    Hast du eine Idee wie man eine tägliche Auswertung in SQLite machen kann?
    Danke im Voraus.

    Antwort
    • 17. Januar 2015 um 11:45 Uhr
      Permalink

      Hi Felix,

      nein, bei mir wir immer nur ein Impuls pro Umdrehung gezählt. Vielleicht zählst Du das Schliessen und Öffnen des Kontaktes?
      Du musst auch dafür sorgen, das nur ein Impuls gezählt wird solange der Magnet genau am Kontakt stehen bleibt.

      Kenne leider SQLite nicht im Detail aber das sollte doch auch normales ANSI SQL verstehen, oder?
      Hab mir das mal kurz angeschaut und gesehen, dass auch “CURRENT_TIMESTAMP” unterstützt wird.
      Was genau funktioniert denn nicht?

      Gruß
      Chris

      Antwort
  • 22. Januar 2015 um 10:28 Uhr
    Permalink

    Hallo,
    “zählst du da Schliessen *und* Öffnen”, mmmhh. Ich habe den Pin genauso wie du gesetzt.
    Das mit der täglichen Auswertung in SQLite sieht bei mir so aus:
    SELECT tstamp,
    CASE CAST (strftime(‘%w’, tstamp) as integer)
    WHEN 1 THEN ‘Monday’
    WHEN 2 THEN ‘Tuesday’
    WHEN 3 THEN ‘Wednesday’
    WHEN 4 THEN ‘Thursday’
    WHEN 5 THEN ‘Friday’
    WHEN 6 THEN ‘Saturday’
    WHEN 0 THEN ‘Sunday’
    ELSE ‘fehler’ END,
    SUM(tick)
    FROM gascounter WHERE tstamp BETWEEN DATE(‘now’, ‘-7 days’) AND DATE(‘now’)
    GROUP BY strftime(‘%w’, tstamp)
    ORDER BY tstamp

    Antwort
    • 23. Januar 2015 um 16:12 Uhr
      Permalink

      Hi,
      das Script schreibt immer nur einen Impuls in die Datenbank wenn ein “geschlossen” auf ein vorheriges “offen” folgt.
      Bleibt es “geschlossen” während die Schleife erneut durchlaufen wird (z.b. weil der Magnet genau in der Position stehen bleibt) oder ist es “offen”, wird nicht gezählt.

      Gruß
      Chris

      Antwort
      • 23. Januar 2015 um 19:57 Uhr
        Permalink

        Hallo Chris,
        ja, schon verstanden. Ich habe deinen Code auch bei mir laufen lassen. Tat aber nicht….
        Ich schau weiter.
        Grüße

        Antwort
  • Pingback: Gaszähler auslesen - Darstellung der Daten - bubuxblog

  • 13. Oktober 2017 um 10:29 Uhr
    Permalink

    Hallo Chris,
    auch wenn das Thema und die letzten Kommentare ein paar Tage alt sind, wollte ich mich trotzdem für die Zusammenstellung bedanken. Habe das Ganze bei mir zuhause nachgebaut.

    Dabei ist mir in den Restart-Logs für das Python-Script aufgefallen, dass die Abstürze immer auftreten, nachdem das Script eine längere Zeit nichts gemacht hat. Ich vermute die Ursache liegt im Verbindungsaufbau außerhalb der WHILE-Schleife. Wird lange Zeit nichts in die DB geschrieben, beendet die DB die Verbindung und beim nächsten Schreibversuch stürzt das Script ab.
    Ich habe es mir relativ einfach gemacht und die Connection-Zeit der Datenbank entsprechend erhöht. Das Script läuft mittlerweile eine Woche ohne einen Absturz.
    Alternativ könnte man den Verbindungsaufbau zur Datenbank wahrscheinlich auch mit in die While-Schleife aufnehmen und die Connection in der Schleife auch wieder beenden.

    Viele Grüße
    Boris

    Antwort
    • 13. Oktober 2017 um 15:08 Uhr
      Permalink

      Hi Boris,

      Du hast recht! Ich habe die gleichen Probleme gehabt und das Connect in die while aufgenommen, damals aber leider nicht im Blog geändert was ich nun nachgeholt habe.
      Vielen Dank für den Hinweis!

      Gruß
      Chris

      Antwort
  • 15. März 2018 um 13:36 Uhr
    Permalink

    Ich bin mit meinen 65 Jahren am Ende.
    Wie muss die Datenbank aussehen. Einfach per Befehl die DB anlegen, oder ????

    M.f.G manfred

    Antwort
  • 15. Juni 2019 um 21:01 Uhr
    Permalink

    Wäre es möglich den kompletten Quelltext der (täglichen) PHP Auswertung online zu stellen? MySQL ist nicht so meins… 🙂

    Leider liefert mein Kontakt auch noch keine Werte am Zähler, mit einem Magneten funktioniert es und die DB wird gefüllt. Gibt es Tipps für die Positionierung des Kontakts am Gaszähler?

    MfG

    Antwort
    • 17. Juni 2019 um 17:33 Uhr
      Permalink

      Hallo Steven,

      die tägliche (und montaliche) Auswertung wir über eine PHP-Webseite erstellt.
      Die gibt es komplett hier als tar.gz zum Download. Die Tabelle des Gaszählers lese ich in “gaszaehler.php” aus.
      In der config.inc muss die DB-Verbindung hinterlegt werden und dort können auch der Startwer, Zustandszahl und Brenwert angegeben werden.

      Gruß
      Chris

      Antwort
    • 28. Juni 2019 um 20:03 Uhr
      Permalink

      Ich habe den gleichen Zähler wie Chris und bei mir war es so, dass ich den Kontakt bis zum Anschlag in die kleine Mulde vom Zähler stecken musste. Dabei musste Kontakt aber grade (parallel zum Zähler) stehen.

      Zur Auswertung: Man kann die Werte auch prima in ein Graphite (Whisper) schreiben. Dann ein Grafana als Auswertung nutzen und man hat alle Möglichkeiten für Graphen. Klappt bei mir prima und sieht auch noch gut aus.

      Antwort
  • 28. Juni 2019 um 20:17 Uhr
    Permalink

    Hi Chris,

    toller Beitrag, der mich inspiriert hat, das gleiche Projekt zu bauen.
    Als ich dann noch sah, dass ich den exakt gleichen Zähler habe, habe ich auch den identischen Reedkontakt geholt.

    Die Schleife, um das Schliessen des Kontakes mitzubekommen, kann man aber deutlich eleganter und mit viel weniger Code, mit “wait_for_edge” lösen. Möglicherweise gab es diese Variante aber beim Erstellen deines Posts noch nicht. So habe ich es implementiert:

    GPIO.setup(25, GPIO.IN, pull_up_down=GPIO.PUD_UP)
    while True:
    channel = GPIO.wait_for_edge(25, GPIO.FALLING, bouncetime=1000)
    with open(logfile, “a”, ) as file:
    file.write(str(time.time()) + ” Umdrehung!\n”)
    time.sleep(1)

    Damit spart man sich die ganzen Überprüfungen auf eine Änderung, da das mit dem “wait_for_edge” erledigt wird.

    Danke für deinen Beitrag und die gute Erklärung!

    Gruss
    Cotti

    Antwort
    • 30. Juni 2019 um 16:47 Uhr
      Permalink

      Hallo Cotti,

      danke für den Hinweis. Werde Deine Variante mal testen.

      Gruß
      Chris

      Antwort
  • Pingback: Gas- und Wasserzähler mit Tasmota digitalisieren - Smart home Ratgeber

Schreibe einen Kommentar zu Felix Antworten abbrechen

Deine E-Mail-Adresse wird nicht veröffentlicht. Erforderliche Felder sind mit * markiert