2016-08-22 18:16:11 +00:00
|
|
|
DSMR Parser
|
|
|
|
===========
|
|
|
|
|
2016-08-22 18:38:16 +00:00
|
|
|
.. image:: https://img.shields.io/pypi/v/dsmr-parser.svg
|
|
|
|
:target: https://pypi.python.org/pypi/dsmr-parser
|
|
|
|
|
2016-11-21 21:31:22 +00:00
|
|
|
.. image:: https://travis-ci.org/ndokter/dsmr_parser.svg?branch=master
|
|
|
|
:target: https://travis-ci.org/ndokter/dsmr_parser
|
|
|
|
|
2016-08-22 18:16:11 +00:00
|
|
|
A library for parsing Dutch Smart Meter Requirements (DSMR) telegram data. It
|
2017-02-03 21:40:39 +00:00
|
|
|
also includes client implementation to directly read and parse smart meter data.
|
2016-08-22 18:16:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
Features
|
|
|
|
--------
|
|
|
|
|
2017-01-26 18:06:03 +00:00
|
|
|
DSMR Parser supports DSMR versions 2, 3, 4 and 5. It has been tested with Python 3.4, 3.5 and 3.6.
|
2016-08-22 18:16:11 +00:00
|
|
|
|
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
Client module usage
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
**Serial client**
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-27 19:26:15 +00:00
|
|
|
Read the serial port and work with the parsed telegrams. It should be run in a separate
|
2017-02-03 21:40:39 +00:00
|
|
|
process because the code is blocking (not asynchronous):
|
2016-08-22 18:16:11 +00:00
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
from dsmr_parser import telegram_specifications
|
|
|
|
from dsmr_parser.clients import SerialReader, SERIAL_SETTINGS_V4
|
|
|
|
|
|
|
|
serial_reader = SerialReader(
|
|
|
|
device='/dev/ttyUSB0',
|
|
|
|
serial_settings=SERIAL_SETTINGS_V4,
|
|
|
|
telegram_specification=telegram_specifications.V4
|
|
|
|
)
|
|
|
|
|
|
|
|
for telegram in serial_reader.read():
|
|
|
|
print(telegram) # see 'Telegram object' docs below
|
|
|
|
|
|
|
|
**AsyncIO client**
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
To be documented.
|
|
|
|
|
|
|
|
|
|
|
|
Parsing module usage
|
|
|
|
--------------------
|
|
|
|
The parsing module accepts complete unaltered telegram strings and parses these
|
|
|
|
into a dictionary.
|
|
|
|
|
|
|
|
.. code-block:: python
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
from dsmr_parser import telegram_specifications
|
|
|
|
from dsmr_parser.parsers import TelegramParser
|
2017-02-03 21:42:31 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
telegram_str = (
|
|
|
|
'/ISk5\2MT382-1000\r\n'
|
|
|
|
'\r\n'
|
|
|
|
'0-0:96.1.1(4B384547303034303436333935353037)\r\n'
|
|
|
|
'1-0:1.8.1(12345.678*kWh)\r\n'
|
|
|
|
'1-0:1.8.2(12345.678*kWh)\r\n'
|
|
|
|
'1-0:2.8.1(12345.678*kWh)\r\n'
|
|
|
|
'1-0:2.8.2(12345.678*kWh)\r\n'
|
|
|
|
'0-0:96.14.0(0002)\r\n'
|
|
|
|
'1-0:1.7.0(001.19*kW)\r\n'
|
|
|
|
'1-0:2.7.0(000.00*kW)\r\n'
|
|
|
|
'0-0:17.0.0(016*A)\r\n'
|
|
|
|
'0-0:96.3.10(1)\r\n'
|
|
|
|
'0-0:96.13.1(303132333435363738)\r\n'
|
|
|
|
'0-0:96.13.0(303132333435363738393A3B3C3D3E3F303132333435363738393A3B3C3D3E'
|
|
|
|
'3F303132333435363738393A3B3C3D3E3F303132333435363738393A3B3C3D3E3F30313233'
|
|
|
|
'3435363738393A3B3C3D3E3F)\r\n'
|
|
|
|
'0-1:96.1.0(3232323241424344313233343536373839)\r\n'
|
|
|
|
'0-1:24.1.0(03)\r\n'
|
|
|
|
'0-1:24.3.0(090212160000)(00)(60)(1)(0-1:24.2.1)(m3)\r\n'
|
|
|
|
'(00001.001)\r\n'
|
|
|
|
'0-1:24.4.0(1)\r\n'
|
|
|
|
'!\r\n'
|
|
|
|
)
|
2017-02-03 21:42:31 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
parser = TelegramParser(telegram_specifications.V3)
|
|
|
|
|
|
|
|
telegram = parser.parse(telegram_str)
|
|
|
|
print(telegram) # see 'Telegram object' docs below
|
|
|
|
|
|
|
|
Telegram object
|
|
|
|
---------------
|
|
|
|
|
|
|
|
A dictionary of which the key indicates the field type. These regex values
|
|
|
|
correspond to one of dsmr_parser.obis_reference constants.
|
|
|
|
|
|
|
|
The value is either a CosemObject or MBusObject. These have a 'value' and 'unit'
|
|
|
|
property. MBusObject's additionally have a 'datetime' property. The 'value' can
|
|
|
|
contain any python type (int, str, Decimal) depending on the field. The 'unit'
|
|
|
|
contains 'kW', 'A', 'kWh' or 'm3'.
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
2017-02-27 19:26:15 +00:00
|
|
|
# Contents of a parsed DSMR v3 telegram
|
2017-02-03 21:40:39 +00:00
|
|
|
{'\\d-\\d:17\\.0\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39eb8>,
|
|
|
|
'\\d-\\d:1\\.7\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10f916390>,
|
|
|
|
'\\d-\\d:1\\.8\\.1.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39e10>,
|
|
|
|
'\\d-\\d:1\\.8\\.2.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39ef0>,
|
|
|
|
'\\d-\\d:24\\.1\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fbaef28>,
|
|
|
|
'\\d-\\d:24\\.3\\.0.+?\\r\\n.+?\\r\\n': <dsmr_parser.objects.MBusObject object at 0x10f9163c8>,
|
|
|
|
'\\d-\\d:24\\.4\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39f60>,
|
|
|
|
'\\d-\\d:2\\.7\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39fd0>,
|
|
|
|
'\\d-\\d:2\\.8\\.1.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fbaee10>,
|
|
|
|
'\\d-\\d:2\\.8\\.2.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39e80>,
|
|
|
|
'\\d-\\d:96\\.13\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39d30>,
|
|
|
|
'\\d-\\d:96\\.13\\.1.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fbaeeb8>,
|
|
|
|
'\\d-\\d:96\\.14\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fbaef98>,
|
|
|
|
'\\d-\\d:96\\.1\\.0.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fbaef60>,
|
|
|
|
'\\d-\\d:96\\.1\\.1.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39f98>,
|
|
|
|
'\\d-\\d:96\\.3\\.10.+?\\r\\n': <dsmr_parser.objects.CosemObject object at 0x10fc39dd8>}
|
|
|
|
|
2017-02-27 19:26:15 +00:00
|
|
|
Example to get some of the values:
|
2017-02-03 21:40:39 +00:00
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
from dsmr_parser import obis_references
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
# The telegram message timestamp.
|
|
|
|
message_datetime = telegram[obis_references.P1_MESSAGE_TIMESTAMP]
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
# Using the active tariff to determine the electricity being used and
|
|
|
|
# delivered for the right tariff.
|
|
|
|
active_tariff = telegram[obis_references.ELECTRICITY_ACTIVE_TARIFF]
|
|
|
|
active_tariff = int(tariff.value)
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
electricity_used_total = telegram[obis_references.ELECTRICITY_USED_TARIFF_ALL[active_tariff - 1]]
|
|
|
|
electricity_delivered_total = telegram[obis_references.ELECTRICITY_DELIVERED_TARIFF_ALL[active_tariff - 1]]
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
gas_reading = telegram[obis_references.HOURLY_GAS_METER_READING]
|
2016-08-22 18:16:11 +00:00
|
|
|
|
2017-02-03 21:40:39 +00:00
|
|
|
# See dsmr_reader.obis_references for all readable telegram values.
|
2017-02-27 19:26:15 +00:00
|
|
|
# Note that the avilable values differ per DSMR version.
|
2016-08-22 18:16:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
Installation
|
|
|
|
------------
|
|
|
|
|
|
|
|
To install DSMR Parser:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
|
|
|
|
|
|
|
$ pip install dsmr-parser
|
|
|
|
|
2016-11-21 20:30:56 +00:00
|
|
|
Known issues
|
|
|
|
------------
|
|
|
|
|
|
|
|
If the serial settings SERIAL_SETTINGS_V2_2 or SERIAL_SETTINGS_V4 don't work.
|
|
|
|
Make sure to try and replace the parity settings to EVEN or NONE.
|
|
|
|
It's possible that alternative settings will be added in the future if these
|
|
|
|
settings don't work for the majority of meters.
|