The SlimmeLezer is a compact build easy to use module to read data via the P1 port on a Smart Meter. Based on an ESP8266 (Wemos D1), the SlimmeLezer is perfect to use with ESPHome and integrates seamless into Home Assistant.
This repository is a fork from the original zuidwijk/dsmr but adapted to support DSMR versions 2.2 and 3.0.
To quickly get DSMR2.2/3.0 support for the SlimmeLezer without impacting working version 4/5 users, this fork was created after some discussion in issue #12.
There has now been work on the upstream ESPHome DSMR code to get the changes below incorporated as well and have all versions supported in the same codebase. See: esphome#2157
You can use the example configuration YAML in that PR to test if the proposed changes work for you. Please test that configuration first before using the one from this repo! The more we test the faster we can get the required changes merged into the main ESPHome codebase.
The changes needed for DSMR v2.2/3 and included in this repo are the following:
DSMR versions 2.2 and 3.0 use different settings for the serial interface:
uart:
baud_rate: 9600
data_bits: 7
parity: EVEN
stop_bits: 1
rx_pin: D7
This can mess with the logging over the same channel, so it is disabled. You can still see your logs through the API.
logger:
baud_rate: 0
Since CRCs are not present in DSMR 2.2, I have removed all the references to the CRC check methods in the P1Parser
of the parser.h
file:
// Look for ! that terminates the data
const char *data_end = data_start;
- uint16_t crc = _crc16_update(0, *str); // Include the / in CRC
while (data_end < str + n && *data_end != '!') {
- crc = _crc16_update(crc, *data_end);
++data_end;
}
- if (data_end >= str + n)
- return res.fail(F("No checksum found"), data_end);
-
- crc = _crc16_update(crc, *data_end); // Include the ! in CRC
-
- ParseResult<uint16_t> check_res = CrcParser::parse(data_end + 1, str + n);
- if (check_res.err)
- return check_res;
-
- // Check CRC
- if (check_res.result != crc) {
- Serial.println(check_res.result);
- Serial.println(crc);
-
- return res.fail(F("Checksum mismatch"), data_end + 1);
- }
-
res = parse_data(data, data_start, data_end, unknown_error);
- res.next = check_res.next;
+ res.next = data_end;
return res;
}
A better solution would of course be to make the CRC check optional by setting a boolean value in the YAML file, which is now being considered in esphome#2157.
Based on an older comment on an upstream discussion a hack has been applied that looks past the carriage return/line feed whether there is a value there and includes it to be parsed.
// Parse data lines
while (line_end < end) {
if (*line_end == '\r' || *line_end == '\n') {
if (!(*(line_end+1) =='(' || *(line_end + 2) =='(')){
ParseResult<void> tmp = parse_line(data, line_start, line_end, unknown_error);
if (tmp.err)
return tmp;
line_start = line_end + 1;
}
}
line_end++;
}
A similar approach is being investigated upstream as well. To make the value parsed in the code itself there is more refactoring needed, as explained in the upstream issue
This non-parsed field is then added as a value to be delivered to Home Assistant (in slimmelezer.yaml
):
text_sensor:
- platform: dsmr
gas_delivered_hack:
name: "Gas Delivered"
What you will end up with in Home Assistant is a bit of an ugly looking text value like this:
(210816110000)(08)(60)(1)(0-1:24.2.0)(m3)
(09833.800)
But with a bit of regex magic you can turn it into a numeric value by adding a template sensor in your Home Assistant configuration:
template:
sensor:
- name: "Gas Delivered Numeric"
unique_id: gas_delivered_numeric
state: '{{states("sensor.gas_delivered_hack")| regex_findall_index(find="\d*\.\d*", index=2)|float}}'
unit_of_measurement: "m³"
state_class: measurement
If this does not result in the right value, you can paste the state template in the Template Editor in the Home Assistant
Developer Tools tab and play with the number behind index=
.
Below is the original README from zuidwijk/dsmr.
The main goal is to create one universal component, which can be used in every country. Though the DSMR (Dutch Smart Meter Requirements) is specified with pre specified OBIS code, not every country has exact the same code. Some examples:
Version information for P1 output
- Default OBIS: 1-3:0.2.8.255
- Belgium OBIS: 0-0:96.1.4.255
Meter Reading electricity delivered to client (Tariff 1) in 0,001 kWh
- Default OBIS: 1-0:1.8.1.255
- Luxembourg OBIS: 1-0:1.8.0.255
Meter Reading electricity delivered by dient (Tariff 1) in 0,001 kWh
- Default OBIS: 1-0:2.8.1.255
- Luxembourg OBIS: 1-0:2.8.0.255
Some countries like Luxembourg, Sweden and Hungary, uses kvar next to kW. Therefor all deviant OBIS code is added as extra fields. This gives more sensors than needed, yet it can be used in every country where DSMR based Smart Meters is being used.
Smart Meters used in Luxembourg are using encryption. Decryption for Luxembourg is build in the code. This can be defined in the code:
dsmr:
id: dsmr_instance
decryption_key: '00112233445566778899AABBCCDDEEFF'
When the key is not set in the code, or when the key changes, it can be set/changes via a Service within Home Assistant, created via below api:
# Enable Home Assistant API
api:
services:
service: set_dsmr_key
variables:
private_key: string
then:
- logger.log:
format: Setting private key %s. Set to empty string to disable
args: [private_key.c_str()]
- globals.set:
id: has_key
value: !lambda "return private_key.length() == 32;"
- lambda: |-
if (private_key.length() == 32)
private_key.copy(id(stored_decryption_key), 32);
id(dsmr_instance).set_decryption_key(private_key);
In Home Assistant go to Services and select the service ESPHome: {name}_set_dsmr_key. There fill in the code received from the provider:
The SlimmeLezer is built with a logic inverter on the pcb. Connecting that directly to the Rx of the Wemos, causes that it can't be flashed via USB as it constanly pulls the Rx either high or low. Therefor I'm using the 2nd uart, on pin D7. That's why the uart is specified on pin D7 in the code:
uart:
baud_rate: 115200
rx_pin: D7