wsc_device_encoder.py 18 KB

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