smart_shooter_class.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
  1. import json
  2. import datetime
  3. import random
  4. import zmq
  5. import asyncio
  6. from PIL import Image
  7. from io import BytesIO
  8. import base64, threading
  9. import zmq, sys, time
  10. from utils.SingletonType import SingletonType
  11. import settings
  12. import logging
  13. from databases import SqlQuery, CRUD, SysConfigs
  14. from utils.common import message_queue
  15. from models import SysConfigParams
  16. logger = logging.getLogger(__name__)
  17. # 定义为单例模式,避免被重复实例化
  18. class SmartShooter(metaclass=SingletonType):
  19. SET_REQ = "tcp://127.0.0.1:54544"
  20. LISTEN_REQ = "tcp://127.0.0.1:54543"
  21. def __init__(self, websocket_manager):
  22. self.msg_type = "smart_shooter"
  23. self.websocket_manager = websocket_manager
  24. # 是否停止监听
  25. self.stop_listen = False
  26. # 相机连接状态
  27. self.connect_status = False
  28. # 实时预览状态
  29. self.perview_state = False
  30. self.callback_listen = None
  31. self.listen_init = None
  32. self.websocket = None
  33. # self.main_loop = None
  34. def __send_tcp_message(self, socket, msg):
  35. # await asyncio.sleep(0.01)
  36. socket.send_string(json.dumps(msg))
  37. rep = socket.recv()
  38. str_msg = rep.decode("utf-8")
  39. json_msg = json.loads(str_msg)
  40. return json_msg
  41. def __create_req(self, time_out=5) -> tuple[zmq.Socket, zmq.Context]:
  42. context = zmq.Context()
  43. req_socket = context.socket(zmq.REQ)
  44. # 设置发送超时为 5000 毫秒(5 秒)
  45. req_socket.setsockopt(zmq.RCVTIMEO, time_out * 1000)
  46. # 设置接收超时为 5000 毫秒(5 秒)
  47. req_socket.setsockopt(zmq.SNDTIMEO, time_out * 1000)
  48. req_socket.setsockopt(zmq.LINGER, 0) # 设置为 0 表示不等待未完成的操作
  49. req_socket.connect(self.SET_REQ)
  50. return req_socket, context
  51. async def sendMessageSocket(self, message):
  52. if self.websocket_manager and self.websocket:
  53. await self.websocket_manager.send_personal_message(message, self.websocket)
  54. def __create_listen(self) -> tuple[zmq.Socket, zmq.Context]:
  55. context = zmq.Context()
  56. listen_socket = context.socket(zmq.SUB)
  57. listen_socket.setsockopt(zmq.SUBSCRIBE, b"")
  58. # 设置发送超时为 5000 毫秒(5 秒)
  59. listen_socket.setsockopt(zmq.RCVTIMEO, 5000)
  60. # 设置接收超时为 5000 毫秒(5 秒)
  61. listen_socket.setsockopt(zmq.SNDTIMEO, 5000)
  62. listen_socket.setsockopt(zmq.LINGER, 0) # 设置为 0 表示不等待未完成的操作
  63. listen_socket.connect(self.LISTEN_REQ)
  64. return listen_socket, context
  65. async def GetCameraProperty(self, CameraKey=None):
  66. """获取相机属性"""
  67. await asyncio.sleep(0.01)
  68. """
  69. 实时获取相机信息,是否连接、软件是否被打开
  70. """
  71. socket, context = self.__create_req(time_out=2)
  72. try:
  73. req = {}
  74. req["msg_type"] = "Request"
  75. req["msg_id"] = "GetCamera"
  76. req["msg_seq_num"] = 0
  77. req["CameraSelection"] = "All"
  78. if CameraKey is not None:
  79. req["CameraSelection"] = "Single"
  80. req["CameraKey"] = CameraKey
  81. json_msg = self.__send_tcp_message(socket, req)
  82. msg_result = json_msg.get("msg_result")
  83. if not msg_result:
  84. self.connect_status = False
  85. msg_send = "相机未连接或软件未打开"
  86. return False, msg_send
  87. cameraInfo = json_msg.get("CameraInfo")
  88. if cameraInfo == None or len(cameraInfo) == 0:
  89. self.connect_status = False
  90. msg_send = "相机未连接"
  91. return False, "相机未连接"
  92. # 链接的相机
  93. CameraStatus = False
  94. CameraIndex = -1
  95. for cam_idx, item in enumerate(cameraInfo):
  96. CameraStatus = item.get("CameraStatus") in ["Ready", "Busy"]
  97. if CameraStatus == True:
  98. CameraIndex = cam_idx
  99. break
  100. # CameraStatus = connect_camera.get("CameraStatus")
  101. if not CameraStatus:
  102. self.connect_status = False
  103. msg_send = "相机未连接"
  104. return False, msg_send
  105. self.connect_status = True
  106. msg_send = "相机已连接"
  107. return True, cameraInfo[CameraIndex].get("CameraPropertyInfo")
  108. except zmq.Again:
  109. msg_send = "相机未连接或软件未打开"
  110. return False, msg_send
  111. except Exception as e:
  112. self.connect_status = False
  113. socket.close()
  114. context.term()
  115. msg_send = "相机未连接或软件未打开"
  116. return False, msg_send
  117. def get_iso_range(self,camera_info):
  118. """
  119. 从相机数据中获取 ISO 的可调范围
  120. :param camera_data: 相机状态数据列表 (即你提供的那个长列表)
  121. :return: ISO 范围列表,如果未找到则返回 None
  122. """
  123. # 获取属性信息列表
  124. property_info_list = camera_info.get('CameraPropertyInfo', [])
  125. for prop in property_info_list:
  126. # 查找 CameraPropertyType 等于 'ISO' 的项
  127. if prop.get('CameraPropertyType') == 'ISO':
  128. return prop.get('CameraPropertyRange')
  129. return None
  130. async def GetCameraInfo(self, is_send=True, msg_type="", CameraKey=None,isMultCameraMode=False):
  131. await asyncio.sleep(0.001)
  132. # self.msg_type = msg_type
  133. """
  134. 实时获取相机信息,是否连接、软件是否被打开
  135. """
  136. socket, context = self.__create_req(time_out=5)
  137. try:
  138. req = {}
  139. req["msg_type"] = "Request"
  140. req["msg_id"] = "GetCamera"
  141. req["msg_seq_num"] = 0
  142. req["CameraSelection"] = "All"
  143. if CameraKey is not None:
  144. req["CameraSelection"] = "Single"
  145. req["CameraKey"] = CameraKey
  146. json_msg = self.__send_tcp_message(socket, req)
  147. msg_result = json_msg.get("msg_result")
  148. if not msg_result:
  149. self.connect_status = False
  150. msg_send = f"请检查{CameraKey},相机是否连接" if CameraKey else f"请检查相机是否连接"
  151. if is_send:
  152. message = {
  153. "code": 1,
  154. "msg": msg_send,
  155. "data": msg_result,
  156. "CameraKey":CameraKey,
  157. "msg_type": msg_type,
  158. "device_status": -1,
  159. }
  160. await self.sendMessageSocket(message)
  161. return False, msg_send
  162. cameraInfo = json_msg.get("CameraInfo")
  163. if cameraInfo == None or len(cameraInfo) == 0:
  164. self.connect_status = False
  165. msg_send = (
  166. f"请检查{CameraKey},相机是否连接"
  167. if CameraKey
  168. else f"请检查相机是否连接"
  169. )
  170. if is_send:
  171. message = {
  172. "code": 1,
  173. "msg": msg_send,
  174. "data": msg_result,
  175. "msg_type": msg_type,
  176. "device_status": -1,
  177. }
  178. await self.sendMessageSocket(message)
  179. return False, "相机未连接"
  180. # 链接的相机
  181. CameraStatus = any(
  182. item.get("CameraStatus") in ["Ready", "Busy"] for item in cameraInfo
  183. )
  184. # 在这里处理
  185. CameraLists = [
  186. {
  187. "CameraSelection": item.get("CameraSelection"),
  188. "CameraKey": item.get("CameraKey"),
  189. "CameraName": item.get("CameraName"),
  190. "CameraStatus": item.get("CameraStatus") in ["Ready", "Busy"],
  191. "CameraISO": self.get_iso_range(item),
  192. }
  193. for item in cameraInfo
  194. ]
  195. if not CameraStatus:
  196. self.connect_status = False
  197. msg_send = (
  198. f"请检查{CameraKey},相机是否连接"
  199. if CameraKey
  200. else f"请检查相机是否连接"
  201. )
  202. if is_send:
  203. message = {
  204. "code": 1,
  205. "msg": msg_send,
  206. "data": msg_result,
  207. "CameraKey":CameraKey,
  208. "msg_type": msg_type,
  209. "device_status": -1,
  210. }
  211. await self.sendMessageSocket(message)
  212. return False, msg_send
  213. self.connect_status = True
  214. msg_send = f"相机{CameraKey}已连接" if CameraKey else "相机已连接"
  215. # print("CameraLists", CameraLists)
  216. if is_send:
  217. message = {
  218. "code": 0,
  219. "msg": msg_send,
  220. "data": msg_result,
  221. "CameraLists":CameraLists,
  222. "msg_type": msg_type,
  223. "device_status": 2,
  224. }
  225. await self.sendMessageSocket(message)
  226. # print("相机已连接状态信息---->", cameraInfo)
  227. # self.initConfigIsoSettings(
  228. # CameraLists=CameraLists, isMultCameraMode=isMultCameraMode
  229. # )
  230. return True, "相机已连接"
  231. except zmq.Again as e:
  232. self.connect_status = False
  233. socket.close()
  234. context.term()
  235. print("获取相机信息超时,继续监听...",e)
  236. msg_send = f"请检查{CameraKey},相机是否连接" if CameraKey else f"请检查相机是否连接"
  237. return False, msg_send
  238. except Exception as e:
  239. print("相机状态获取异常", e)
  240. self.connect_status = False
  241. socket.close()
  242. context.term()
  243. msg_send = (
  244. f"请检查{CameraKey},相机是否连接"
  245. if CameraKey
  246. else f"请检查相机是否连接"
  247. )
  248. if is_send:
  249. message = {
  250. "code": 1,
  251. "msg": msg_send,
  252. "data": None,
  253. "msg_type": msg_type,
  254. "device_status": -1,
  255. }
  256. await self.sendMessageSocket(message)
  257. return False, msg_send
  258. async def SetCameraFPS(self, fps=5, CameraKey=None):
  259. """
  260. 激活相机预览
  261. """
  262. camera_states, _ = await self.GetCameraInfo(is_send=False, CameraKey=CameraKey)
  263. if not camera_states:
  264. return False, "请先连接相机"
  265. socket, context = self.__create_req()
  266. try:
  267. req = {}
  268. req["msg_type"] = "Request"
  269. req["msg_id"] = "LiveviewFPS"
  270. req["msg_seq_num"] = 0
  271. req["CameraSelection"] = "All"
  272. if CameraKey is not None:
  273. req["CameraSelection"] = "Single"
  274. req["CameraKey"] = CameraKey
  275. req["CameraLiveviewFPS"] = fps
  276. json_msg = self.__send_tcp_message(socket, req)
  277. msg_result = json_msg.get("msg_result")
  278. if not msg_result:
  279. self.perview_state = False
  280. return False, "设置失败"
  281. return True, "设置失败"
  282. except:
  283. self.perview_state = False
  284. socket.close()
  285. context.term()
  286. return False, "相机未连接或软件未打开"
  287. async def setCameraProperty(self, property="ISO", value=0, CameraKey=None):
  288. # SetProperty
  289. camera_states, _ = await self.GetCameraInfo(is_send=False, CameraKey=CameraKey)
  290. if not camera_states:
  291. return False, "请先连接相机"
  292. socket, context = self.__create_req()
  293. try:
  294. req = {}
  295. req["msg_type"] = "Request"
  296. req["msg_id"] = "SetProperty"
  297. req["msg_seq_num"] = 0
  298. req["CameraSelection"] = "All"
  299. if CameraKey is not None:
  300. req["CameraSelection"] = "Single"
  301. req["CameraKey"] = CameraKey
  302. req["CameraPropertyType"] = str(property)
  303. req["CameraPropertyValue"] = str(value)
  304. print("发送设置属性请求", req)
  305. json_msg = self.__send_tcp_message(socket, req)
  306. msg_result = json_msg.get("msg_result")
  307. if not msg_result:
  308. return False, f"{property}设置失败"
  309. return True, f"{property}设置成功"
  310. except zmq.Again:
  311. print(f"设置{property}超时,继续监听...")
  312. except:
  313. self.perview_state = False
  314. socket.close()
  315. context.term()
  316. msg_send = "相机未连接或软件未打开"
  317. return False, msg_send
  318. def getConfigIso(self, CameraKey=None):
  319. """获取ISO配置信息"""
  320. camera_configs = settings.getSysConfigs(
  321. "camera_configs",
  322. "iso_config",
  323. None,
  324. )
  325. Itemiso = {}
  326. for idx, item in enumerate(camera_configs):
  327. itemConfig = camera_configs[item]
  328. if itemConfig == {}:
  329. continue
  330. ItemCameraKey = itemConfig.get("CameraKey")
  331. if ItemCameraKey == CameraKey:
  332. Itemiso = itemConfig.get("iso")
  333. break
  334. low_iso = Itemiso.get("low", 100)
  335. high_iso = Itemiso.get("high", 6400)
  336. return low_iso, high_iso
  337. def initConfigIsoSettings(self, CameraLists=[],isMultCameraMode=False):
  338. if not CameraLists:
  339. return None
  340. """获取ISO配置信息"""
  341. camera_configs = settings.getSysConfigs(
  342. "camera_configs",
  343. "iso_config",
  344. None,
  345. )
  346. temp_A_point = camera_configs.get("A", None)
  347. if temp_A_point is not None:
  348. print("已配置无需更新")
  349. itemSettings = CameraLists[0]
  350. OldCameraKey = temp_A_point.get("CameraKey", None)
  351. if OldCameraKey == itemSettings.get("CameraKey", None):
  352. print("相机无变动。无需自动更改配置")
  353. return None
  354. basic_iso = temp_A_point.get("iso", {"low": 100, "high": 6400})
  355. if isMultCameraMode == False:
  356. points = {"A": {}}
  357. print("相机发生变动。需要更改配置")
  358. # 如果客户是单相机版本用户,需要每次同步camera信息
  359. points["A"] = {
  360. **itemSettings,
  361. "iso": basic_iso,
  362. }
  363. sys_iso_config = {"key": "camera_configs", "value": {"iso_config": points}}
  364. sys_iso_config: SysConfigParams
  365. print("单相机用户同步相机配置", sys_iso_config)
  366. settings.updateSysConfigs(params=sys_iso_config)
  367. # 同步本地到线上
  368. settings.sync_sys_configs2Online()
  369. return None
  370. points = {"A": {}, "B": {}, "C": {}}
  371. for idx,item in enumerate(points):
  372. low_iso = camera_configs.get("low", 100)
  373. high_iso = camera_configs.get("high", 6400)
  374. if idx > len(CameraLists) -1:
  375. points[item] = {
  376. "iso": {"low": low_iso, "high": high_iso},
  377. }
  378. continue
  379. itemSettings = CameraLists[idx]
  380. points[item] = {**itemSettings, "iso": {"low": low_iso, "high": high_iso}}
  381. sys_iso_config = {"key": "camera_configs", "value": {"iso_config": points}}
  382. sys_iso_config: SysConfigParams
  383. print("首次初始化", sys_iso_config)
  384. settings.updateSysConfigs(params=sys_iso_config)
  385. # 同步本地到线上
  386. settings.sync_sys_configs2Online()
  387. async def EnableCameraPreview(
  388. self, enable_status=True, msg_type="", CameraKey=None
  389. ):
  390. print("收到得msg_type====>>>", msg_type)
  391. self.msg_type = msg_type
  392. await self.SetCameraFPS(5, CameraKey=CameraKey)
  393. """
  394. 激活相机预览
  395. """
  396. camera_states, _ = await self.GetCameraInfo(is_send=False, CameraKey=CameraKey)
  397. if not camera_states:
  398. msg_send = "预览启用失败"
  399. message = {
  400. "code": 0,
  401. "msg": msg_send,
  402. "data": None,
  403. "msg_type": msg_type,
  404. "device_status": -1,
  405. }
  406. logger.error(message)
  407. await self.sendMessageSocket(message)
  408. return False, "请先连接相机"
  409. try:
  410. low_iso, high_iso = self.getConfigIso(CameraKey=CameraKey)
  411. print("LOW_ISO", low_iso)
  412. print("HIGH_ISO", high_iso)
  413. # 等于auto就不设置
  414. if enable_status == True:
  415. if str(high_iso).lower() != "auto":
  416. set_state,set_msg = await self.setCameraProperty(
  417. property="ISO", value=str(high_iso), CameraKey=CameraKey
  418. )
  419. else:
  420. print("high_iso 等于auto就不设置")
  421. if enable_status == False:
  422. if str(low_iso).lower() != "auto":
  423. set_state,set_msg = await self.setCameraProperty(
  424. property="ISO", value=str(low_iso), CameraKey=CameraKey
  425. )
  426. else:
  427. print("low_iso 等于auto就不设置")
  428. # print("设置状态", set_state, set_msg)
  429. socket, context = self.__create_req()
  430. req = {}
  431. req["msg_type"] = "Request"
  432. req["msg_id"] = "EnableLiveview"
  433. req["msg_seq_num"] = 0
  434. req["CameraSelection"] = "All"
  435. if CameraKey is not None:
  436. req["CameraSelection"] = "Single"
  437. req["CameraKey"] = CameraKey
  438. req["Enable"] = enable_status
  439. json_msg = self.__send_tcp_message(socket, req)
  440. msg_result = json_msg.get("msg_result")
  441. if not msg_result:
  442. self.perview_state = False
  443. msg_send = "预览启用失败"
  444. message = {
  445. "code": 0,
  446. "msg": msg_send,
  447. "data": None,
  448. "msg_type": msg_type,
  449. "device_status": -1,
  450. }
  451. logger.error(message)
  452. await self.sendMessageSocket(message)
  453. return False, "预览启用失败"
  454. msg_send = "预览启用成功" if enable_status else "预览关闭成功"
  455. message = {
  456. "code": 0,
  457. "msg": msg_send,
  458. "data": None,
  459. "msg_type": msg_type,
  460. "device_status": 2,
  461. }
  462. logger.error(message)
  463. await self.sendMessageSocket(message)
  464. return True, "预览启用成功" if enable_status else "预览关闭成功"
  465. except zmq.Again:
  466. print("启动预览超时,继续监听...")
  467. except:
  468. self.perview_state = False
  469. socket.close()
  470. context.term()
  471. msg_send = "相机未连接或软件未打开"
  472. message = {
  473. "code": 1,
  474. "msg": msg_send,
  475. "data": None,
  476. "msg_type": msg_type,
  477. "device_status": -1,
  478. }
  479. logger.error(message)
  480. await self.sendMessageSocket(message)
  481. return False, "相机未连接或软件未打开"
  482. async def CameraAutofocus(self, CameraKey=None):
  483. """
  484. 相机自动对焦
  485. """
  486. camera_states, _ = await self.GetCameraInfo(is_send=False, CameraKey=CameraKey)
  487. print("CameraAutofocus 执行对焦")
  488. if not camera_states:
  489. return False, "请先连接相机"
  490. socket, context = self.__create_req()
  491. try:
  492. req = {}
  493. req["msg_type"] = "Request"
  494. req["msg_id"] = "Autofocus"
  495. req["msg_seq_num"] = 0
  496. req["CameraSelection"] = "All"
  497. if CameraKey is not None:
  498. req["CameraSelection"] = "Single"
  499. req["CameraKey"] = CameraKey
  500. start_time = time.time() # 添加对焦开始时间记录
  501. json_msg = self.__send_tcp_message(socket, req)
  502. end_time = time.time() # 添加对焦结束时间记录
  503. logger.info(f"对焦通信耗时: {end_time - start_time:.4f} 秒")
  504. print("json_msg", json_msg)
  505. msg_result = json_msg.get("msg_result")
  506. if not msg_result:
  507. return False, "对焦失败"
  508. return True, "对焦成功"
  509. except zmq.Again:
  510. print("对焦超时,继续监听...")
  511. except:
  512. socket.close()
  513. context.term()
  514. return False, "相机未连接或软件未打开"
  515. async def CameraShooter(
  516. self, msg_type="", goods_art_no="", id=0, is_af=False, delay=0.5, CameraKey=None
  517. ):
  518. # 这里延迟一秒钟 等待前置命令完成
  519. await asyncio.sleep(delay)
  520. # 对焦混用
  521. if is_af:
  522. # await self.EnableCameraPreview(
  523. # enable_status=True,
  524. # msg_type="smart_shooter_enable_preview_status",
  525. # CameraKey=CameraKey,
  526. # )
  527. # start_time = time.time()
  528. await self.CameraAutofocus(CameraKey=CameraKey)
  529. # end_time = time.time()
  530. # elapsed_time = end_time - start_time
  531. # logger.info(f"自动对焦耗时 {elapsed_time:.4f} 秒")
  532. # await self.EnableCameraPreview(
  533. # enable_status=False, msg_type="smart_shooter_enable_preview_status", CameraKey=CameraKey
  534. # )
  535. self.msg_type = msg_type
  536. print("camera_states", msg_type)
  537. """
  538. 执行拍照
  539. """
  540. camera_states, _ = await self.GetCameraInfo(is_send=True, CameraKey=CameraKey)
  541. print("camera_states CameraShooter", camera_states)
  542. if not camera_states:
  543. return False, "请先连接相机"
  544. socket, context = self.__create_req()
  545. try:
  546. low_iso, high_iso = self.getConfigIso(CameraKey=CameraKey)
  547. print("LOW_ISO", low_iso)
  548. print("HIGH_ISO", high_iso)
  549. if str(low_iso).lower() != "auto":
  550. await self.setCameraProperty(
  551. property="ISO", value=str(low_iso), CameraKey=CameraKey
  552. )
  553. else:
  554. print("low_iso 等于auto就不设置")
  555. req = {}
  556. req["msg_type"] = "Request"
  557. req["msg_id"] = "Shoot"
  558. req["msg_seq_num"] = 1
  559. req["CameraSelection"] = "All"
  560. if CameraKey is not None:
  561. req["CameraSelection"] = "Single"
  562. req["CameraKey"] = CameraKey
  563. if goods_art_no != "" and id != 0:
  564. # 此处用逗号分割,货号和id,需要在监听部分进行切割保存处理,如果使用下划线或者减号,可能货号中存在对应符号
  565. req["PhotoOrigin"] = f"{goods_art_no},{id}"
  566. else:
  567. req["PhotoOrigin"] = ""
  568. print("发送拍照请求", req)
  569. json_msg = self.__send_tcp_message(socket, req)
  570. print("CameraShooter", json_msg)
  571. msg_result = json_msg.get("msg_result")
  572. if not msg_result:
  573. msg_send = "拍照失败"
  574. message = {
  575. "code": 1,
  576. "msg": msg_send,
  577. "data": None,
  578. "msg_type": self.msg_type,
  579. "device_status": -1,
  580. }
  581. await self.sendMessageSocket(message)
  582. return False, msg_send
  583. msg_send = "拍照成功"
  584. message = {
  585. "code": 0,
  586. "msg": msg_send,
  587. "data": None,
  588. "msg_type": self.msg_type,
  589. "device_status": 2,
  590. }
  591. await self.sendMessageSocket(message)
  592. return True, "拍照成功"
  593. except zmq.Again:
  594. print("拍照超时")
  595. msg_send = "相机未连接或软件未打开"
  596. message = {
  597. "code": 1,
  598. "msg": msg_send,
  599. "data": {goods_art_no: goods_art_no, "id": id},
  600. "msg_type": self.msg_type,
  601. "device_status": -1,
  602. }
  603. await self.sendMessageSocket(message)
  604. return True, "拍照失败"
  605. except Exception as e:
  606. print("拍照出错",e)
  607. socket.close()
  608. context.term()
  609. msg_send = "相机未连接或软件未打开"
  610. message = {
  611. "code": 1,
  612. "msg": msg_send,
  613. "data": None,
  614. "msg_type": self.msg_type,
  615. "device_status": -1,
  616. }
  617. await self.sendMessageSocket(message)
  618. return False, msg_send
  619. async def asyncMessageListen(self):
  620. if self.websocket.client_state.name != "CONNECTED":
  621. print("WebSocket连接已断开,停止发送消息")
  622. return
  623. try:
  624. # 使用非阻塞方式检查队列中是否有项目
  625. if not message_queue.empty():
  626. message = message_queue.get_nowait()
  627. print("发送消息中。。。。。", message)
  628. await self.sendMessageSocket(message)
  629. message_queue.task_done()
  630. except Exception as e:
  631. # 处理可能的异常,如队列为空等
  632. pass
  633. def connect_listen(self):
  634. print("smart shooter connect_listen START")
  635. if self.connect_status == True or self.listen_init == True:
  636. return True
  637. sub_socket, context = self.__create_listen()
  638. print("构建监听", self.connect_status)
  639. # 不再需要 self.listen_loop,我们只使用主循环
  640. try:
  641. while True:
  642. self.listen_init = True
  643. if self.stop_listen:
  644. break
  645. # 1. 阻塞接收 ZMQ 消息 (这是唯一的阻塞点,但在子线程,所以没问题)
  646. try:
  647. raw = sub_socket.recv() # 这里会阻塞直到有消息或超时
  648. str_msg = raw.decode("utf-8")
  649. json_msg = json.loads(str_msg)
  650. if json_msg.get("msg_id") == "NetworkPing":
  651. continue
  652. # 2. 将回调提交到【主事件循环】
  653. if hasattr(self, "main_loop") and self.main_loop:
  654. # 非阻塞提交,立即返回
  655. asyncio.run_coroutine_threadsafe(
  656. self.callback_listen(json_msg), self.main_loop
  657. )
  658. else:
  659. print("Error: main_loop not set in SmartShooter")
  660. except zmq.Again:
  661. # 超时,继续循环
  662. continue
  663. except Exception as e:
  664. print(f"ZMQ Error: {e}")
  665. break
  666. finally:
  667. self.listen_init = False
  668. self.connect_status = False
  669. sub_socket.close()
  670. context.term()
  671. print("smart shooter连接断开")