Merge branch 'master' of github.com:ndokter/dsmr_parser
This commit is contained in:
commit
2dafc7e7e4
19
README.rst
19
README.rst
@ -46,6 +46,25 @@ process because the code is blocking (not asynchronous):
|
|||||||
|
|
||||||
To be documented.
|
To be documented.
|
||||||
|
|
||||||
|
**Socket client**
|
||||||
|
|
||||||
|
Read a remote serial port (for example using ser2net) and work with the parsed telegrams.
|
||||||
|
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 SocketReader
|
||||||
|
|
||||||
|
socket_reader = SocketReader(
|
||||||
|
host='127.0.0.1',
|
||||||
|
port=2001,
|
||||||
|
telegram_specification=telegram_specifications.V4
|
||||||
|
)
|
||||||
|
|
||||||
|
for telegram in socket_reader.read():
|
||||||
|
print(telegram) # see 'Telegram object' docs below
|
||||||
|
|
||||||
|
|
||||||
Parsing module usage
|
Parsing module usage
|
||||||
--------------------
|
--------------------
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
from dsmr_parser.clients.settings import SERIAL_SETTINGS_V2_2, \
|
from dsmr_parser.clients.settings import SERIAL_SETTINGS_V2_2, \
|
||||||
SERIAL_SETTINGS_V4, SERIAL_SETTINGS_V5
|
SERIAL_SETTINGS_V4, SERIAL_SETTINGS_V5
|
||||||
from dsmr_parser.clients.serial_ import SerialReader, AsyncSerialReader
|
from dsmr_parser.clients.serial_ import SerialReader, AsyncSerialReader
|
||||||
|
from dsmr_parser.clients.socket_ import SocketReader
|
||||||
from dsmr_parser.clients.protocol import create_dsmr_protocol, \
|
from dsmr_parser.clients.protocol import create_dsmr_protocol, \
|
||||||
create_dsmr_reader, create_tcp_dsmr_reader
|
create_dsmr_reader, create_tcp_dsmr_reader
|
||||||
|
@ -16,6 +16,13 @@ from dsmr_parser.clients.settings import SERIAL_SETTINGS_V2_2, \
|
|||||||
|
|
||||||
def create_dsmr_protocol(dsmr_version, telegram_callback, loop=None, **kwargs):
|
def create_dsmr_protocol(dsmr_version, telegram_callback, loop=None, **kwargs):
|
||||||
"""Creates a DSMR asyncio protocol."""
|
"""Creates a DSMR asyncio protocol."""
|
||||||
|
protocol = _create_dsmr_protocol(dsmr_version, telegram_callback,
|
||||||
|
DSMRProtocol, loop, **kwargs)
|
||||||
|
return protocol
|
||||||
|
|
||||||
|
|
||||||
|
def _create_dsmr_protocol(dsmr_version, telegram_callback, protocol, loop=None, **kwargs):
|
||||||
|
"""Creates a DSMR asyncio protocol."""
|
||||||
|
|
||||||
if dsmr_version == '2.2':
|
if dsmr_version == '2.2':
|
||||||
specification = telegram_specifications.V2_2
|
specification = telegram_specifications.V2_2
|
||||||
@ -45,7 +52,7 @@ def create_dsmr_protocol(dsmr_version, telegram_callback, loop=None, **kwargs):
|
|||||||
raise NotImplementedError("No telegram parser found for version: %s",
|
raise NotImplementedError("No telegram parser found for version: %s",
|
||||||
dsmr_version)
|
dsmr_version)
|
||||||
|
|
||||||
protocol = partial(DSMRProtocol, loop, TelegramParser(specification),
|
protocol = partial(protocol, loop, TelegramParser(specification),
|
||||||
telegram_callback=telegram_callback, **kwargs)
|
telegram_callback=telegram_callback, **kwargs)
|
||||||
|
|
||||||
return protocol, serial_settings
|
return protocol, serial_settings
|
||||||
@ -113,7 +120,7 @@ class DSMRProtocol(asyncio.Protocol):
|
|||||||
self.telegram_buffer.append(data)
|
self.telegram_buffer.append(data)
|
||||||
|
|
||||||
for telegram in self.telegram_buffer.get_all():
|
for telegram in self.telegram_buffer.get_all():
|
||||||
# ensure actual telegram is ascii (7-bit) only (IEC 646 required in section 5.4 of IEC 62056-21)
|
# ensure actual telegram is ascii (7-bit) only (ISO 646:1991 IRV required in section 5.5 of IEC 62056-21)
|
||||||
telegram = telegram.encode("latin1").decode("ascii")
|
telegram = telegram.encode("latin1").decode("ascii")
|
||||||
self.handle_telegram(telegram)
|
self.handle_telegram(telegram)
|
||||||
|
|
||||||
|
62
dsmr_parser/clients/rfxtrx_protocol.py
Normal file
62
dsmr_parser/clients/rfxtrx_protocol.py
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
"""Asyncio protocol implementation for handling telegrams over a RFXtrx connection ."""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
|
||||||
|
from serial_asyncio import create_serial_connection
|
||||||
|
from .protocol import DSMRProtocol, _create_dsmr_protocol
|
||||||
|
|
||||||
|
|
||||||
|
def create_rfxtrx_dsmr_protocol(dsmr_version, telegram_callback, loop=None, **kwargs):
|
||||||
|
"""Creates a RFXtrxDSMR asyncio protocol."""
|
||||||
|
protocol = _create_dsmr_protocol(dsmr_version, telegram_callback,
|
||||||
|
RFXtrxDSMRProtocol, loop, **kwargs)
|
||||||
|
return protocol
|
||||||
|
|
||||||
|
|
||||||
|
def create_rfxtrx_dsmr_reader(port, dsmr_version, telegram_callback, loop=None):
|
||||||
|
"""Creates a DSMR asyncio protocol coroutine using a RFXtrx serial port."""
|
||||||
|
protocol, serial_settings = create_rfxtrx_dsmr_protocol(
|
||||||
|
dsmr_version, telegram_callback, loop=None)
|
||||||
|
serial_settings['url'] = port
|
||||||
|
|
||||||
|
conn = create_serial_connection(loop, protocol, **serial_settings)
|
||||||
|
return conn
|
||||||
|
|
||||||
|
|
||||||
|
def create_rfxtrx_tcp_dsmr_reader(host, port, dsmr_version,
|
||||||
|
telegram_callback, loop=None,
|
||||||
|
keep_alive_interval=None):
|
||||||
|
"""Creates a DSMR asyncio protocol coroutine using a RFXtrx TCP connection."""
|
||||||
|
if not loop:
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
protocol, _ = create_rfxtrx_dsmr_protocol(
|
||||||
|
dsmr_version, telegram_callback, loop=loop,
|
||||||
|
keep_alive_interval=keep_alive_interval)
|
||||||
|
conn = loop.create_connection(protocol, host, port)
|
||||||
|
return conn
|
||||||
|
|
||||||
|
|
||||||
|
PACKETTYPE_DSMR = 0x62
|
||||||
|
SUBTYPE_P1 = 0x01
|
||||||
|
|
||||||
|
|
||||||
|
class RFXtrxDSMRProtocol(DSMRProtocol):
|
||||||
|
|
||||||
|
remaining_data = b''
|
||||||
|
|
||||||
|
def data_received(self, data):
|
||||||
|
"""Add incoming data to buffer."""
|
||||||
|
|
||||||
|
data = self.remaining_data + data
|
||||||
|
|
||||||
|
packetlength = data[0] + 1 if len(data) > 0 else 1
|
||||||
|
while packetlength <= len(data):
|
||||||
|
packettype = data[1]
|
||||||
|
subtype = data[2]
|
||||||
|
if (packettype == PACKETTYPE_DSMR and subtype == SUBTYPE_P1):
|
||||||
|
dsmr_data = data[4:packetlength]
|
||||||
|
super().data_received(dsmr_data)
|
||||||
|
data = data[packetlength:]
|
||||||
|
packetlength = data[0] + 1 if len(data) > 0 else 1
|
||||||
|
|
||||||
|
self.remaining_data = data
|
90
dsmr_parser/clients/socket_.py
Normal file
90
dsmr_parser/clients/socket_.py
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
import logging
|
||||||
|
import socket
|
||||||
|
|
||||||
|
from dsmr_parser.clients.telegram_buffer import TelegramBuffer
|
||||||
|
from dsmr_parser.exceptions import ParseError, InvalidChecksumError
|
||||||
|
from dsmr_parser.parsers import TelegramParser
|
||||||
|
from dsmr_parser.objects import Telegram
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SocketReader(object):
|
||||||
|
|
||||||
|
BUFFER_SIZE = 256
|
||||||
|
|
||||||
|
def __init__(self, host, port, telegram_specification):
|
||||||
|
self.host = host
|
||||||
|
self.port = port
|
||||||
|
|
||||||
|
self.telegram_parser = TelegramParser(telegram_specification)
|
||||||
|
self.telegram_buffer = TelegramBuffer()
|
||||||
|
self.telegram_specification = telegram_specification
|
||||||
|
|
||||||
|
def read(self):
|
||||||
|
"""
|
||||||
|
Read complete DSMR telegram's from remote interface and parse it
|
||||||
|
into CosemObject's and MbusObject's
|
||||||
|
|
||||||
|
:rtype: generator
|
||||||
|
"""
|
||||||
|
buffer = b""
|
||||||
|
|
||||||
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as socket_handle:
|
||||||
|
|
||||||
|
socket_handle.connect((self.host, self.port))
|
||||||
|
|
||||||
|
while True:
|
||||||
|
buffer += socket_handle.recv(self.BUFFER_SIZE)
|
||||||
|
|
||||||
|
lines = buffer.splitlines(keepends=True)
|
||||||
|
|
||||||
|
if len(lines) == 0:
|
||||||
|
continue
|
||||||
|
|
||||||
|
for data in lines:
|
||||||
|
self.telegram_buffer.append(data.decode('ascii'))
|
||||||
|
|
||||||
|
for telegram in self.telegram_buffer.get_all():
|
||||||
|
try:
|
||||||
|
yield self.telegram_parser.parse(telegram)
|
||||||
|
except InvalidChecksumError as e:
|
||||||
|
logger.warning(str(e))
|
||||||
|
except ParseError as e:
|
||||||
|
logger.error('Failed to parse telegram: %s', e)
|
||||||
|
|
||||||
|
buffer = b""
|
||||||
|
|
||||||
|
def read_as_object(self):
|
||||||
|
"""
|
||||||
|
Read complete DSMR telegram's from remote and return a Telegram object.
|
||||||
|
|
||||||
|
:rtype: generator
|
||||||
|
"""
|
||||||
|
buffer = b""
|
||||||
|
|
||||||
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as socket_handle:
|
||||||
|
|
||||||
|
socket_handle.connect((self.host, self.port))
|
||||||
|
|
||||||
|
while True:
|
||||||
|
buffer += socket_handle.recv(self.BUFFER_SIZE)
|
||||||
|
|
||||||
|
lines = buffer.splitlines(keepends=True)
|
||||||
|
|
||||||
|
if len(lines) == 0:
|
||||||
|
continue
|
||||||
|
|
||||||
|
for data in lines:
|
||||||
|
self.telegram_buffer.append(data.decode('ascii'))
|
||||||
|
|
||||||
|
for telegram in self.telegram_buffer.get_all():
|
||||||
|
try:
|
||||||
|
yield Telegram(telegram, self.telegram_parser, self.telegram_specification)
|
||||||
|
except InvalidChecksumError as e:
|
||||||
|
logger.warning(str(e))
|
||||||
|
except ParseError as e:
|
||||||
|
logger.error('Failed to parse telegram: %s', e)
|
||||||
|
|
||||||
|
buffer = b""
|
@ -52,10 +52,6 @@ EN = {
|
|||||||
obis.VALVE_POSITION_GAS: 'VALVE_POSITION_GAS',
|
obis.VALVE_POSITION_GAS: 'VALVE_POSITION_GAS',
|
||||||
obis.BELGIUM_HOURLY_GAS_METER_READING: 'BELGIUM_HOURLY_GAS_METER_READING',
|
obis.BELGIUM_HOURLY_GAS_METER_READING: 'BELGIUM_HOURLY_GAS_METER_READING',
|
||||||
obis.LUXEMBOURG_EQUIPMENT_IDENTIFIER: 'LUXEMBOURG_EQUIPMENT_IDENTIFIER',
|
obis.LUXEMBOURG_EQUIPMENT_IDENTIFIER: 'LUXEMBOURG_EQUIPMENT_IDENTIFIER',
|
||||||
obis.LUXEMBOURG_ELECTRICITY_USED_TARIFF_GLOBAL: 'LUXEMBOURG_ELECTRICITY_USED_TARIFF_GLOBAL',
|
|
||||||
obis.LUXEMBOURG_ELECTRICITY_DELIVERED_TARIFF_GLOBAL: 'LUXEMBOURG_ELECTRICITY_DELIVERED_TARIFF_GLOBAL',
|
|
||||||
obis.SWEDEN_ELECTRICITY_USED_TARIFF_GLOBAL: 'SWEDEN_ELECTRICITY_USED_TARIFF_GLOBAL',
|
|
||||||
obis.SWEDEN_ELECTRICITY_DELIVERED_TARIFF_GLOBAL: 'SWEDEN_ELECTRICITY_DELIVERED_TARIFF_GLOBAL',
|
|
||||||
obis.Q3D_EQUIPMENT_IDENTIFIER: 'Q3D_EQUIPMENT_IDENTIFIER',
|
obis.Q3D_EQUIPMENT_IDENTIFIER: 'Q3D_EQUIPMENT_IDENTIFIER',
|
||||||
obis.Q3D_EQUIPMENT_STATE: 'Q3D_EQUIPMENT_STATE',
|
obis.Q3D_EQUIPMENT_STATE: 'Q3D_EQUIPMENT_STATE',
|
||||||
obis.Q3D_EQUIPMENT_SERIALNUMBER: 'Q3D_EQUIPMENT_SERIALNUMBER',
|
obis.Q3D_EQUIPMENT_SERIALNUMBER: 'Q3D_EQUIPMENT_SERIALNUMBER',
|
||||||
|
@ -8,10 +8,8 @@ objects are introduced.
|
|||||||
"""
|
"""
|
||||||
P1_MESSAGE_HEADER = r'\d-\d:0\.2\.8.+?\r\n'
|
P1_MESSAGE_HEADER = r'\d-\d:0\.2\.8.+?\r\n'
|
||||||
P1_MESSAGE_TIMESTAMP = r'\d-\d:1\.0\.0.+?\r\n'
|
P1_MESSAGE_TIMESTAMP = r'\d-\d:1\.0\.0.+?\r\n'
|
||||||
ELECTRICITY_IMPORTED_TOTAL = r'\d-\d:1\.8\.0.+?\r\n'
|
|
||||||
ELECTRICITY_USED_TARIFF_1 = r'\d-\d:1\.8\.1.+?\r\n'
|
ELECTRICITY_USED_TARIFF_1 = r'\d-\d:1\.8\.1.+?\r\n'
|
||||||
ELECTRICITY_USED_TARIFF_2 = r'\d-\d:1\.8\.2.+?\r\n'
|
ELECTRICITY_USED_TARIFF_2 = r'\d-\d:1\.8\.2.+?\r\n'
|
||||||
ELECTRICITY_EXPORTED_TOTAL = r'\d-\d:2\.8\.0.+?\r\n'
|
|
||||||
ELECTRICITY_DELIVERED_TARIFF_1 = r'\d-\d:2\.8\.1.+?\r\n'
|
ELECTRICITY_DELIVERED_TARIFF_1 = r'\d-\d:2\.8\.1.+?\r\n'
|
||||||
ELECTRICITY_DELIVERED_TARIFF_2 = r'\d-\d:2\.8\.2.+?\r\n'
|
ELECTRICITY_DELIVERED_TARIFF_2 = r'\d-\d:2\.8\.2.+?\r\n'
|
||||||
ELECTRICITY_ACTIVE_TARIFF = r'\d-\d:96\.14\.0.+?\r\n'
|
ELECTRICITY_ACTIVE_TARIFF = r'\d-\d:96\.14\.0.+?\r\n'
|
||||||
@ -62,13 +60,13 @@ ELECTRICITY_DELIVERED_TARIFF_ALL = (
|
|||||||
ELECTRICITY_DELIVERED_TARIFF_2
|
ELECTRICITY_DELIVERED_TARIFF_2
|
||||||
)
|
)
|
||||||
|
|
||||||
# Alternate codes for foreign countries.
|
# International generalized additions
|
||||||
|
ELECTRICITY_IMPORTED_TOTAL = r'\d-\d:1\.8\.0.+?\r\n' # Total imported energy register (P+)
|
||||||
|
ELECTRICITY_EXPORTED_TOTAL = r'\d-\d:2\.8\.0.+?\r\n' # Total exported energy register (P-)
|
||||||
|
|
||||||
|
# International non generalized additions (country specific) / risk for necessary refactoring
|
||||||
BELGIUM_HOURLY_GAS_METER_READING = r'\d-\d:24\.2\.3.+?\r\n' # Different code, same format.
|
BELGIUM_HOURLY_GAS_METER_READING = r'\d-\d:24\.2\.3.+?\r\n' # Different code, same format.
|
||||||
LUXEMBOURG_EQUIPMENT_IDENTIFIER = r'\d-\d:42\.0\.0.+?\r\n' # Logical device name
|
LUXEMBOURG_EQUIPMENT_IDENTIFIER = r'\d-\d:42\.0\.0.+?\r\n' # Logical device name
|
||||||
LUXEMBOURG_ELECTRICITY_USED_TARIFF_GLOBAL = r'\d-\d:1\.8\.0.+?\r\n' # Total imported energy register (P+)
|
|
||||||
LUXEMBOURG_ELECTRICITY_DELIVERED_TARIFF_GLOBAL = r'\d-\d:2\.8\.0.+?\r\n' # Total exported energy register (P-)
|
|
||||||
SWEDEN_ELECTRICITY_USED_TARIFF_GLOBAL = r'\d-\d:1\.8\.0.+?\r\n' # Total imported energy register (P+)
|
|
||||||
SWEDEN_ELECTRICITY_DELIVERED_TARIFF_GLOBAL = r'\d-\d:2\.8\.0.+?\r\n' # Total exported energy register (P-)
|
|
||||||
Q3D_EQUIPMENT_IDENTIFIER = r'\d-\d:0\.0\.0.+?\r\n' # Logical device name
|
Q3D_EQUIPMENT_IDENTIFIER = r'\d-\d:0\.0\.0.+?\r\n' # Logical device name
|
||||||
Q3D_EQUIPMENT_STATE = r'\d-\d:96\.5\.5.+?\r\n' # Device state (hexadecimal)
|
Q3D_EQUIPMENT_STATE = r'\d-\d:96\.5\.5.+?\r\n' # Device state (hexadecimal)
|
||||||
Q3D_EQUIPMENT_SERIALNUMBER = r'\d-\d:96\.1\.255.+?\r\n' # Device Serialnumber
|
Q3D_EQUIPMENT_SERIALNUMBER = r'\d-\d:96\.1\.255.+?\r\n' # Device Serialnumber
|
||||||
|
@ -179,7 +179,7 @@ class ProfileGenericObject(DSMRObject):
|
|||||||
self._buffer_list = []
|
self._buffer_list = []
|
||||||
values_offset = 2
|
values_offset = 2
|
||||||
for i in range(self.buffer_length):
|
for i in range(self.buffer_length):
|
||||||
offset = values_offset + i*2
|
offset = values_offset + i * 2
|
||||||
self._buffer_list.append(MBusObject([self.values[offset], self.values[offset + 1]]))
|
self._buffer_list.append(MBusObject([self.values[offset], self.values[offset + 1]]))
|
||||||
return self._buffer_list
|
return self._buffer_list
|
||||||
|
|
||||||
|
@ -7,4 +7,4 @@ PG_HEAD_PARSERS = [ValueParser(int), ValueParser(str)]
|
|||||||
PG_UNIDENTIFIED_BUFFERTYPE_PARSERS = [ValueParser(str), ValueParser(str)]
|
PG_UNIDENTIFIED_BUFFERTYPE_PARSERS = [ValueParser(str), ValueParser(str)]
|
||||||
BUFFER_TYPES = {
|
BUFFER_TYPES = {
|
||||||
PG_FAILURE_EVENT: [ValueParser(timestamp), ValueParser(int)]
|
PG_FAILURE_EVENT: [ValueParser(timestamp), ValueParser(int)]
|
||||||
}
|
}
|
||||||
|
@ -153,8 +153,8 @@ BELGIUM_FLUVIUS['objects'].update({
|
|||||||
LUXEMBOURG_SMARTY = deepcopy(V5)
|
LUXEMBOURG_SMARTY = deepcopy(V5)
|
||||||
LUXEMBOURG_SMARTY['objects'].update({
|
LUXEMBOURG_SMARTY['objects'].update({
|
||||||
obis.LUXEMBOURG_EQUIPMENT_IDENTIFIER: CosemParser(ValueParser(str)),
|
obis.LUXEMBOURG_EQUIPMENT_IDENTIFIER: CosemParser(ValueParser(str)),
|
||||||
obis.LUXEMBOURG_ELECTRICITY_USED_TARIFF_GLOBAL: CosemParser(ValueParser(Decimal)),
|
obis.ELECTRICITY_IMPORTED_TOTAL: CosemParser(ValueParser(Decimal)),
|
||||||
obis.LUXEMBOURG_ELECTRICITY_DELIVERED_TARIFF_GLOBAL: CosemParser(ValueParser(Decimal)),
|
obis.ELECTRICITY_EXPORTED_TOTAL: CosemParser(ValueParser(Decimal)),
|
||||||
})
|
})
|
||||||
|
|
||||||
# Source: https://www.energiforetagen.se/globalassets/energiforetagen/det-erbjuder-vi/kurser-och-konferenser/elnat/
|
# Source: https://www.energiforetagen.se/globalassets/energiforetagen/det-erbjuder-vi/kurser-och-konferenser/elnat/
|
||||||
@ -164,8 +164,8 @@ SWEDEN = {
|
|||||||
'objects': {
|
'objects': {
|
||||||
obis.P1_MESSAGE_HEADER: CosemParser(ValueParser(str)),
|
obis.P1_MESSAGE_HEADER: CosemParser(ValueParser(str)),
|
||||||
obis.P1_MESSAGE_TIMESTAMP: CosemParser(ValueParser(timestamp)),
|
obis.P1_MESSAGE_TIMESTAMP: CosemParser(ValueParser(timestamp)),
|
||||||
obis.SWEDEN_ELECTRICITY_USED_TARIFF_GLOBAL: CosemParser(ValueParser(Decimal)),
|
obis.ELECTRICITY_IMPORTED_TOTAL: CosemParser(ValueParser(Decimal)),
|
||||||
obis.SWEDEN_ELECTRICITY_DELIVERED_TARIFF_GLOBAL: CosemParser(ValueParser(Decimal)),
|
obis.ELECTRICITY_EXPORTED_TOTAL: CosemParser(ValueParser(Decimal)),
|
||||||
obis.CURRENT_ELECTRICITY_USAGE: CosemParser(ValueParser(Decimal)),
|
obis.CURRENT_ELECTRICITY_USAGE: CosemParser(ValueParser(Decimal)),
|
||||||
obis.CURRENT_ELECTRICITY_DELIVERY: CosemParser(ValueParser(Decimal)),
|
obis.CURRENT_ELECTRICITY_DELIVERY: CosemParser(ValueParser(Decimal)),
|
||||||
obis.INSTANTANEOUS_ACTIVE_POWER_L1_POSITIVE: CosemParser(ValueParser(Decimal)),
|
obis.INSTANTANEOUS_ACTIVE_POWER_L1_POSITIVE: CosemParser(ValueParser(Decimal)),
|
||||||
|
@ -136,7 +136,7 @@ TELEGRAM_V5 = (
|
|||||||
#
|
#
|
||||||
# last two lines are added by the COM-1 Ethernet Gateway
|
# last two lines are added by the COM-1 Ethernet Gateway
|
||||||
|
|
||||||
TELEGRAM_ESY5Q3DB1024_V304 = ( # Easymeter an Hauptstromzähler
|
TELEGRAM_ESY5Q3DB1024_V304 = (
|
||||||
'/ESY5Q3DB1024 V3.04\r\n'
|
'/ESY5Q3DB1024 V3.04\r\n'
|
||||||
'\r\n'
|
'\r\n'
|
||||||
'1-0:0.0.0*255(0272031312565)\r\n'
|
'1-0:0.0.0*255(0272031312565)\r\n'
|
||||||
@ -150,10 +150,11 @@ TELEGRAM_ESY5Q3DB1024_V304 = ( # Easymeter an Hauptstromzähler
|
|||||||
'0-0:96.1.255*255(1ESY1313002565)\r\n'
|
'0-0:96.1.255*255(1ESY1313002565)\r\n'
|
||||||
'!\r\n'
|
'!\r\n'
|
||||||
' 25803103\r\n'
|
' 25803103\r\n'
|
||||||
'\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\r\n'
|
'\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff'
|
||||||
|
'\xff\xff\xff\xff\xff\r\n'
|
||||||
)
|
)
|
||||||
|
|
||||||
TELEGRAM_ESY5Q3DA1004_V304 = ( # Easymeter an Wärmepumpe
|
TELEGRAM_ESY5Q3DA1004_V304 = (
|
||||||
'/ESY5Q3DA1004 V3.04\r\n'
|
'/ESY5Q3DA1004 V3.04\r\n'
|
||||||
'\r\n'
|
'\r\n'
|
||||||
'1-0:0.0.0*255(1336001560)\r\n'
|
'1-0:0.0.0*255(1336001560)\r\n'
|
||||||
|
77
test/test_rfxtrx_protocol.py
Normal file
77
test/test_rfxtrx_protocol.py
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
from unittest.mock import Mock
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from dsmr_parser import obis_references as obis
|
||||||
|
from dsmr_parser.clients.rfxtrx_protocol import create_rfxtrx_dsmr_protocol, PACKETTYPE_DSMR, SUBTYPE_P1
|
||||||
|
|
||||||
|
|
||||||
|
TELEGRAM_V2_2 = (
|
||||||
|
'/ISk5\2MT382-1004\r\n'
|
||||||
|
'\r\n'
|
||||||
|
'0-0:96.1.1(00000000000000)\r\n'
|
||||||
|
'1-0:1.8.1(00001.001*kWh)\r\n'
|
||||||
|
'1-0:1.8.2(00001.001*kWh)\r\n'
|
||||||
|
'1-0:2.8.1(00001.001*kWh)\r\n'
|
||||||
|
'1-0:2.8.2(00001.001*kWh)\r\n'
|
||||||
|
'0-0:96.14.0(0001)\r\n'
|
||||||
|
'1-0:1.7.0(0001.01*kW)\r\n'
|
||||||
|
'1-0:2.7.0(0000.00*kW)\r\n'
|
||||||
|
'0-0:17.0.0(0999.00*kW)\r\n'
|
||||||
|
'0-0:96.3.10(1)\r\n'
|
||||||
|
'0-0:96.13.1()\r\n'
|
||||||
|
'0-0:96.13.0()\r\n'
|
||||||
|
'0-1:24.1.0(3)\r\n'
|
||||||
|
'0-1:96.1.0(000000000000)\r\n'
|
||||||
|
'0-1:24.3.0(161107190000)(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'
|
||||||
|
)
|
||||||
|
|
||||||
|
OTHER_RF_PACKET = b'\x03\x01\x02\x03'
|
||||||
|
|
||||||
|
|
||||||
|
def encode_telegram_as_RF_packets(telegram):
|
||||||
|
data = b''
|
||||||
|
|
||||||
|
for line in telegram.split('\n'):
|
||||||
|
packet_data = (line + '\n').encode('ascii')
|
||||||
|
packet_header = bytes(bytearray([
|
||||||
|
len(packet_data) + 3, # excluding length byte
|
||||||
|
PACKETTYPE_DSMR,
|
||||||
|
SUBTYPE_P1,
|
||||||
|
0 # seq num (ignored)
|
||||||
|
]))
|
||||||
|
|
||||||
|
data += packet_header + packet_data
|
||||||
|
# other RF packets can pass by on the line
|
||||||
|
data += OTHER_RF_PACKET
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
class RFXtrxProtocolTest(unittest.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
new_protocol, _ = create_rfxtrx_dsmr_protocol('2.2',
|
||||||
|
telegram_callback=Mock(),
|
||||||
|
keep_alive_interval=1)
|
||||||
|
self.protocol = new_protocol()
|
||||||
|
|
||||||
|
def test_complete_packet(self):
|
||||||
|
"""Protocol should assemble incoming lines into complete packet."""
|
||||||
|
|
||||||
|
data = encode_telegram_as_RF_packets(TELEGRAM_V2_2)
|
||||||
|
# send data broken up in two parts
|
||||||
|
self.protocol.data_received(data[0:200])
|
||||||
|
self.protocol.data_received(data[200:])
|
||||||
|
|
||||||
|
telegram = self.protocol.telegram_callback.call_args_list[0][0][0]
|
||||||
|
assert isinstance(telegram, dict)
|
||||||
|
|
||||||
|
assert float(telegram[obis.CURRENT_ELECTRICITY_USAGE].value) == 1.01
|
||||||
|
assert telegram[obis.CURRENT_ELECTRICITY_USAGE].unit == 'kW'
|
||||||
|
|
||||||
|
assert float(telegram[obis.GAS_METER_READING].value) == 1.001
|
||||||
|
assert telegram[obis.GAS_METER_READING].unit == 'm3'
|
Loading…
Reference in New Issue
Block a user