A clone of: https://github.com/nutechsoftware/alarmdecoder This is requires as they dropped support for older firmware releases w/o building in backward compatibility code, and they had previously hardcoded pyserial to a python2 only version.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

254 lines
7.1 KiB

  1. import usb.core
  2. import usb.util
  3. import time
  4. import threading
  5. import serial
  6. import serial.tools.list_ports
  7. import traceback
  8. from pyftdi.pyftdi.ftdi import *
  9. from pyftdi.pyftdi.usbtools import *
  10. from . import util
  11. from .event import event
  12. class Device(object):
  13. on_open = event.Event('Called when the device has been opened')
  14. on_close = event.Event('Called when the device has been closed')
  15. on_read = event.Event('Called when a line has been read from the device')
  16. on_write = event.Event('Called when data has been written to the device')
  17. def __init__(self):
  18. pass
  19. def __del__(self):
  20. pass
  21. class ReadThread(threading.Thread):
  22. def __init__(self, device):
  23. threading.Thread.__init__(self)
  24. self._device = device
  25. self._running = False
  26. def stop(self):
  27. self._running = False
  28. def run(self):
  29. self._running = True
  30. while self._running:
  31. try:
  32. self._device.read_line()
  33. except util.CommError, err:
  34. self.stop()
  35. time.sleep(0.01)
  36. class USBDevice(Device):
  37. FTDI_VENDOR_ID = 0x0403
  38. FTDI_PRODUCT_ID = 0x6001
  39. BAUDRATE = 115200
  40. @staticmethod
  41. def find_all():
  42. devices = []
  43. try:
  44. devices = Ftdi.find_all([(USBDevice.FTDI_VENDOR_ID, USBDevice.FTDI_PRODUCT_ID)], nocache=True)
  45. except (usb.core.USBError, FtdiError), err:
  46. raise util.CommError('Error enumerating AD2USB devices: {0}'.format(str(err)))
  47. return devices
  48. def __init__(self, vid=FTDI_VENDOR_ID, pid=FTDI_PRODUCT_ID, serial=None, description=None):
  49. Device.__init__(self)
  50. self._vendor_id = vid
  51. self._product_id = pid
  52. self._serial_number = serial
  53. self._description = description
  54. self._buffer = ''
  55. self._device = Ftdi()
  56. self._running = False
  57. self._read_thread = Device.ReadThread(self)
  58. def open(self, baudrate=BAUDRATE, interface=0, index=0):
  59. self._running = True
  60. try:
  61. self._device.open(self._vendor_id,
  62. self._product_id,
  63. interface,
  64. index,
  65. self._serial_number,
  66. self._description)
  67. self._device.set_baudrate(baudrate)
  68. except (usb.core.USBError, FtdiError), err:
  69. self.on_close()
  70. raise util.CommError('Error opening AD2USB device: {0}'.format(str(err)))
  71. else:
  72. self._read_thread.start()
  73. self.on_open((self._serial_number, self._description))
  74. def close(self):
  75. try:
  76. self._running = False
  77. self._read_thread.stop()
  78. self._device.close()
  79. except (FtdiError, usb.core.USBError):
  80. pass
  81. self.on_close()
  82. def write(self, data):
  83. self._device.write_data(data)
  84. self.on_write(data)
  85. def read_line(self, timeout=0.0):
  86. start_time = time.time()
  87. got_line = False
  88. ret = None
  89. try:
  90. while self._running:
  91. buf = self._device.read_data(1)
  92. if buf != '':
  93. self._buffer += buf
  94. if buf == "\n":
  95. if len(self._buffer) > 1:
  96. if self._buffer[-2] == "\r":
  97. self._buffer = self._buffer[:-2]
  98. # ignore if we just got \r\n with nothing else in the buffer.
  99. if len(self._buffer) != 0:
  100. got_line = True
  101. break
  102. else:
  103. self._buffer = self._buffer[:-1]
  104. if timeout > 0 and time.time() - start_time > timeout:
  105. break
  106. time.sleep(0.01)
  107. except (usb.core.USBError, FtdiError), err:
  108. self.close()
  109. raise util.CommError('Error reading from AD2USB device: {0}'.format(str(err)))
  110. else:
  111. if got_line:
  112. ret = self._buffer
  113. self._buffer = ''
  114. self.on_read(ret)
  115. return ret
  116. class SerialDevice(Device):
  117. BAUDRATE = 19200
  118. @staticmethod
  119. def find_all():
  120. devices = []
  121. try:
  122. devices = serial.tools.list_ports.comports()
  123. except Exception, err:
  124. raise util.CommError('Error enumerating AD2SERIAL devices: {0}'.format(str(err)))
  125. return devices
  126. def __init__(self):
  127. Device.__init__(self)
  128. self._device = serial.Serial(timeout=0) # Timeout = non-blocking to match pyftdi.
  129. self._read_thread = Device.ReadThread(self)
  130. self._buffer = ''
  131. self._running = False
  132. def __del__(self):
  133. pass
  134. def open(self, baudrate=BAUDRATE, interface=0, index=0):
  135. self._device.baudrate = baudrate
  136. self._device.port = interface
  137. try:
  138. self._device.open()
  139. self._running = True
  140. except (serial.SerialException, ValueError), err:
  141. self.on_close()
  142. raise util.NoDeviceError('Error opening AD2SERIAL device on port {0}.'.format(interface))
  143. else:
  144. self.on_open((None, "AD2SERIAL")) # TODO: Fixme.
  145. self._read_thread.start()
  146. def close(self):
  147. try:
  148. self._running = False
  149. self._read_thread.stop()
  150. self._device.close()
  151. except Exception, err:
  152. pass
  153. self.on_close()
  154. def write(self, data):
  155. try:
  156. self._device.write(data)
  157. except serial.SerialTimeoutException, err:
  158. pass
  159. else:
  160. self.on_write(data)
  161. def read_line(self, timeout=0.0):
  162. start_time = time.time()
  163. got_line = False
  164. ret = None
  165. try:
  166. while self._running:
  167. buf = self._device.read(1)
  168. if buf != '' and buf != "\xff": # WTF is this \xff and why is it in my buffer?!
  169. self._buffer += buf
  170. if buf == "\n":
  171. if len(self._buffer) > 1:
  172. if self._buffer[-2] == "\r":
  173. self._buffer = self._buffer[:-2]
  174. # ignore if we just got \r\n with nothing else in the buffer.
  175. if len(self._buffer) != 0:
  176. got_line = True
  177. break
  178. else:
  179. self._buffer = self._buffer[:-1]
  180. if timeout > 0 and time.time() - start_time > timeout:
  181. break
  182. time.sleep(0.01)
  183. except serial.SerialException, err:
  184. self.close()
  185. raise util.CommError('Error reading from AD2SERIAL device: {0}'.format(str(err)))
  186. else:
  187. if got_line:
  188. ret = self._buffer
  189. self._buffer = ''
  190. self.on_read(ret)
  191. return ret