Digitalen Stromzähler ins openHAB

Für einen weiteren Stromanschluss in der Garage habe ich einen digitalen Zähler bekomme den ich gerne auslesen möchte um den Stromverbrauch der Wallbox zu analysieren. Dafür natürlich ein Wemos D1 Mini mit einem (modifizierten) TCRT5000 und einem BME280 für ein paar Klimadaten.

Material

ESP8266-12F * Affiliatelink

Jumper-Kabel * Affiliatelink

TCRT5000 * Affiliatelink

BME280 * Affiliatelink (aktuell sehr teuer, müsst ihr mal nach günstigeren suchen oder evtl. drauf verzichten)

Anschlüsse

Stromzähler

Ich habe einen Stromzähler mit IR-LED-Ausgang mit 10.000 Imp/kWh. Meine Wallbox in der Garage hat eine Nennleistung 11kW, man erlaube mir meine Gedanken auf die Leistung einzugrenzen 🙂

Also, 11 kW * 10.000 Imp/kWh = 110.000 Imp/h = 1833 Imp/Min = 30,56 Imp/Sek.

Warum die Gedanken: kommt gleich beim Arduino-Sketch.

Um die Diode am Zähler anzubringen hatte ich Glück, dass zwischen Zähler und Rückwandplatte noch ein wenig Platz war und ich ein gedrucktes Teil dort anbringen konnte wodurch die Diode genau auf die IR-LED „guckt“. Damit ist die Erfassung perfekt (glaube ich jedenfalls).

Arduino

Der Code beginnt wie immer mit den Variablen. Dann Setup mit dem üblichen, WLAN, MQTT, etc.

Ich habe im loop auf alle Ausgaben verzichtet um das Programm möglichst schnell zu lassen. Wie man sieht gibt es am Ende des loops kein „delay()“. Hintergrund ist, dass ich durch ein delay durchaus einen Impuls verpassen könnte. Also möglichst oft die Eingänge prüfen.

Die Frage ist: Wie schnell ist eigentlich der Wemos D1 mini den ich verwende? Ich hab einen einfachen Sketch erstellt um das zu prüfen. Ergebnis: 5.054.487 Loops/Minute, also ca. 85 Loops/ms. Sollte also für meinen Stromzähler reichen.

