wsc_device_encoder.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. import logging
  2. import sys
  3. import time
  4. import sip
  5. from wsc_tools import ParsingTools
  6. from wsc_tools import Switch
  7. from PyQt5 import QtCore
  8. from PyQt5 import QtGui
  9. from PyQt5 import QtWidgets
  10. from pyqt.ideasxdevice import Ui_IdeasXDevice
  11. from pyqt.encoderconfigurationdialog import Ui_SwitchConfigDialog
  12. from pyqt.devicedialog import Ui_Dialog
  13. pt = ParsingTools()
  14. logging.basicConfig( level=logging.DEBUG)
  15. log = logging.getLogger("wsc_device_encoder")
  16. class Encoder():
  17. FAIL = 0
  18. SUCCESS = 1
  19. ALIVE_VALUE = b'1'
  20. ALIVE_TOPIC = "alive"
  21. DEVICE_HEALTH_TOPIC = "encoder/+/health/#"
  22. DEVICE_HEALTH_QOS = 0
  23. SHUTDOWN_COMMAND_TOPIC = "shutdown"
  24. RESTART_COMMAND_TOPIC = "restart"
  25. OTA_COMMAND_TOPIC = "ota"
  26. UART_TX_COMMAND_TOPIC = "uart"
  27. COMMAND_RETAIN = False
  28. COMMAND_QOS = 1
  29. DATA_BUTTON_QOS = 0
  30. def __init__(self, device_id, mqttc):
  31. self.__device_id = device_id
  32. self.__label_default = b""
  33. self.__hw_version_default = b"0,0"
  34. self.__fw_version_default = b"0,0"
  35. self.__alive_default = b"0"
  36. self.__vcell_default = b"0"
  37. self.__charge_default = b"0"
  38. self.__lbi_default = b"0"
  39. self.__soc_default = b"0"
  40. self.__rom_default = b"0"
  41. self.__ota_default = b"0"
  42. self.__wireless_default = b"0"
  43. self.__ssid_default = b""
  44. self.__bssid_default = b""
  45. self.__rssi_default = b"0"
  46. self.__auth_default = b"0"
  47. self.__time_default = time.time()
  48. self.__fields = {"device_id": self.__device_id,
  49. "label": self.__label_default,
  50. "hw_ver": self.__hw_version_default,
  51. "fw_ver": self.__fw_version_default,
  52. "alive": self.__alive_default,
  53. "vcell": self.__vcell_default,
  54. "charge": self.__charge_default,
  55. "lbi": self.__lbi_default,
  56. "soc": self.__soc_default,
  57. "rom": self.__rom_default,
  58. "ota": self.__ota_default,
  59. "wireless": self.__wireless_default,
  60. "ssid": self.__ssid_default,
  61. "bssid": self.__bssid_default,
  62. "rssi": self.__rssi_default,
  63. "auth": self.__auth_default,
  64. "time": self.__time_default}
  65. self.__commands = {'update': self.update,
  66. 'restart': self.restart,
  67. 'shutdown': self.shutdown}
  68. self.__mqttc = mqttc
  69. self.switchOne = Switch()
  70. self.switchOne.setConfig("1")
  71. self.switchTwo = Switch()
  72. self.switchTwo.setConfig("2")
  73. self.switchAdaptive = Switch()
  74. self.RAW_COMMAND_TOPIC = "encoder/" + device_id + "/command/"
  75. self.RAW_DATA_TOPIC = "encoder/" + device_id + "/data/"
  76. self.__previous_button_payload = 0b0
  77. def getDeviceID(self):
  78. return self.__device_id
  79. def updateField(self, field, value):
  80. if field in self.__fields.keys():
  81. self.__fields[field] = value
  82. self.__fields['time'] = time.time()
  83. return Encoder.SUCCESS
  84. else:
  85. return Encoder.FAIL
  86. def listFieldNames(self):
  87. return self.__fields.keys()
  88. def listFields(self):
  89. return self.__fields
  90. def listCommandNames(self):
  91. return self.__commands.keys()
  92. def listCommands(self):
  93. return self.__commmands
  94. def getField(self, field):
  95. return self.__fields[field]
  96. def activate(self):
  97. self.__mqttc.subscribe(self.RAW_DATA_TOPIC + "button", qos=self.DATA_BUTTON_QOS)
  98. self.__mqttc.message_callback_add(self.RAW_DATA_TOPIC + "button", self.data_button_cb)
  99. log.info("activated device" + self.__device_id)
  100. def deactivate(self):
  101. self.__mqttc.unsubscribe(self.RAW_DATA_TOPIC + "button")
  102. log.info("deactivated device" + self.__device_id)
  103. # This needs to be modified for the following:
  104. # 1. only activate switch if there is change....i.e. don't activating switch 1 again which switch 1 is already
  105. # depressed and switch 2 is pressed
  106. # 2. don't be pushing mad buttons because shit is broken
  107. def data_button_cb(self, mqttc, backend_data, msg):
  108. topic = msg.topic
  109. payload = int.from_bytes(msg.payload, 'little')
  110. switchTwo = 0b100
  111. switchOne = 0b100000
  112. # press / release key
  113. if ((payload & switchOne) ^ (self.__previous_button_payload & switchOne)):
  114. print("switchOne Change")
  115. if (payload & switchOne):
  116. self.switchOne.releaseKey()
  117. else:
  118. print("Activated!")
  119. self.switchOne.pressKey()
  120. if ((payload & switchTwo) ^ (self.__previous_button_payload & switchTwo)):
  121. print("switchTwo Change")
  122. if (payload & switchTwo):
  123. self.switchTwo.releaseKey()
  124. else:
  125. print("Activated!")
  126. self.switchTwo.pressKey()
  127. self.__previous_button_payload = payload
  128. log.debug("encoder " + self.__device_id + " button payload: " + bin(payload))
  129. def update(self):
  130. self.__mqttc.publish(self.RAW_COMMAND_TOPIC+self.OTA_COMMAND_TOPIC, b'1', qos=self.COMMAND_QOS, retain=False)
  131. log.info("sent OTA command")
  132. def restart(self):
  133. self.__mqttc.publish(self.RAW_COMMAND_TOPIC+self.RESTART_COMMAND_TOPIC, b'1', qos=self.COMMAND_QOS, retain=False)
  134. log.info("sent restart command")
  135. def shutdown(self):
  136. self.__mqttc.publish(self.RAW_COMMAND_TOPIC+self.SHUTDOWN_COMMAND_TOPIC, b'1', qos=self.COMMAND_QOS, retain=False)
  137. log.info("sent shutdown command")
  138. def locate(self):
  139. self.__mqttc.publish(self.RAW_COMMAND_TOPIC+self.LOCATE_COMMAND_TOPIC, b'1', qos=self.COMMAND_QOS, retain=False)
  140. class EncoderUI(QtWidgets.QWidget):
  141. sendCommand = QtCore.pyqtSignal(['QString'], name='sendCommand')
  142. activateDevice = QtCore.pyqtSignal(['QString', 'QString'], name='deactivateDevice')
  143. deactivateDevice = QtCore.pyqtSignal(['QString', 'QString'], name='activateDevice')
  144. __pathToIcon = {'network': './icon/network/',
  145. 'battery': './icon/battery/',
  146. 'battery_charging': './icon/battery/',
  147. 'switch': './icon/switch/'
  148. }
  149. __icon = {'network': ['network-wireless-offline-symbolic.png',
  150. 'network-wireless-signal-weak-symbolic.png',
  151. 'network-wireless-signal-ok-symbolic.png',
  152. 'network-wireless-signal-good-symbolic.png',
  153. 'network-wireless-signal-excellent-symbolic.png'],
  154. 'battery': ['battery-empty-symbolic.png',
  155. 'battery-caution-symbolic.png',
  156. 'battery-low-symbolic.png',
  157. 'battery-good-symbolic.png',
  158. 'battery-full-symbolic.png'],
  159. 'battery_charging': ['battery-empty-charging-symbolic.png',
  160. 'battery-caution-charging-symbolic.png',
  161. 'battery-low-charging-symbolic.png',
  162. 'battery-good-charging-symbolic.png',
  163. 'battery-full-charged-symbolic.png'],
  164. 'switch': ['switch-one-enabled.png',
  165. 'switch-one-disabled.png',
  166. 'switch-two-enabled.png',
  167. 'switch-two-disabled.png',
  168. 'switch-adaptive-enabled.png',
  169. 'switch-adaptive-disabled.png']
  170. }
  171. __deviceType = 'encoder/'
  172. def __init__(self, encoder):
  173. self.__deviceName = None
  174. self.__encoder = encoder
  175. self.__org = 'IdeasX'
  176. self.__app = 'Workstation-Client'
  177. self.restoreSettings()
  178. # Setup UI components
  179. super(EncoderUI, self).__init__()
  180. self.__ui = Ui_IdeasXDevice()
  181. self.__ui.setupUi(self)
  182. self.updateDevice(encoder)
  183. self.updateSwitchIcons()
  184. # Setup Signals
  185. self.setupMenu()
  186. self.__ui.buttonActivate.clicked.connect(self.activateEncoder)
  187. self.__ui.buttonSwitchOne.clicked.connect(lambda: self.openSwitchDialog(self.__encoder.switchOne))
  188. self.__ui.buttonSwitchTwo.clicked.connect(lambda: self.openSwitchDialog(self.__encoder.switchTwo))
  189. #self.activateDevice.connect(self.__wsc.activateEncoder)
  190. #self.deactivateDevice.connect(self.__wsc.deactivateEncoder)
  191. def saveSettings(self, device_name):
  192. settings = QtCore.QSettings(self.__org, self.__app)
  193. settings.beginGroup(self.__encoder.getDeviceID())
  194. settings.setValue("name", device_name)
  195. settings.setValue("type", "encoder")
  196. settings.endGroup()
  197. self.setDeviceAlisas(device_name)
  198. def restoreSettings(self):
  199. settings = QtCore.QSettings(self.__org, self.__app)
  200. settings = QtCore.QSettings(self.__org, self.__app)
  201. settings.beginGroup(self.__encoder.getDeviceID())
  202. self.__deviceName = settings.value("name", self.__encoder.getDeviceID())
  203. settings.endGroup()
  204. def openDeviceInformation(self):
  205. dialog = InfoUI()
  206. dialog.updateDisplay(self.__encoder.listFields())
  207. dialog.newDeviceName.connect(self.saveSettings)
  208. dialog.exec()
  209. def openSwitchDialog(self, switch):
  210. dialog = SwitchUI(switch)
  211. if dialog.exec_():
  212. if dialog.key != None and len(dialog.key) == 1:
  213. switch.setConfig(dialog.key, latch=False,interval=0.0, release=False, enable=dialog.enable)
  214. self.updateSwitchIcons()
  215. def setupSwitchIcon(self, path):
  216. icon = QtGui.QIcon()
  217. iconPath = self.__pathToIcon['switch']
  218. iconPath = iconPath + path
  219. icon.addPixmap(QtGui.QPixmap(iconPath), QtGui.QIcon.Normal, QtGui.QIcon.Off)
  220. return icon
  221. def setupMenu(self):
  222. # create menu actions
  223. shutdownAction = QtWidgets.QAction('Shutdown', self)
  224. resetAction = QtWidgets.QAction("Reset", self)
  225. testKeysAction = QtWidgets.QAction("Test Keys", self)
  226. openInfoAction = QtWidgets.QAction("Information", self)
  227. startOTAAction = QtWidgets.QAction("OTA Update", self)
  228. # connect signals to funcitons
  229. #testKeysAction.triggered.connect(self.testKeys)
  230. openInfoAction.triggered.connect(self.openDeviceInformation)
  231. startOTAAction.triggered.connect(lambda: self.__encoder.update())
  232. shutdownAction.triggered.connect(lambda: self.__encoder.shutdown())
  233. resetAction.triggered.connect(lambda: self.__encoder.restart())
  234. # create menu options
  235. deviceMenu = QtWidgets.QMenu()
  236. deviceMenu.addAction(shutdownAction)
  237. deviceMenu.addAction(resetAction)
  238. deviceMenu.addAction(openInfoAction)
  239. deviceMenu.addSection("Engineering Tools")
  240. #deviceMenu.addAction(testKeysAction)
  241. #deviceMenu.addAction(startOTAAction)
  242. self.__ui.buttonMenu.setPopupMode(2)
  243. self.__ui.buttonMenu.setMenu(deviceMenu)
  244. self.__ui.buttonMenu.setStyleSheet("* { padding-right: 3px } QToolButton::menu-indicator { image: none }")
  245. def activateEncoder(self):
  246. if self.__ui.buttonActivate.text() == "Activate":
  247. self.__encoder.activate()
  248. self.__ui.buttonActivate.setText("Deactivate")
  249. else:
  250. self.__encoder.deactivate()
  251. self.__ui.buttonActivate.setText("Activate")
  252. def updateDevice(self, encoder):
  253. self.__encoder = encoder
  254. self.__rssi = encoder.getField('rssi')
  255. self.__soc = pt.calculateSOC(encoder.getField('soc'))
  256. self.__vcell = pt.calculateVCell(encoder.getField('vcell'))
  257. self.__strModuleID = encoder.getField('device_id')
  258. self.__updateTime = encoder.getField('time')
  259. self.__ota = encoder.getField('ota')
  260. if self.__deviceName == None:
  261. self.setModuleID(self.__strModuleID)
  262. else:
  263. self.setDeviceAlisas(self.__deviceName)
  264. self.setSOCIcon(self.__soc)
  265. self.setRSSIIcon(self.__rssi)
  266. self.setStatusTime(self.__updateTime)
  267. self.setOTAIcon(self.__ota)
  268. def setOTAIcon(self, ota):
  269. if ota == '1':
  270. self.__ui.labelOTA.show()
  271. else:
  272. self.__ui.labelOTA.hide()
  273. def setModuleID(self, strModuleID):
  274. self.__ui.labelModuleID.setText(strModuleID)
  275. def setDeviceAlisas(self, label):
  276. self.__deviceName = label
  277. if label != None or label != "":
  278. self.__ui.labelModuleID.setText(label)
  279. else:
  280. self.__ui.labelModuleID.setText(self.__strModuleID)
  281. def setSOCIcon(self, soc):
  282. soc = int(soc)
  283. if soc >= 75:
  284. batteryIcon = 4
  285. elif soc >= 50 and soc < 75:
  286. batteryIcon = 3
  287. elif soc >= 25 and soc < 50:
  288. batteryIcon = 2
  289. elif soc >=10 and soc < 25:
  290. batteryIcon = 1
  291. elif soc < 10:
  292. batteryIcon = 0
  293. batteryIcon = self.__pathToIcon['battery']+self.__icon['battery'][batteryIcon]
  294. self.__ui.labelBattery.setPixmap(QtGui.QPixmap(batteryIcon))
  295. self.__ui.labelBattery.setToolTip(str(soc) + "%")
  296. def setStatusTime(self, updateTime):
  297. lastUpdate = time.ctime(updateTime).replace(" ", " ").split(" ")
  298. currentTime = time.ctime().replace(" ", " ").split(" ")
  299. if currentTime[1] != lastUpdate[1] or currentTime[2] != lastUpdate[2] or currentTime[4] != lastUpdate[4]:
  300. lastUpdate = lastUpdate[1] + " " + lastUpdate[2] + " " + lastUpdate[4]
  301. else:
  302. lastUpdate = lastUpdate[3]
  303. self.__ui.labelStatus.setText("Last Update: " + lastUpdate)
  304. def setRSSIIcon(self, rssi):
  305. rssi = int(rssi)
  306. if rssi >= -50:
  307. rssiIcon = 4
  308. elif rssi >= -60 and rssi < -50:
  309. rssiIcon = 3
  310. elif rssi >= -70 and rssi < -60:
  311. rssiIcon = 2
  312. elif rssi < -70:
  313. rssiIcon = 1
  314. rssiIcon = self.__pathToIcon['network'] + self.__icon['network'][rssiIcon]
  315. self.__ui.labelSignal.setPixmap(QtGui.QPixmap(rssiIcon))
  316. self.__ui.labelSignal.setToolTip(str(rssi) + " dBm")
  317. def updateSwitchIcons(self):
  318. switch1 = self.__encoder.switchOne
  319. switch2 = self.__encoder.switchTwo
  320. switchA = self.__encoder.switchAdaptive
  321. switchA.deactivate()
  322. if switch1.active:
  323. iconPath = self.__icon['switch'][0]
  324. else:
  325. iconPath = self.__icon['switch'][1]
  326. self.__ui.buttonSwitchOne.setIcon(self.setupSwitchIcon(iconPath))
  327. if switch2.active:
  328. iconPath = self.__icon['switch'][2]
  329. else:
  330. iconPath = self.__icon['switch'][3]
  331. self.__ui.buttonSwitchTwo.setIcon(self.setupSwitchIcon(iconPath))
  332. if switchA.active:
  333. iconPath = self.__icon['switch'][4]
  334. else:
  335. iconPath = self.__icon['switch'][5]
  336. self.__ui.buttonSwitchAdaptive.setIcon(self.setupSwitchIcon(iconPath))
  337. def testKeys(self):
  338. time.sleep(3)
  339. for payload in [1, 0, 2, 0, 4, 0]:
  340. self.__wsc.keyEmulator.emulateKey(self.__strModuleID, payload)
  341. time.sleep(0.1)
  342. class SwitchUI(QtWidgets.QDialog):
  343. def __init__(self, switch):
  344. super(SwitchUI, self).__init__()
  345. self.__ui = Ui_SwitchConfigDialog()
  346. self.__ui.setupUi(self)
  347. self.__ui.buttonApply.clicked.connect(self.submitOnClose)
  348. self.key = switch.getKey()
  349. self.enable = switch.getActive()
  350. self.switch = switch
  351. self.__ui.checkSwitchEnable.setChecked(self.enable)
  352. self.__ui.lineSwitchKey.setText(self.key)
  353. def submitOnClose(self):
  354. self.key = self.__ui.lineSwitchKey.text()
  355. self.enable = self.__ui.checkSwitchEnable.isChecked()
  356. self.accept()
  357. class InfoUI(QtWidgets.QDialog):
  358. newDeviceName = QtCore.pyqtSignal(['QString'], name='newDeviceName')
  359. def __init__(self):
  360. super(InfoUI, self).__init__()
  361. self.__ui = Ui_Dialog()
  362. self.__ui.setupUi(self)
  363. self.__ui.lineAlias.textEdited.connect(lambda: self.newDeviceName.emit(self.__ui.lineAlias.text()))
  364. def updateDisplay(self, encoder):
  365. self.__ui.labelBatteryCapacity.setText(encoder['soc'].decode('utf-8'))
  366. self.__ui.labelBatteryVoltage.setText(str(encoder['vcell'].decode('utf-8')))
  367. self.__ui.labelLowBattery.setText(str(encoder['lbi'].decode('utf-8')))
  368. self.__ui.labelChargeState.setText('N/A')
  369. self.__ui.labelActiveFlag.setText('N/A')
  370. self.__ui.labelAliveFlag.setText(str(encoder['alive'].decode('utf-8')))
  371. self.__ui.labelFirmwareVersion.setText(str(encoder['fw_ver'].decode('utf-8')))
  372. self.__ui.labelHardwareVersion.setText(str(encoder['hw_ver'].decode('utf-8')))
  373. self.__ui.labelOTAFlag.setText(str(encoder['ota'].decode('utf-8')))
  374. self.__ui.labelROMSlot.setText(str(encoder['rom'].decode('utf-8')))
  375. self.__ui.labelMAC.setText(str(encoder['device_id']))
  376. self.__ui.labelRSSI.setText(str(encoder['rssi'].decode('utf-8')))
  377. self.__ui.labelSSID.setText(str(encoder['ssid'].decode('utf-8')))
  378. if __name__ == "__main__":
  379. deviceID = '23:45:21:23:32'
  380. encoder = Encoder(deviceID, None)
  381. print(pt.calculateSOC(encoder.getField('soc')))
  382. app = QtWidgets.QApplication(sys.argv)
  383. encoderUI = EncoderUI(encoder)
  384. encoderUI.show()
  385. sys.exit(app.exec_())