All source listed below is under MIT license if no LICENSE file stating different is available.
DSMR Parser
===========

.. image:: https://img.shields.io/pypi/v/dsmr-parser.svg
    :target: https://pypi.python.org/pypi/dsmr-parser

.. image:: https://travis-ci.org/ndokter/dsmr_parser.svg?branch=master
    :target: https://travis-ci.org/ndokter/dsmr_parser

A library for parsing Dutch Smart Meter Requirements (DSMR) telegram data. It
also includes client implementation to directly read and parse smart meter data.


Features
--------

DSMR Parser supports DSMR versions 2, 3, 4 and 5. It has been tested with Python 3.4, 3.5 and 3.6.


Client module usage
-------------------

**Serial client**

The most simple client is the serial client. It should be run in a separate
process because the code is blocking (not asynchronous):

.. code-block:: python

     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**

To be documented.


Parsing module usage
--------------------
The parsing module accepts complete unaltered telegram strings and parses these
into a dictionary.

.. code-block:: python

     from dsmr_parser import telegram_specifications
     from dsmr_parser.parsers import TelegramParser
     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'
     )
     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

    # Contents of parsed DSMR v3 telegram
    {'\\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>}

Example to get values:

.. code-block:: python

    from dsmr_parser import obis_references

     # The telegram message timestamp.
     message_datetime = telegram[obis_references.P1_MESSAGE_TIMESTAMP]

     # 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)

     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]]

     gas_reading = telegram[obis_references.HOURLY_GAS_METER_READING]

    # See dsmr_reader.obis_references for all readable telegram values.


Installation
------------

To install DSMR Parser:

.. code-block:: bash

    $ pip install dsmr-parser

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.
dsmr_parser
test
.coveragerc
.gitignore
.travis.yml
CHANGELOG.rst
LICENSE
README.rst
setup.py
tox.ini