#include <ESP8266WiFi.h>
#include <PubSubClient.h>
#include <Adafruit_Sensor.h>
#include <Adafruit_BME280.h>
#include <ArduinoOTA.h>
#include <NTPClient.h>
#include <WiFiUdp.h> 
#include <time.h>
// #############################################
// WLAN-Zugang
// #############################################
#define wifi_ssid "Garagen-WLAN"
#define wifi_password "Garagen-Passwort"
// #############################################
// MQTT-Server (Openhab)
// #############################################
#define mqtt_server "192.168.2.2"
#define mqtt_user "openhabian"
#define mqtt_password ""
// #############################################
// NTP-Client
// #############################################
#define MY_NTP_SERVER "192.168.2.1"           
#define MY_TZ "CET-1CEST,M3.5.0/02,M10.5.0/03"
// #############################################
// MQTT-Topics
// ############################################
// BME280
#define topic_temperature "garage/temperature"
#define topic_humidity "garage/humidity"
#define topic_pressure "garage/pressure"
#define topic_dewpoint "garage/dewpoint"
#define topic_heatindex "garage/heatindex"
// IR-LED Stromzähler
#define topic_energy_daily "stromGarage/tagesstand"
#define topic_energy_diff "stromGarage/diff"
// Connections
WiFiClient espClient;
PubSubClient client(espClient);
// Sensoren
Adafruit_BME280 bme; // I2C
// PIN-Assignment
int Digital_Eingang = 16; // Binärer Wert
// Variablen zur Laufzeit
long lastMsg = 0;           // Timestamp der letzten Messung
int DigitalLast = 0;        // Energie-Impuls im letzten Loop
int daily_count_energy = 0; // Tagesstand
int daily_count_diff = 0; // Tagesstand
time_t now;                 // Zeiten:
tm tm;                      // ||
void setup ()
{
  pinMode (Digital_Eingang, INPUT);
  Serial.begin(115200);
  setup_wifi();
  client.setServer(mqtt_server, 1883);
  while(!Serial);    // time to get serial running
    unsigned status;
    status = bme.begin(0x76);  
    if (!status) {
      Serial.println("Could not find a valid BME280 sensor, check wiring, address, sensor ID!");
      while (1);
    }
  configTime(MY_TZ, MY_NTP_SERVER);
  
  ArduinoOTA.setHostname("ESPGarageStrom");
  ArduinoOTA.setPassword("OTAupdate");
  ArduinoOTA.begin();
}
void loop ()
{
  if (!client.connected()) {
    reconnect();
  }
// ## ###############
// ## Stromzähler
// ## ###############
  int Digital = digitalRead (Digital_Eingang);
  if(Digital==1 && Digital!=DigitalLast) {
      // Zählimpuls
      daily_count_energy++;
      daily_count_diff++;
  }
  if (Digital!=DigitalLast) {
    DigitalLast = Digital;
  }
  time(&now);                       // read the current time
  localtime_r(&now, &tm);           // update the structure tm with the current time
  // Um Null Uhr den Zähler wieder zurück setzen!
  if (tm.tm_hour == 0 && tm.tm_min == 0 && tm.tm_sec == 0) {
    // Ein letztes mal setzen!
    client.publish(topic_energy_daily, String(daily_count_energy).c_str(), true);
    daily_count_energy = 0;
  }
// ## ###############
// ## Temperaturen
// ## ###############
  long now = millis();
  if (now - lastMsg > (1 * 60 * 1000)) {
    // Nur jede Minute Messwerte ermitteln!
    lastMsg = now;
    
    float bme_temp = bme.readTemperature();
    float bme_humidity = bme.readHumidity();
    float bme_pressure = bme.readPressure() / 100.0F;
    float T = (bme_temp * 9 / 5) + 32;           // Convert back to deg-F for the RH equation
    float RHx = bme_humidity;                    // Short form of RH for inclusion in the equation makes it easier to read
    float heat_index = (-42.379+(2.04901523*T)+(10.14333127*RHx)-(0.22475541*T*RHx)-(0.00683783*sq(T))-(0.05481717*sq(RHx))+(0.00122874*sq(T)*RHx)+(0.00085282*T*sq(RHx))-(0.00000199*sq(T)*sq(RHx))-32)*5/9;
    if ((bme_temp <= 26.66) || (bme_humidity <= 40)) heat_index = bme_temp; // The convention is not to report heat Index when temperature is < 26.6 Deg-C or humidity < 40%
    float dew_point = 243.04*(log(bme_humidity/100)+((17.625*bme_temp)/(243.04+bme_temp)))/(17.625-log(bme_humidity/100)-((17.625*bme_temp)/(243.04+bme_temp)));
    client.publish(topic_temperature, String(bme_temp).c_str(), true);
    client.publish(topic_humidity, String(bme_humidity).c_str(), true);
    client.publish(topic_pressure, String(bme_pressure).c_str(), true);
    client.publish(topic_dewpoint, String(dew_point).c_str(), true);
    client.publish(topic_heatindex, String(heat_index).c_str(), true);
    client.publish(topic_energy_daily, String(daily_count_energy).c_str(), true);
    client.publish(topic_energy_diff, String(daily_count_diff).c_str(), true);
    daily_count_diff = 0;
  }
}
void setup_wifi() {
  delay(10);
  // We start by connecting to a WiFi network
  Serial.println();
  Serial.print("Connecting to ");
  Serial.println(wifi_ssid);
  WiFi.begin(wifi_ssid, wifi_password);
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }
  Serial.println("");
  Serial.println("WiFi connected");
  Serial.println("IP address: ");
  Serial.print(WiFi.localIP());
}
void reconnect() {
  // Loop until we're reconnected
  while (!client.connected()) {
    Serial.print("Attempting MQTT connection...");
    // Create a random client ID
    String clientId = "ESP8266-";
    clientId += String(random(0xffff), HEX);
    // Attempt to connect
    if (client.connect(clientId.c_str(), mqtt_user, mqtt_password)) {
      Serial.println("connected");
      // Once connected, publish an announcement...
      client.publish("outTopic", "ESP01 alive");
      // ... and resubscribe
      client.subscribe("inTopic");
    } else {
        Serial.print("failed, rc=");
        Serial.print(client.state());
        Serial.println(" try again in 5 seconds");
        // Wait 5 seconds before retrying
        delay(5000);
      }
   }
}

