DeviceControl.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  1. import asyncio
  2. import serial.tools.list_ports
  3. import time, json
  4. from .SerialIns import SerialIns
  5. from utils.SingletonType import SingletonType
  6. from .BaseClass import BaseClass
  7. from sockets import ConnectionManager
  8. # mcu命令
  9. class DeviceControl(BaseClass,metaclass=SingletonType):
  10. def __init__(self, websocket_manager: ConnectionManager):
  11. super().__init__(websocket_manager)
  12. self.serial_ins = None
  13. self.connected_ports_dict = {} # 已连接的ports
  14. self.p_list = []
  15. self.temp_ports_dict = {}
  16. self.is_running = False
  17. def scan_serial_port(self) -> dict:
  18. # 获取所有可用串口列表
  19. ports_dict = {}
  20. ports = serial.tools.list_ports.comports()
  21. # 遍历所有端口并打印信息
  22. for port in ports:
  23. if "CH340" in port.description:
  24. ports_dict[port.name] = {
  25. "name": port.name,
  26. "device": port.device,
  27. "description": port.description,
  28. "hwid": port.hwid,
  29. "manufacturer": port.manufacturer,
  30. "product": port.product,
  31. "serial_number": port.serial_number,
  32. }
  33. if len(ports_dict) <= 0:
  34. return {}
  35. return ports_dict
  36. def remove_port(self, port_name):
  37. """移除串口"""
  38. print("remove", port_name)
  39. data = {
  40. "_type": "remove_port",
  41. "plugins_mode": "auto_select_com",
  42. "data": {"port_name": port_name},
  43. }
  44. self.sendSocketMessage(1, "串口被移除", data)
  45. def add_port_by_linkage(self, port_name):
  46. # port_value :串口基础信息
  47. # todo 根据prot_value 信息自动进行连接
  48. print("add", port_name)
  49. # 对没有连接的设备进行尝试连接
  50. message_data = {
  51. "_type": "show_info",
  52. "plugins_mode": "auto_select_com",
  53. "data": {"text": "开始识别接口:{}".format(port_name)},
  54. }
  55. self.sendSocketMessage(msg="开始识别接口:{}".format(port_name), data=message_data)
  56. time.sleep(1)
  57. """
  58. 步骤:
  59. 1、进行临时连接,并发送命令,成功后,自动连接对应设备
  60. """
  61. try:
  62. # 尝试使用115200波特率链接
  63. serial_handle = serial.Serial(port=port_name, baudrate=115200, timeout=0.5)
  64. except:
  65. message_data = {
  66. "_type": "show_info",
  67. "plugins_mode": "auto_select_com",
  68. "data": {"text": "串口:{} 被占用,或无法识别".format(port_name)},
  69. }
  70. self.sendSocketMessage(
  71. 1,
  72. msg="串口:{} 被占用,或无法识别".format(port_name).format(port_name),
  73. data=message_data,
  74. )
  75. print("串口:{} 被占用".format(port_name))
  76. return
  77. time.sleep(2)
  78. print("开始发送命令")
  79. data = [90, 1]
  80. try:
  81. serial_handle.flushInput() # 尝试重置输入缓冲区
  82. except serial.SerialTimeoutException:
  83. print("超时错误:无法在规定时间内重置输入缓冲区。")
  84. self.sendSocketMessage(
  85. 1,
  86. msg="超时错误:无法在规定时间内重置输入缓冲区。",
  87. data=None,
  88. )
  89. serial_handle.close()
  90. return
  91. print("尝试写入数据")
  92. buf = bytearray(b"")
  93. buf.extend([0x55, 0x55, (0xFF & len(data))])
  94. buf.extend(data)
  95. buf.extend([0xFF & ~sum(data)])
  96. try:
  97. self.receive_data = b""
  98. serial_handle.write(buf)
  99. except serial.SerialTimeoutException:
  100. print("写入数据错误")
  101. serial_handle.close()
  102. return
  103. time.sleep(0.3)
  104. print("尝试接收命令")
  105. receive_data = self.read_cmd(serial_handle)
  106. device_id = 0
  107. if receive_data:
  108. print("receive_data", receive_data)
  109. if receive_data[0] == 90:
  110. connect_flag = receive_data[1]
  111. device_id = receive_data[2]
  112. print("关闭串口:{}".format(port_name))
  113. serial_handle.close()
  114. if device_id > 0:
  115. if device_id == 1:
  116. self.to_connect_com(port_name)
  117. message_data = {
  118. "_type": "show_info",
  119. "plugins_mode": "auto_select_com",
  120. "data": {"text": "MCU开始连接"},
  121. }
  122. self.sendSocketMessage(
  123. msg="MCU开始连接",
  124. data=message_data,
  125. )
  126. self.connected_ports_dict[port_name] = "MCU"
  127. message_data = {
  128. "_type": "select_port_name",
  129. "plugins_mode": "auto_select_com",
  130. "data": {
  131. "device_name": "mcu" if device_id == 1 else "remote_control",
  132. "port_name": port_name,
  133. },
  134. }
  135. self.sendSocketMessage(
  136. msg="MCU连接成功",
  137. data=message_data,
  138. )
  139. else:
  140. print("串口无法识别")
  141. # 走其他途径处理
  142. # 检查当前MCU链接是否正常
  143. # 正常跳过;记录为其他列表
  144. # 不正常进行尝试连接
  145. # 连接不上,记录为其他列表
  146. def to_connect_com(self, port_name):
  147. # 关闭串口
  148. print("to_connect_com", port_name)
  149. self.close_connect()
  150. time.sleep(0.3)
  151. self.connect_state = False
  152. try:
  153. self.serial_ins = SerialIns(port_name=port_name, baud=115200, timeout=0.1)
  154. if not self.serial_ins.serial_handle:
  155. message_data= {
  156. "_type": "show_info",
  157. "plugins_mode": "mcu",
  158. "data": "MCU 打开串口失败",
  159. }
  160. self.sendSocketMessage(
  161. msg="MCU 打开串口失败",
  162. data=message_data,
  163. )
  164. self.serial_ins = None
  165. self.connect_state = False
  166. return False
  167. except:
  168. message_data={
  169. "_type": "show_info",
  170. "plugins_mode": "mcu",
  171. "data": "MCU 打开串口失败",
  172. }
  173. self.sendSocketMessage(
  174. msg="MCU 打开串口失败",
  175. data=message_data,
  176. )
  177. self.serial_ins = None
  178. self.connect_state = False
  179. return False
  180. message_data={"_type": "show_info", "plugins_mode": "mcu", "data": "MCU 开始连接"}
  181. self.sendSocketMessage(
  182. msg="MCU 开始连接",
  183. data=message_data,
  184. )
  185. # =======================发送连接请求=================================
  186. cmd = 90
  187. data = [cmd, 1]
  188. print("405 发送 连接请求 -----------------------------------------")
  189. print(self.serial_ins)
  190. # self.serial_ins.clearn_flush()
  191. self.serial_ins.write_cmd(data)
  192. # 延迟接收数据
  193. time.sleep(0.3)
  194. receive_data = self.serial_ins.read_cmd(out_time=1)
  195. if receive_data:
  196. print(
  197. "409 receive_data--90:{}".format(self.change_hex_to_int(receive_data))
  198. )
  199. if receive_data:
  200. # receive_data[2]=1 表示为MCU设备编号
  201. if receive_data[0] == 90 and receive_data[2] == 1:
  202. connect_flag = receive_data[1]
  203. # 是否有初始化
  204. try:
  205. mcu_has_been_set = receive_data[
  206. 6
  207. ] # 设备是否有初始化 ,1 表示已初始化
  208. except:
  209. mcu_has_been_set = 99 # 未知状态
  210. print("MCU初始化信息{}".format(mcu_has_been_set))
  211. message_data = {
  212. "_type": "show_info",
  213. "plugins_mode": "mcu",
  214. "data": "MCU 已连接",
  215. }
  216. self.sendSocketMessage(
  217. msg="MCU 已连接",
  218. data=message_data,
  219. )
  220. self.connect_state = True
  221. if not self.is_running:
  222. print("MCU start")
  223. self.is_running = True
  224. print("MCU 已连接")
  225. self.port_name = port_name
  226. return
  227. print("MCU 连接失败")
  228. message_data = {
  229. "_type": "show_info",
  230. "plugins_mode": "mcu",
  231. "data": "MCU 连接失败",
  232. }
  233. self.sendSocketMessage(
  234. msg="MCU 连接失败",
  235. data=message_data,
  236. )
  237. self.close_connect()
  238. def close_connect(self):
  239. self.port_name = ""
  240. if self.serial_ins:
  241. self.serial_ins.close_serial_port()
  242. self.connect_state = False
  243. async def checkMcuConnection(device_ctrl: DeviceControl):
  244. """实时检测串口是否连接"""
  245. while True:
  246. await asyncio.sleep(0.5)
  247. ports_dict = device_ctrl.scan_serial_port()
  248. device_ctrl.temp_ports_dict = ports_dict
  249. if not ports_dict:
  250. # 全部清空 移除所有串口
  251. if device_ctrl.p_list:
  252. _p = device_ctrl.p_list.pop()
  253. device_ctrl.remove_port(_p)
  254. print("串口未连接,请检查")
  255. device_ctrl.sendSocketMessage(code=1, msg="串口未连接,请检查")
  256. continue
  257. if ports_dict:
  258. for index, _i in enumerate(device_ctrl.p_list):
  259. if _i not in ports_dict:
  260. _p = device_ctrl.p_list.pop(index)
  261. device_ctrl.remove_port(_p)
  262. for _port_name, _port_value in ports_dict.items():
  263. if _port_name not in device_ctrl.p_list:
  264. # try:
  265. device_ctrl.p_list.append(_port_name)
  266. device_ctrl.add_port_by_linkage(_port_name)
  267. # except BaseException as e:
  268. # print("串口不存在{} {}".format(_port_name, e))