1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
|
import sys
import struct
import logging
from logging import StreamHandler, Formatter
from functools import wraps
from umodbus import log
def log_to_stream(stream=sys.stderr, level=logging.NOTSET,
fmt=logging.BASIC_FORMAT):
""" Add :class:`logging.StreamHandler` to logger which logs to a stream.
:param stream. Stream to log to, default STDERR.
:param level: Log level, default NOTSET.
:param fmt: String with log format, default is BASIC_FORMAT.
"""
fmt = Formatter(fmt)
handler = StreamHandler()
handler.setFormatter(fmt)
handler.setLevel(level)
log.addHandler(handler)
def unpack_mbap(mbap):
""" Parse MBAP of 7 bytes and return tuple with fields.
>>> parse_mbap(b'\x00\x08\x00\x00\x00\x06\x01')
(8, 0, 6, 1)
:param mbap: Array of 7 bytes.
:return: Tuple with 4 values: Transaction identifier, Protocol identifier,
Length and Unit identifier.
"""
# '>' indicates data is big-endian. Modbus uses this alignment. 'H' and 'B'
# are format characters. 'H' is unsigned short of 2 bytes. 'B' is an
# unsigned char of 1 byte. HHHB sums up to 2 + 2 + 2 + 1 = 7 bytes.
# TODO What it right exception to raise? Error code 04, Server failure,
# seems most appropriate.
return struct.unpack('>HHHB', mbap)
def pack_mbap(transaction_id, protocol_id, length, unit_id):
""" Create and return response MBAP.
:param transaction_id: Transaction id.
:param protocol_id: Protocol id.
:param length: Length of following bytes in ADU.
:param unit_id: Unit id.
:return: Byte array of 7 bytes.
"""
return struct.pack('>HHHB', transaction_id, protocol_id, length, unit_id)
def pack_exception_pdu(function_code, error_code):
""" Return exception PDU of 2 bytes.
"The exception response message has two fields that differentiate it
from a nor mal response: Function Code Field: In a normal response, the
server echoes the function code of the original request in the function
code field of the response. All function codes have a most -
significant bit (MSB) of 0 (their values are all below 80 hexadecimal).
In an exception response, the server sets the MSB of the function code
to 1. This makes the function code value in an exception response
exactly 80 hexadecimal higher than the value would be for a normal
response.
With the function code's MSB set, the client's application program can
recognize the exception response and can examine the data field for the
exception code. Data Field: In a normal response, the server may
return data or statistics in the data field (any information that was
requested in the request). In an exception response, the server returns
an exception code in the data field. This defines the server condition
that caused the exception."
-- MODBUS Application Protocol Specification V1.1b3, chapter 7
================ ===============
Field Length (bytes)
================ ===============
Error code 1
Function code 1
================ ===============
:param error_code: Error code.
:param function_code: Function code.
:return: PDU of 2 bytes.
"""
return struct.pack('>BB', function_code + 0x80, error_code)
def get_function_code_from_request_pdu(pdu):
""" Return function code from request PDU.
:return pdu: Array with bytes.
:return: Function code.
"""
return struct.unpack('>B', pdu[:1])[0]
def memoize(f):
""" Decorator which caches function's return value each it is called.
If called later with same arguments, the cached value is returned.
"""
cache = {}
@wraps(f)
def inner(arg):
if arg not in cache:
cache[arg] = f(arg)
return cache[arg]
return inner
def recv_exactly(recv_fn, size):
""" Use the function to read and return exactly number of bytes desired.
https://docs.python.org/3/howto/sockets.html#socket-programming-howto for
more information about why this is necessary.
:param recv_fn: Function that can return up to given bytes
(i.e. socket.recv, file.read)
:param size: Number of bytes to read.
:return: Byte string with length size.
:raises ValueError: Could not receive enough data (usually timeout).
"""
recv_bytes = 0
chunks = []
while recv_bytes < size:
chunk = recv_fn(size - recv_bytes)
if len(chunk) == 0: # when closed or empty
break
recv_bytes += len(chunk)
chunks.append(chunk)
response = b''.join(chunks)
if len(response) != size:
raise ValueError
return response
|