Zusätzlich habe ich noch einen BME280 angeschlosen um die Temperatur in der Garage ebenfalls messen zu können.

Der Loop läuft durch, wenn sich der Wert ändert, werden interne Zähler hochgesetzt. Einmal einen Tageszähler der Nachts noch mal per MQTT gesendet und dann wieder auf 0 gesetzt wird. Per MQTT werden damit nur jede Minute Werte übertragen: Klimadaten, Tageswert und Minutenwert des Stromzählers.

openHAB

Im openHAB werden dann die weiteren Auswertungen gemacht die man so glaubt zu brauchen:

var Number Letzter_abgelesener_Zaehlerstand = 37.5
// Ab 01.10.2022: 19.269ct
var Number Strompreis_Monat = 81.12	// € pro Jahr
var Number Strompreis_Tag = 22.22465753424658 // ct pro Tag
var Number Strompreis_kWh = 33.33	// ct pro kWh
rule "Increment Strommeter"
when
	Item  MQTT_Garage_Energie changed
then
// Initialer Start - die beiden folgenden Zeilen auskommentieren und anschließend einen Zählimpuls auslösen damit das Regelwerk einmal durchlaufen wird!
//  postUpdate(Strom_Garage_Heute, 0.0)
//  postUpdate(Strom_Garage_Gestern, 0.0)
//  postUpdate(Strom_Garage_Stand, 0.0)
// Aktuellen Zaehlerstand setzten - die folgenden drei Zeilen auskommentieren und anschließend einen Zählimpuls auslösen damit das Regelwerk einmal durchlaufen wird!
	if (Strom_Garage_Stand.state != NULL && Strom_Garage_Stand.state < Letzter_abgelesener_Zaehlerstand)  {
		postUpdate(Strom_Garage_Stand, (Letzter_abgelesener_Zaehlerstand))
	}
    // Kommender Wert:
    // MQTT_Garage_Energie  := Tageswert bisher (in Impulsen)
    //  10.000 Impulse      := 1kWh
    //  -> 1 Impuls         := 0,0001 kWh || 0,1 Wh || 100mWh
	postUpdate(Strom_Garage_Heute, (Strom_Garage_Heute.state as DecimalType + (MQTT_Garage_EnergieDiff.state as DecimalType / 10)))    // 0,1Wh
	postUpdate(Strom_Garage_Stand, (Strom_Garage_Stand.state as DecimalType + ((MQTT_Garage_EnergieDiff.state as DecimalType / 10) / 1000))    // 0,1Wh
	postUpdate(Stromkosten_Garage_Heute, ((Strom_Garage_Heute.state as DecimalType * Strompreis_kWh) + Strompreis_Tag) / 100)
end
rule "Reset Tagesverbrauch"
when
	Time is midnight
then
    postUpdate(Strom_Garage_Gestern, Strom_Garage_Heute.state)
    postUpdate(Strom_Garage_Heute, 0.0)
	
    postUpdate(Stromkosten_Garage_Gestern, Stromkosten_Garage_Heute.state)
    postUpdate(Stromkosten_Garage_Heute, (Strompreis_Tag / 100))
end

Das wars dann auch schon. Damit kann man dann noch schöne Grafiken in Grafane bauen wenn man will:

Schreibe einen Kommentar

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