2017-05-02 16:18:47 +00:00
|
|
|
"""
|
|
|
|
Support for Dutch Smart Meter Requirements.
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
Also known as: Smartmeter or P1 port.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/sensor.dsmr/
|
|
|
|
|
|
|
|
Technical overview:
|
|
|
|
|
|
|
|
DSMR is a standard to which Dutch smartmeters must comply. It specifies that
|
|
|
|
the smartmeter must send out a 'telegram' every 10 seconds over a serial port.
|
|
|
|
|
|
|
|
The contents of this telegram differ between version but they generally consist
|
|
|
|
of lines with 'obis' (Object Identification System, a numerical ID for a value)
|
|
|
|
followed with the value and unit.
|
|
|
|
|
|
|
|
This module sets up a asynchronous reading loop using the `dsmr_parser` module
|
|
|
|
which waits for a complete telegram, parser it and puts it on an async queue as
|
|
|
|
a dictionary of `obis`/object mapping. The numeric value and unit of each value
|
|
|
|
can be read from the objects attributes. Because the `obis` are know for each
|
|
|
|
DSMR version the Entities for this component are create during bootstrap.
|
|
|
|
|
|
|
|
Another loop (DSMR class) is setup which reads the telegram queue,
|
|
|
|
stores/caches the latest telegram and notifies the Entities that the telegram
|
|
|
|
has been updated.
|
|
|
|
"""
|
|
|
|
import asyncio
|
|
|
|
from datetime import timedelta
|
2017-01-17 07:56:00 +00:00
|
|
|
from functools import partial
|
2016-12-04 04:45:42 +00:00
|
|
|
import logging
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
2016-12-04 04:45:42 +00:00
|
|
|
from homeassistant.const import (
|
2017-01-17 07:56:00 +00:00
|
|
|
CONF_HOST, CONF_PORT, EVENT_HOMEASSISTANT_STOP, STATE_UNKNOWN)
|
|
|
|
from homeassistant.core import CoreState
|
2016-12-04 04:45:42 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-11-23 07:03:39 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2016-12-04 04:45:42 +00:00
|
|
|
import voluptuous as vol
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2016-11-23 23:27:31 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2017-05-29 14:19:50 +00:00
|
|
|
REQUIREMENTS = ['dsmr_parser==0.9']
|
|
|
|
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
CONF_DSMR_VERSION = 'dsmr_version'
|
2017-01-17 07:56:00 +00:00
|
|
|
CONF_RECONNECT_INTERVAL = 'reconnect_interval'
|
2016-11-23 23:27:31 +00:00
|
|
|
|
2016-11-23 07:03:39 +00:00
|
|
|
DEFAULT_DSMR_VERSION = '2.2'
|
2016-11-23 23:27:31 +00:00
|
|
|
DEFAULT_PORT = '/dev/ttyUSB0'
|
|
|
|
DOMAIN = 'dsmr'
|
|
|
|
|
|
|
|
ICON_GAS = 'mdi:fire'
|
|
|
|
ICON_POWER = 'mdi:flash'
|
|
|
|
|
|
|
|
# Smart meter sends telegram every 10 seconds
|
|
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(seconds=10)
|
2017-01-17 07:56:00 +00:00
|
|
|
RECONNECT_INTERVAL = 5
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.string,
|
2017-01-17 07:56:00 +00:00
|
|
|
vol.Optional(CONF_HOST, default=None): cv.string,
|
2016-11-23 07:03:39 +00:00
|
|
|
vol.Optional(CONF_DSMR_VERSION, default=DEFAULT_DSMR_VERSION): vol.All(
|
2017-05-29 14:19:50 +00:00
|
|
|
cv.string, vol.In(['5', '4', '2.2'])),
|
2017-01-17 07:56:00 +00:00
|
|
|
vol.Optional(CONF_RECONNECT_INTERVAL, default=30): int,
|
2016-11-23 07:03:39 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup_platform(hass, config, async_add_devices, discovery_info=None):
|
2016-11-23 23:27:31 +00:00
|
|
|
"""Set up the DSMR sensor."""
|
|
|
|
# Suppress logging
|
2016-11-23 07:03:39 +00:00
|
|
|
logging.getLogger('dsmr_parser').setLevel(logging.ERROR)
|
|
|
|
|
2016-12-04 04:45:42 +00:00
|
|
|
from dsmr_parser import obis_references as obis_ref
|
2017-05-02 16:18:47 +00:00
|
|
|
from dsmr_parser.clients.protocol import (
|
|
|
|
create_dsmr_reader, create_tcp_dsmr_reader)
|
2017-01-17 07:56:00 +00:00
|
|
|
import serial
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
dsmr_version = config[CONF_DSMR_VERSION]
|
|
|
|
|
2016-11-23 23:27:31 +00:00
|
|
|
# Define list of name,obis mappings to generate entities
|
2016-11-23 07:03:39 +00:00
|
|
|
obis_mapping = [
|
2016-12-04 04:45:42 +00:00
|
|
|
['Power Consumption', obis_ref.CURRENT_ELECTRICITY_USAGE],
|
|
|
|
['Power Production', obis_ref.CURRENT_ELECTRICITY_DELIVERY],
|
|
|
|
['Power Tariff', obis_ref.ELECTRICITY_ACTIVE_TARIFF],
|
|
|
|
['Power Consumption (low)', obis_ref.ELECTRICITY_USED_TARIFF_1],
|
|
|
|
['Power Consumption (normal)', obis_ref.ELECTRICITY_USED_TARIFF_2],
|
|
|
|
['Power Production (low)', obis_ref.ELECTRICITY_DELIVERED_TARIFF_1],
|
|
|
|
['Power Production (normal)', obis_ref.ELECTRICITY_DELIVERED_TARIFF_2],
|
2016-11-23 07:03:39 +00:00
|
|
|
]
|
2016-12-04 04:45:42 +00:00
|
|
|
|
|
|
|
# Generate device entities
|
|
|
|
devices = [DSMREntity(name, obis) for name, obis in obis_mapping]
|
|
|
|
|
2016-11-23 23:27:31 +00:00
|
|
|
# Protocol version specific obis
|
2017-05-29 14:19:50 +00:00
|
|
|
if dsmr_version in ('4', '5'):
|
2016-12-04 04:45:42 +00:00
|
|
|
gas_obis = obis_ref.HOURLY_GAS_METER_READING
|
2016-11-23 07:03:39 +00:00
|
|
|
else:
|
2016-12-04 04:45:42 +00:00
|
|
|
gas_obis = obis_ref.GAS_METER_READING
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2016-12-04 04:45:42 +00:00
|
|
|
# add gas meter reading and derivative for usage
|
|
|
|
devices += [
|
|
|
|
DSMREntity('Gas Consumption', gas_obis),
|
|
|
|
DerivativeDSMREntity('Hourly Gas Consumption', gas_obis),
|
|
|
|
]
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2017-03-01 04:33:19 +00:00
|
|
|
async_add_devices(devices)
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
def update_entities_telegram(telegram):
|
|
|
|
"""Update entities with latests telegram & trigger state update."""
|
2016-11-23 23:27:31 +00:00
|
|
|
# Make all device entities aware of new telegram
|
2016-11-23 07:03:39 +00:00
|
|
|
for device in devices:
|
|
|
|
device.telegram = telegram
|
2017-03-16 05:58:54 +00:00
|
|
|
hass.async_add_job(device.async_update_ha_state())
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2017-01-17 07:56:00 +00:00
|
|
|
# Creates a asyncio.Protocol factory for reading DSMR telegrams from serial
|
2016-11-23 07:03:39 +00:00
|
|
|
# and calls update_entities_telegram to update entities on arrival
|
2017-01-17 07:56:00 +00:00
|
|
|
if config[CONF_HOST]:
|
2017-05-02 16:18:47 +00:00
|
|
|
reader_factory = partial(
|
|
|
|
create_tcp_dsmr_reader, config[CONF_HOST], config[CONF_PORT],
|
|
|
|
config[CONF_DSMR_VERSION], update_entities_telegram,
|
|
|
|
loop=hass.loop)
|
2017-01-17 07:56:00 +00:00
|
|
|
else:
|
2017-05-02 16:18:47 +00:00
|
|
|
reader_factory = partial(
|
|
|
|
create_dsmr_reader, config[CONF_PORT], config[CONF_DSMR_VERSION],
|
|
|
|
update_entities_telegram, loop=hass.loop)
|
2016-11-23 07:03:39 +00:00
|
|
|
|
2017-01-17 07:56:00 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def connect_and_reconnect():
|
|
|
|
"""Connect to DSMR and keep reconnecting until HA stops."""
|
|
|
|
while hass.state != CoreState.stopping:
|
2017-03-16 05:58:54 +00:00
|
|
|
# Start DSMR asyncio.Protocol reader
|
2017-01-17 07:56:00 +00:00
|
|
|
try:
|
|
|
|
transport, protocol = yield from hass.loop.create_task(
|
|
|
|
reader_factory())
|
|
|
|
except (serial.serialutil.SerialException, ConnectionRefusedError,
|
|
|
|
TimeoutError):
|
|
|
|
# log any error while establishing connection and drop to retry
|
|
|
|
# connection wait
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.exception("Error connecting to DSMR")
|
2017-01-17 07:56:00 +00:00
|
|
|
transport = None
|
|
|
|
|
|
|
|
if transport:
|
|
|
|
# register listener to close transport on HA shutdown
|
|
|
|
stop_listerer = hass.bus.async_listen_once(
|
|
|
|
EVENT_HOMEASSISTANT_STOP, transport.close)
|
|
|
|
|
|
|
|
# wait for reader to close
|
|
|
|
yield from protocol.wait_closed()
|
|
|
|
|
|
|
|
if hass.state != CoreState.stopping:
|
2017-04-19 03:24:44 +00:00
|
|
|
# unexpected disconnect
|
2017-01-17 07:56:00 +00:00
|
|
|
if transport:
|
|
|
|
# remove listerer
|
|
|
|
stop_listerer()
|
|
|
|
|
2017-04-19 03:24:44 +00:00
|
|
|
# reflect disconnect state in devices state by setting an
|
|
|
|
# empty telegram resulting in `unkown` states
|
|
|
|
update_entities_telegram({})
|
|
|
|
|
2017-01-17 07:56:00 +00:00
|
|
|
# throttle reconnect attempts
|
|
|
|
yield from asyncio.sleep(config[CONF_RECONNECT_INTERVAL],
|
|
|
|
loop=hass.loop)
|
|
|
|
|
2017-03-16 05:58:54 +00:00
|
|
|
# Cannot be hass.async_add_job because job runs forever
|
2017-01-17 07:56:00 +00:00
|
|
|
hass.loop.create_task(connect_and_reconnect())
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
class DSMREntity(Entity):
|
|
|
|
"""Entity reading values from DSMR telegram."""
|
|
|
|
|
|
|
|
def __init__(self, name, obis):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Initialize entity."""
|
2016-11-23 07:03:39 +00:00
|
|
|
self._name = name
|
|
|
|
self._obis = obis
|
|
|
|
self.telegram = {}
|
|
|
|
|
|
|
|
def get_dsmr_object_attr(self, attribute):
|
|
|
|
"""Read attribute from last received telegram for this DSMR object."""
|
2016-11-23 23:27:31 +00:00
|
|
|
# Make sure telegram contains an object for this entities obis
|
2016-11-23 07:03:39 +00:00
|
|
|
if self._obis not in self.telegram:
|
|
|
|
return None
|
|
|
|
|
|
|
|
# get the attibute value if the object has it
|
|
|
|
dsmr_object = self.telegram[self._obis]
|
|
|
|
return getattr(dsmr_object, attribute, None)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Icon to use in the frontend, if any."""
|
|
|
|
if 'Power' in self._name:
|
|
|
|
return ICON_POWER
|
|
|
|
elif 'Gas' in self._name:
|
|
|
|
return ICON_GAS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of sensor, if available, translate if needed."""
|
|
|
|
from dsmr_parser import obis_references as obis
|
|
|
|
|
|
|
|
value = self.get_dsmr_object_attr('value')
|
|
|
|
|
|
|
|
if self._obis == obis.ELECTRICITY_ACTIVE_TARIFF:
|
|
|
|
return self.translate_tariff(value)
|
|
|
|
else:
|
2017-04-19 03:24:44 +00:00
|
|
|
if value is not None:
|
2016-12-04 04:45:42 +00:00
|
|
|
return value
|
|
|
|
else:
|
|
|
|
return STATE_UNKNOWN
|
2016-11-23 07:03:39 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement of this entity, if any."""
|
|
|
|
return self.get_dsmr_object_attr('unit')
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def translate_tariff(value):
|
|
|
|
"""Convert 2/1 to normal/low."""
|
2016-11-23 23:27:31 +00:00
|
|
|
# DSMR V2.2: Note: Rate code 1 is used for low rate and rate code 2 is
|
|
|
|
# used for normal rate.
|
2016-11-23 07:03:39 +00:00
|
|
|
if value == '0002':
|
|
|
|
return 'normal'
|
|
|
|
elif value == '0001':
|
|
|
|
return 'low'
|
|
|
|
else:
|
2016-12-04 04:45:42 +00:00
|
|
|
return STATE_UNKNOWN
|
|
|
|
|
|
|
|
|
|
|
|
class DerivativeDSMREntity(DSMREntity):
|
|
|
|
"""Calculated derivative for values where the DSMR doesn't offer one.
|
|
|
|
|
|
|
|
Gas readings are only reported per hour and don't offer a rate only
|
|
|
|
the current meter reading. This entity converts subsequents readings
|
|
|
|
into a hourly rate.
|
2017-01-17 07:56:00 +00:00
|
|
|
|
2016-12-04 04:45:42 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
_previous_reading = None
|
|
|
|
_previous_timestamp = None
|
|
|
|
_state = STATE_UNKNOWN
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the calculated current hourly rate."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_update(self):
|
|
|
|
"""Recalculate hourly rate if timestamp has changed.
|
|
|
|
|
2017-01-17 07:56:00 +00:00
|
|
|
DSMR updates gas meter reading every hour. Along with the new
|
|
|
|
value a timestamp is provided for the reading. Test if the last
|
|
|
|
known timestamp differs from the current one then calculate a
|
|
|
|
new rate for the previous hour.
|
|
|
|
|
2016-12-04 04:45:42 +00:00
|
|
|
"""
|
|
|
|
# check if the timestamp for the object differs from the previous one
|
|
|
|
timestamp = self.get_dsmr_object_attr('datetime')
|
|
|
|
if timestamp and timestamp != self._previous_timestamp:
|
|
|
|
current_reading = self.get_dsmr_object_attr('value')
|
|
|
|
|
|
|
|
if self._previous_reading is None:
|
|
|
|
# can't calculate rate without previous datapoint
|
|
|
|
# just store current point
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# recalculate the rate
|
|
|
|
diff = current_reading - self._previous_reading
|
|
|
|
self._state = diff
|
|
|
|
|
|
|
|
self._previous_reading = current_reading
|
|
|
|
self._previous_timestamp = timestamp
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement of this entity, per hour, if any."""
|
|
|
|
unit = self.get_dsmr_object_attr('unit')
|
|
|
|
if unit:
|
|
|
|
return unit + '/h'
|