add Modbus retrasmissions
This commit is contained in:
@@ -36,7 +36,7 @@ class Control:
|
|||||||
|
|
||||||
class Talent(Message):
|
class Talent(Message):
|
||||||
def __init__(self, server_side: bool, id_str=b''):
|
def __init__(self, server_side: bool, id_str=b''):
|
||||||
super().__init__(server_side, self.send_modbus_cb, 15)
|
super().__init__(server_side, self.send_modbus_cb, 11)
|
||||||
self.await_conn_resp_cnt = 0
|
self.await_conn_resp_cnt = 0
|
||||||
self.id_str = id_str
|
self.id_str = id_str
|
||||||
self.contact_name = b''
|
self.contact_name = b''
|
||||||
@@ -122,13 +122,21 @@ class Talent(Message):
|
|||||||
f' Ctl: {int(self.ctrl):#02x} Msg: {fnc.__name__!r}')
|
f' Ctl: {int(self.ctrl):#02x} Msg: {fnc.__name__!r}')
|
||||||
return
|
return
|
||||||
|
|
||||||
def send_modbus_cb(self, modbus_pdu: bytearray):
|
def send_modbus_cb(self, modbus_pdu: bytearray, retrans: bool):
|
||||||
|
if self.state != self.STATE_UP:
|
||||||
|
return
|
||||||
|
|
||||||
self.__build_header(0x70, 0x77)
|
self.__build_header(0x70, 0x77)
|
||||||
self._send_buffer += b'\x00\x01\xa3\x28' # fixme
|
self._send_buffer += b'\x00\x01\xa3\x28' # fixme
|
||||||
self._send_buffer += struct.pack('!B', len(modbus_pdu))
|
self._send_buffer += struct.pack('!B', len(modbus_pdu))
|
||||||
self._send_buffer += modbus_pdu
|
self._send_buffer += modbus_pdu
|
||||||
self.__finish_send_msg()
|
self.__finish_send_msg()
|
||||||
hex_dump_memory(logging.INFO, f'Send Modbus Command:{self.addr}:',
|
if retrans:
|
||||||
|
cmd = 'Retrans'
|
||||||
|
else:
|
||||||
|
cmd = 'Command'
|
||||||
|
|
||||||
|
hex_dump_memory(logging.INFO, f'Send Modbus {cmd}:{self.addr}:',
|
||||||
self._send_buffer, len(self._send_buffer))
|
self._send_buffer, len(self._send_buffer))
|
||||||
self.writer.write(self._send_buffer)
|
self.writer.write(self._send_buffer)
|
||||||
self._send_buffer = bytearray(0) # self._send_buffer[sent:]
|
self._send_buffer = bytearray(0) # self._send_buffer[sent:]
|
||||||
@@ -392,11 +400,14 @@ class Talent(Message):
|
|||||||
self.header_len+self.data_len]
|
self.header_len+self.data_len]
|
||||||
|
|
||||||
if self.ctrl.is_req():
|
if self.ctrl.is_req():
|
||||||
|
# if (self.remoteStream.state != self.STATE_UP):
|
||||||
|
# logger.info('ignore Modbus Request in wrong state')
|
||||||
|
# return
|
||||||
if self.remoteStream.mb.recv_req(data[hdr_len:],
|
if self.remoteStream.mb.recv_req(data[hdr_len:],
|
||||||
self.msg_forward):
|
self.msg_forward):
|
||||||
self.remoteStream.inc_counter('Modbus_Command')
|
self.inc_counter('Modbus_Command')
|
||||||
else:
|
else:
|
||||||
self.remoteStream.inc_counter('Invalid_Msg_Format')
|
self.inc_counter('Invalid_Msg_Format')
|
||||||
elif self.ctrl.is_ind():
|
elif self.ctrl.is_ind():
|
||||||
# logger.debug(f'Modbus Ind MsgLen: {modbus_len}')
|
# logger.debug(f'Modbus Ind MsgLen: {modbus_len}')
|
||||||
self.modbus_elms = 0
|
self.modbus_elms = 0
|
||||||
|
|||||||
@@ -301,13 +301,19 @@ class SolarmanV5(Message):
|
|||||||
self._heartbeat())
|
self._heartbeat())
|
||||||
self.__finish_send_msg()
|
self.__finish_send_msg()
|
||||||
|
|
||||||
def send_modbus_cb(self, pdu: bytearray):
|
def send_modbus_cb(self, pdu: bytearray, retrans: bool):
|
||||||
|
if self.state != self.STATE_UP:
|
||||||
|
return
|
||||||
self.__build_header(0x4510)
|
self.__build_header(0x4510)
|
||||||
self._send_buffer += struct.pack('<BHLLL', self.MB_RTU_CMD,
|
self._send_buffer += struct.pack('<BHLLL', self.MB_RTU_CMD,
|
||||||
0x2b0, 0, 0, 0)
|
0x2b0, 0, 0, 0)
|
||||||
self._send_buffer += pdu
|
self._send_buffer += pdu
|
||||||
self.__finish_send_msg()
|
self.__finish_send_msg()
|
||||||
hex_dump_memory(logging.INFO, f'Send Modbus Command:{self.addr}:',
|
if retrans:
|
||||||
|
cmd = 'Retrans'
|
||||||
|
else:
|
||||||
|
cmd = 'Command'
|
||||||
|
hex_dump_memory(logging.INFO, f'Send Modbus {cmd}:{self.addr}:',
|
||||||
self._send_buffer, len(self._send_buffer))
|
self._send_buffer, len(self._send_buffer))
|
||||||
self.writer.write(self._send_buffer)
|
self.writer.write(self._send_buffer)
|
||||||
self._send_buffer = bytearray(0) # self._send_buffer[sent:]
|
self._send_buffer = bytearray(0) # self._send_buffer[sent:]
|
||||||
|
|||||||
@@ -73,6 +73,14 @@ class Modbus():
|
|||||||
self.snd_handler = snd_handler
|
self.snd_handler = snd_handler
|
||||||
self.rsp_handler = None
|
self.rsp_handler = None
|
||||||
self.timeout = timeout
|
self.timeout = timeout
|
||||||
|
self.max_retries = 3
|
||||||
|
self.retry_cnt = 0
|
||||||
|
self.last_req = b''
|
||||||
|
self.counter = {}
|
||||||
|
self.counter['timeouts'] = 0
|
||||||
|
self.counter['retries'] = {}
|
||||||
|
for i in range(0, self.max_retries):
|
||||||
|
self.counter['retries'][i] = 0
|
||||||
self.last_addr = 0
|
self.last_addr = 0
|
||||||
self.last_fcode = 0
|
self.last_fcode = 0
|
||||||
self.last_len = 0
|
self.last_len = 0
|
||||||
@@ -82,6 +90,10 @@ class Modbus():
|
|||||||
self.req_pend = False
|
self.req_pend = False
|
||||||
self.tim = None
|
self.tim = None
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
if type(self.counter) is not None:
|
||||||
|
logging.info(f'Modbus __del__:\n {self.counter}')
|
||||||
|
|
||||||
def start_timer(self):
|
def start_timer(self):
|
||||||
if self.req_pend:
|
if self.req_pend:
|
||||||
return
|
return
|
||||||
@@ -97,8 +109,16 @@ class Modbus():
|
|||||||
|
|
||||||
def timeout_cb(self):
|
def timeout_cb(self):
|
||||||
self.req_pend = False
|
self.req_pend = False
|
||||||
logging.info(f'Modbus timeout {self}')
|
|
||||||
self.get_next_req()
|
if self.retry_cnt < self.max_retries:
|
||||||
|
logging.debug(f'Modbus retrans {self}')
|
||||||
|
self.retry_cnt += 1
|
||||||
|
self.start_timer()
|
||||||
|
self.snd_handler(self.last_req, retrans=True)
|
||||||
|
else:
|
||||||
|
logging.info(f'Modbus timeout {self}')
|
||||||
|
self.counter['timeouts'] += 1
|
||||||
|
self.get_next_req()
|
||||||
|
|
||||||
def get_next_req(self) -> None:
|
def get_next_req(self) -> None:
|
||||||
if self.req_pend:
|
if self.req_pend:
|
||||||
@@ -106,6 +126,7 @@ class Modbus():
|
|||||||
try:
|
try:
|
||||||
item = self.que.get_nowait()
|
item = self.que.get_nowait()
|
||||||
req = item['req']
|
req = item['req']
|
||||||
|
self.last_req = req
|
||||||
self.rsp_handler = item['rsp_hdl']
|
self.rsp_handler = item['rsp_hdl']
|
||||||
self.last_addr = req[0]
|
self.last_addr = req[0]
|
||||||
self.last_fcode = req[1]
|
self.last_fcode = req[1]
|
||||||
@@ -113,8 +134,9 @@ class Modbus():
|
|||||||
res = struct.unpack_from('>HH', req, 2)
|
res = struct.unpack_from('>HH', req, 2)
|
||||||
self.last_reg = res[0]
|
self.last_reg = res[0]
|
||||||
self.last_len = res[1]
|
self.last_len = res[1]
|
||||||
|
self.retry_cnt = 0
|
||||||
self.start_timer()
|
self.start_timer()
|
||||||
self.snd_handler(req)
|
self.snd_handler(self.last_req, retrans=False)
|
||||||
except asyncio.QueueEmpty:
|
except asyncio.QueueEmpty:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -140,7 +162,7 @@ class Modbus():
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
def recv_resp(self, info_db, buf: bytearray, node_id: str) -> \
|
def recv_resp(self, info_db, buf: bytearray, node_id: str) -> \
|
||||||
Generator[tuple[str, bool], None, None]:
|
Generator[tuple[str, bool, any], None, None]:
|
||||||
# logging.info(f'recv_resp: first byte modbus:{buf[0]} len:{len(buf)}')
|
# logging.info(f'recv_resp: first byte modbus:{buf[0]} len:{len(buf)}')
|
||||||
if not self.req_pend:
|
if not self.req_pend:
|
||||||
self.err = 5
|
self.err = 5
|
||||||
@@ -194,7 +216,7 @@ class Modbus():
|
|||||||
f' : {result}{unit}')
|
f' : {result}{unit}')
|
||||||
else:
|
else:
|
||||||
self.stop_timer()
|
self.stop_timer()
|
||||||
|
self.counter['retries'][self.retry_cnt] += 1
|
||||||
if self.rsp_handler:
|
if self.rsp_handler:
|
||||||
self.rsp_handler()
|
self.rsp_handler()
|
||||||
self.get_next_req()
|
self.get_next_req()
|
||||||
|
|||||||
Reference in New Issue
Block a user