From 61163b33d6f3baa368702c34c7bcfa4819e42ad0 Mon Sep 17 00:00:00 2001 From: dhoomakethu Date: Mon, 16 Nov 2015 10:56:01 -0800 Subject: [PATCH] pymodbus fix for modes RTU error cases. --- pymodbus/transaction.py | 49 ++++++++++++++++++++++++++++++++--------- 1 file changed, 39 insertions(+), 10 deletions(-) diff --git a/pymodbus/transaction.py b/pymodbus/transaction.py index 2871737ec..71cc79b3d 100644 --- a/pymodbus/transaction.py +++ b/pymodbus/transaction.py @@ -473,7 +473,11 @@ def advanceFrame(self): it or determined that it contains an error. It also has to reset the current frame header handle ''' - self.__buffer = self.__buffer[self.__header['len']:] + try: + self.__buffer = self.__buffer[self.__header['len']:] + except KeyError: + # Error response, no header len found + self.resetFrame() self.__header = {} def resetFrame(self): @@ -562,15 +566,21 @@ def processIncomingPacket(self, data, callback): :param callback: The function to send results to ''' self.addToFrame(data) - while self.isFrameReady(): - if self.checkFrame(): - result = self.decoder.decode(self.getFrame()) - if result is None: - raise ModbusIOException("Unable to decode response") - self.populateResult(result) - self.advanceFrame() - callback(result) # defer or push to a thread? - else: self.resetFrame() # clear possible errors + while True: + if self.isFrameReady(): + if self.checkFrame(): + self._process(callback) + else: + # Could be an error response + if len(self.__buffer): + # Possible error ??? + self._process(callback, error=True) + else: + if len(self.__buffer): + # Possible error ??? + if self.__header['len'] < 2: + self._process(callback, error=True) + break def buildPacket(self, message): ''' Creates a ready to send modbus packet @@ -584,6 +594,25 @@ def buildPacket(self, message): packet += struct.pack(">H", computeCRC(packet)) return packet + def _process(self, callback, error=False): + """ + Process incoming packets irrespective error condition + """ + data = self.getRawFrame() if error else self.getFrame() + result = self.decoder.decode(data) + if result is None: + raise ModbusIOException("Unable to decode request") + self.populateResult(result) + self.advanceFrame() + callback(result) # defer or push to a thread? + + def getRawFrame(self): + """ + Returns the complete buffer + """ + return self.__buffer + + #---------------------------------------------------------------------------# # Modbus ASCII Message