smart_shooter_class.py 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  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
  9. import zmq, sys
  10. from utils.SingletonType import SingletonType
  11. from mcu.BaseClass import BaseClass
  12. # 定义为单例模式,避免被重复实例化
  13. class SmartShooter(BaseClass,metaclass=SingletonType):
  14. SET_REQ = "tcp://127.0.0.1:54544"
  15. LISTEN_REQ = "tcp://127.0.0.1:54543"
  16. def __init__(self, websocket_manager):
  17. super().__init__(websocket_manager)
  18. self.msg_type = "smart_shooter"
  19. self.websocket_manager = websocket_manager
  20. # 是否停止监听
  21. self.stop_listen = False
  22. # 相机连接状态
  23. self.connect_status = False
  24. # 实时预览状态
  25. self.perview_state = False
  26. self.callback_listen = None
  27. # loop = asyncio.get_event_loop()
  28. # loop.create_task(
  29. # self.connect_listen(),
  30. # name="connect_listen",
  31. # )
  32. def __send_tcp_message(self, socket, msg):
  33. # await asyncio.sleep(0.01)
  34. socket.send_string(json.dumps(msg))
  35. rep = socket.recv()
  36. str_msg = rep.decode("utf-8")
  37. json_msg = json.loads(str_msg)
  38. return json_msg
  39. def __create_req(self) -> tuple[zmq.Socket, zmq.Context]:
  40. context = zmq.Context()
  41. req_socket = context.socket(zmq.REQ)
  42. # 设置发送超时为 5000 毫秒(5 秒)
  43. req_socket.setsockopt(zmq.RCVTIMEO, 5000)
  44. # 设置接收超时为 5000 毫秒(5 秒)
  45. req_socket.setsockopt(zmq.SNDTIMEO, 5000)
  46. req_socket.setsockopt(zmq.LINGER, 0) # 设置为 0 表示不等待未完成的操作
  47. req_socket.connect(self.SET_REQ)
  48. return req_socket, context
  49. def __create_listen(self) -> tuple[zmq.Socket, zmq.Context]:
  50. context = zmq.Context()
  51. listen_socket = context.socket(zmq.SUB)
  52. listen_socket.setsockopt(zmq.SUBSCRIBE, b"")
  53. # 设置发送超时为 5000 毫秒(5 秒)
  54. listen_socket.setsockopt(zmq.RCVTIMEO, 5000)
  55. # 设置接收超时为 5000 毫秒(5 秒)
  56. listen_socket.setsockopt(zmq.SNDTIMEO, 5000)
  57. listen_socket.setsockopt(zmq.LINGER, 0) # 设置为 0 表示不等待未完成的操作
  58. listen_socket.connect(self.LISTEN_REQ)
  59. return listen_socket, context
  60. async def GetCameraInfo(self):
  61. await asyncio.sleep(0.01)
  62. '''
  63. 实时获取相机信息,是否连接、软件是否被打开
  64. '''
  65. try:
  66. socket, context = self.__create_req()
  67. req = {}
  68. req["msg_type"] = "Request"
  69. req["msg_id"] = "GetCamera"
  70. req["msg_seq_num"] = 0
  71. req["CameraSelection"] = "All"
  72. json_msg = self.__send_tcp_message(socket, req)
  73. msg_result = json_msg.get("msg_result")
  74. if not msg_result:
  75. self.connect_status = False
  76. msg_send = "相机未连接或软件未打开"
  77. # self.websocket_manager
  78. self.sendSocketMessage(
  79. code=0,
  80. msg=msg_send,
  81. device_status=-1,
  82. )
  83. return False, msg_send
  84. cameraInfo = json_msg.get("CameraInfo")
  85. if cameraInfo == None or len(cameraInfo) == 0:
  86. self.connect_status = False
  87. msg_send = "相机未连接"
  88. self.sendSocketMessage(
  89. code=0,
  90. msg=msg_send,
  91. device_status=-1,
  92. )
  93. return False, "相机未连接"
  94. # 链接的相机
  95. connect_camera = cameraInfo[0]
  96. CameraStatus = connect_camera.get("CameraStatus")
  97. if CameraStatus != "Ready":
  98. self.connect_status = False
  99. msg_send = "相机未连接"
  100. self.sendSocketMessage(
  101. code=0,
  102. msg=msg_send,
  103. device_status=-1,
  104. )
  105. return False, msg_send
  106. self.connect_status = True
  107. msg_send = "相机已连接"
  108. self.sendSocketMessage(
  109. code=0,
  110. msg=msg_send,
  111. device_status=2,
  112. )
  113. return True, "相机已连接"
  114. except:
  115. self.connect_status = False
  116. socket.close()
  117. context.term()
  118. msg_send = "相机未连接或软件未打开"
  119. self.sendSocketMessage(
  120. code=0,
  121. msg=msg_send,
  122. device_status=-1,
  123. )
  124. return False, msg_send
  125. async def EnableCameraPreview(self,enable_status=True):
  126. """
  127. 激活相机预览
  128. """
  129. camera_states,_ = await self.GetCameraInfo()
  130. if not camera_states:
  131. return False, "请先连接相机"
  132. try:
  133. socket, context = self.__create_req()
  134. req = {}
  135. req["msg_type"] = "Request"
  136. req["msg_id"] = "EnableLiveview"
  137. req["msg_seq_num"] = 0
  138. req["CameraSelection"] = "All"
  139. req["Enable"] = enable_status
  140. json_msg = self.__send_tcp_message(socket, req)
  141. msg_result = json_msg.get("msg_result")
  142. if not msg_result:
  143. self.perview_state = False
  144. return False, "预览启用失败"
  145. return True, "预览启用成功" if enable_status else "预览关闭成功"
  146. except:
  147. self.perview_state = False
  148. socket.close()
  149. context.term()
  150. return False, "相机未连接或软件未打开"
  151. async def CameraAutofocus(self):
  152. """
  153. 相机自动对焦
  154. """
  155. camera_states, _ = await self.GetCameraInfo()
  156. if not camera_states:
  157. return False, "请先连接相机"
  158. try:
  159. socket, context = self.__create_req()
  160. req = {}
  161. req["msg_type"] = "Request"
  162. req["msg_id"] = "Autofocus"
  163. req["msg_seq_num"] = 0
  164. req["CameraSelection"] = "All"
  165. json_msg = self.__send_tcp_message(socket, req)
  166. print("json_msg", json_msg)
  167. msg_result = json_msg.get("msg_result")
  168. if not msg_result:
  169. return False, "对焦失败"
  170. return True, "对焦成功"
  171. except:
  172. socket.close()
  173. context.term()
  174. return False, "相机未连接或软件未打开"
  175. async def CameraShooter(self):
  176. """
  177. 执行拍照
  178. """
  179. camera_states, _ = await self.GetCameraInfo()
  180. if not camera_states:
  181. return False, "请先连接相机"
  182. try:
  183. socket, context = self.__create_req()
  184. req = {}
  185. req["msg_type"] = "Request"
  186. req["msg_id"] = "Shoot"
  187. req["msg_seq_num"] = 0
  188. req["CameraSelection"] = "All"
  189. json_msg = self.__send_tcp_message(socket, req)
  190. msg_result = json_msg.get("msg_result")
  191. if not msg_result:
  192. return False, "拍照失败"
  193. return True, "拍照成功"
  194. except:
  195. socket.close()
  196. context.term()
  197. return False, "相机未连接或软件未打开"
  198. async def connect_listen(self):
  199. # 发起监听
  200. sub_socket, context = self.__create_listen()
  201. print("构建监听")
  202. while True:
  203. await asyncio.sleep(0.01)
  204. if self.callback_listen == None:
  205. continue
  206. camera_states, camera_msg = await self.GetCameraInfo()
  207. if not camera_states:
  208. print("相机未连接")
  209. await asyncio.sleep(5) # 等待相机连接
  210. continue
  211. if self.stop_listen:
  212. break
  213. try:
  214. self.connect_status = True
  215. raw = sub_socket.recv()
  216. str_msg = raw.decode("utf-8")
  217. json_msg = json.loads(str_msg)
  218. if json_msg["msg_id"] == "NetworkPing":
  219. continue
  220. self.callback_listen(json_msg)
  221. except zmq.Again:
  222. print("接收超时,继续监听...")
  223. continue
  224. except Exception as e:
  225. self.connect_status = False
  226. print(f"发生错误: {e}")
  227. break
  228. sub_socket.close()
  229. context.term()
  230. print("smart shooter连接断开")