ptz_camera.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. """
  2. 球机(PTZ)控制模块
  3. 负责PTZ控制、精确定位和视频流获取
  4. """
  5. import os
  6. # 必须在导入cv2之前设置,防止FFmpeg多线程解码崩溃
  7. os.environ['OPENCV_FFMPEG_CAPTURE_OPTIONS'] = 'rtsp_transport;tcp|threads;1'
  8. import math
  9. import time
  10. import threading
  11. import queue
  12. from typing import Optional, Tuple, Dict
  13. from dataclasses import dataclass
  14. import cv2
  15. import numpy as np
  16. from config import PTZ_CAMERA, PTZ_CONFIG
  17. from dahua_sdk import DahuaSDK, PTZCommand
  18. from video_lock import safe_read, safe_is_opened
  19. @dataclass
  20. class PTZPosition:
  21. """PTZ位置信息"""
  22. pan: float # 水平角度 (0-360度)
  23. tilt: float # 垂直角度 (-90到90度)
  24. zoom: float # 变倍 (1-最大倍数)
  25. class PTZCamera:
  26. """球机控制类"""
  27. def __init__(self, sdk: DahuaSDK, camera_config: Dict = None):
  28. """
  29. 初始化球机
  30. Args:
  31. sdk: 大华SDK实例
  32. camera_config: 摄像头配置
  33. """
  34. self.sdk = sdk
  35. self.config = camera_config or PTZ_CAMERA
  36. # 全局 PTZ 配置作为默认值,camera_config 中的同名 key 可覆盖
  37. self.ptz_config = dict(PTZ_CONFIG)
  38. if camera_config:
  39. for key in ['mount_type', 'pan_flip', 'tilt_flip', 'coordinate_offset',
  40. 'tilt_offset', 'pan_offset', 'pan_edge_offset', 'pan_curve_power',
  41. 'tilt_linear_enabled', 'tilt_y0', 'tilt_y1', 'tilt_curve_power',
  42. 'pan_range', 'tilt_range', 'pan_center', 'tilt_center',
  43. 'overlap_pan_range', 'overlap_tilt_range',
  44. 'overlap_pan_step', 'overlap_tilt_step']:
  45. if key in camera_config:
  46. self.ptz_config[key] = camera_config[key]
  47. self.login_handle = None
  48. self.connected = False
  49. # 当前位置
  50. self.current_position = PTZPosition(pan=0, tilt=0, zoom=1)
  51. self.position_lock = threading.Lock()
  52. # 视频流 (用于校准时抓拍球机画面)
  53. self.rtsp_cap = None
  54. self._rtsp_lock = threading.Lock()
  55. self.current_frame = None
  56. self.frame_lock = threading.Lock()
  57. self.stream_thread = None
  58. self.running_stream = False
  59. self._camera_id = 'ptz' # 用于per-camera锁
  60. def connect(self) -> bool:
  61. """
  62. 连接球机
  63. Returns:
  64. 是否成功
  65. """
  66. print(f"[PTZCamera] 正在连接球机: IP={self.config['ip']}:{self.config['port']}, 通道={self.config['channel']}")
  67. login_handle, error = self.sdk.login(
  68. self.config['ip'],
  69. self.config['port'],
  70. self.config['username'],
  71. self.config['password']
  72. )
  73. if login_handle is None:
  74. print(f"[PTZCamera] 连接球机失败: IP={self.config['ip']}, 错误码={error}")
  75. print(f"[PTZCamera] 请检查: 1)IP地址是否正确 2)网络是否连通 3)用户名密码是否正确")
  76. return False
  77. self.login_handle = login_handle
  78. self.connected = True
  79. print(f"[PTZCamera] 成功连接球机: {self.config['ip']}, handle={login_handle}")
  80. print(f"[PTZCamera] 配置: 通道={self.config['channel']}, 默认变倍={self.ptz_config['default_zoom']}")
  81. return True
  82. def disconnect(self):
  83. """断开连接"""
  84. self.stop_stream()
  85. if self.login_handle:
  86. self.sdk.logout(self.login_handle)
  87. self.login_handle = None
  88. self.connected = False
  89. def is_connected(self) -> bool:
  90. """是否已连接"""
  91. return self.connected
  92. def start_stream_rtsp(self, rtsp_url: str = None) -> bool:
  93. """启动RTSP视频流 (用于校准时获取球机画面)"""
  94. if rtsp_url is None:
  95. rtsp_url = self.config.get('rtsp_url') or \
  96. f"rtsp://{self.config['username']}:{self.config['password']}@{self.config['ip']}:{self.config.get('rtsp_port', 554)}/cam/realmonitor?channel=1&subtype=1"
  97. try:
  98. # 先尝试FFmpeg后端
  99. new_cap = cv2.VideoCapture(rtsp_url, cv2.CAP_FFMPEG)
  100. if not new_cap.isOpened():
  101. # FFmpeg失败,尝试GStreamer后端(ARM64上更稳定)
  102. print(f"[PTZCamera] FFmpeg后端无法打开RTSP流,尝试GStreamer后端...")
  103. try:
  104. gst_cap = cv2.VideoCapture(rtsp_url, cv2.CAP_GSTREAMER)
  105. if gst_cap.isOpened():
  106. new_cap = gst_cap
  107. print(f"[PTZCamera] 使用GStreamer后端打开RTSP流成功")
  108. else:
  109. print(f"[PTZCamera] 无法打开RTSP流: {rtsp_url}")
  110. return False
  111. except Exception as ge:
  112. print(f"[PTZCamera] GStreamer后端也不可用: {ge}")
  113. return False
  114. new_cap.set(cv2.CAP_PROP_BUFFERSIZE, 1)
  115. with self._rtsp_lock:
  116. self.rtsp_cap = new_cap
  117. self.running_stream = True
  118. self.stream_thread = threading.Thread(target=self._stream_worker, daemon=True)
  119. self.stream_thread.start()
  120. print(f"[PTZCamera] RTSP视频流已启动")
  121. return True
  122. except Exception as e:
  123. print(f"[PTZCamera] RTSP流启动失败: {e}")
  124. return False
  125. def _stream_worker(self):
  126. """视频流工作线程"""
  127. import signal
  128. if hasattr(signal, 'pthread_sigmask'):
  129. try:
  130. signal.pthread_sigmask(signal.SIG_BLOCK, {signal.SIGINT})
  131. except (AttributeError, OSError):
  132. pass
  133. max_consecutive_errors = 50
  134. error_count = 0
  135. while self.running_stream:
  136. try:
  137. with self._rtsp_lock:
  138. cap = self.rtsp_cap
  139. if cap is None or not safe_is_opened(cap, self._camera_id):
  140. time.sleep(0.1)
  141. continue
  142. ret, frame = safe_read(cap, self._camera_id)
  143. if not ret or frame is None:
  144. error_count += 1
  145. if error_count > max_consecutive_errors:
  146. print("[PTZCamera] RTSP流连续读取失败,尝试重连...")
  147. self._reconnect_rtsp()
  148. error_count = 0
  149. time.sleep(0.01)
  150. continue
  151. error_count = 0
  152. with self.frame_lock:
  153. self.current_frame = frame.copy()
  154. time.sleep(0.001)
  155. except Exception as e:
  156. err_str = str(e)
  157. if 'async_lock' in err_str or 'Assertion' in err_str:
  158. print(f"[PTZCamera] FFmpeg内部错误,3秒后重建连接: {e}")
  159. time.sleep(3)
  160. self._reconnect_rtsp()
  161. else:
  162. print(f"[PTZCamera] 视频流错误: {e}")
  163. time.sleep(0.5)
  164. def _reconnect_rtsp(self):
  165. rtsp_url = self.config.get('rtsp_url') or \
  166. f"rtsp://{self.config['username']}:{self.config['password']}@{self.config['ip']}:{self.config.get('rtsp_port', 554)}/cam/realmonitor?channel=1&subtype=1"
  167. with self._rtsp_lock:
  168. if self.rtsp_cap is not None:
  169. try:
  170. self.rtsp_cap.release()
  171. except Exception:
  172. pass
  173. self.rtsp_cap = None
  174. time.sleep(1)
  175. try:
  176. new_cap = cv2.VideoCapture(rtsp_url, cv2.CAP_FFMPEG)
  177. if safe_is_opened(new_cap, self._camera_id):
  178. new_cap.set(cv2.CAP_PROP_BUFFERSIZE, 1)
  179. with self._rtsp_lock:
  180. self.rtsp_cap = new_cap
  181. print("[PTZCamera] RTSP流重连成功")
  182. else:
  183. print("[PTZCamera] RTSP流重连失败")
  184. try:
  185. new_cap.release()
  186. except Exception:
  187. pass
  188. except Exception as e:
  189. print(f"[PTZCamera] RTSP流重连异常: {e}")
  190. def get_frame(self) -> Optional[np.ndarray]:
  191. """获取球机当前帧"""
  192. with self.frame_lock:
  193. return self.current_frame.copy() if self.current_frame is not None else None
  194. def stop_stream(self):
  195. """停止视频流"""
  196. self.running_stream = False
  197. if self.stream_thread:
  198. self.stream_thread.join(timeout=2)
  199. self.stream_thread = None
  200. with self._rtsp_lock:
  201. if self.rtsp_cap:
  202. self.rtsp_cap.release()
  203. self.rtsp_cap = None
  204. def ptz_control(self, command: int, param1: int = 0, param2: int = 0,
  205. param3: int = 0, stop: bool = False) -> bool:
  206. """
  207. PTZ控制
  208. Args:
  209. command: 控制命令
  210. param1-3: 参数
  211. stop: 是否停止
  212. Returns:
  213. 是否成功
  214. """
  215. if not self.connected:
  216. print(f"[PTZCamera] PTZ控制失败: 未连接球机")
  217. return False
  218. if self.login_handle is None or self.login_handle <= 0:
  219. print(f"[PTZCamera] PTZ控制失败: 登录句柄无效 (handle={self.login_handle})")
  220. return False
  221. return self.sdk.ptz_control(
  222. self.login_handle,
  223. self.config['channel'],
  224. command, param1, param2, param3, stop
  225. )
  226. def move_up(self, speed: int = 4, stop: bool = False) -> bool:
  227. """向上移动"""
  228. return self.ptz_control(PTZCommand.UP, 0, speed, 0, stop)
  229. def move_down(self, speed: int = 4, stop: bool = False) -> bool:
  230. """向下移动"""
  231. return self.ptz_control(PTZCommand.DOWN, 0, speed, 0, stop)
  232. def move_left(self, speed: int = 4, stop: bool = False) -> bool:
  233. """向左移动"""
  234. return self.ptz_control(PTZCommand.LEFT, 0, speed, 0, stop)
  235. def move_right(self, speed: int = 4, stop: bool = False) -> bool:
  236. """向右移动"""
  237. return self.ptz_control(PTZCommand.RIGHT, 0, speed, 0, stop)
  238. def zoom_in(self, speed: int = 4, stop: bool = False) -> bool:
  239. """放大"""
  240. return self.ptz_control(PTZCommand.ZOOM_ADD, 0, speed, 0, stop)
  241. def zoom_out(self, speed: int = 4, stop: bool = False) -> bool:
  242. """缩小"""
  243. return self.ptz_control(PTZCommand.ZOOM_DEC, 0, speed, 0, stop)
  244. def stop_move(self) -> bool:
  245. """停止移动"""
  246. # 发送停止命令
  247. self.move_up(0, True)
  248. self.move_left(0, True)
  249. self.zoom_in(0, True)
  250. return True
  251. def goto_exact_position(self, pan: float, tilt: float, zoom: int) -> bool:
  252. """
  253. 三维精确定位
  254. Args:
  255. pan: 水平角度 (0-360度)
  256. tilt: 垂直角度 (-90到90度)
  257. zoom: 变倍 (1-128)
  258. Returns:
  259. 是否成功
  260. """
  261. param1 = int(pan * 10)
  262. param2 = int(tilt * 10)
  263. param3 = int(min(zoom, 128))
  264. print(f"[PTZCamera] goto_exact_position: pan={pan:.1f}° tilt={tilt:.1f}° zoom={zoom} → p1={param1} p2={param2} p3={param3}")
  265. result = self.ptz_control(PTZCommand.EXACTGOTO, param1, param2, param3)
  266. if result:
  267. with self.position_lock:
  268. self.current_position = PTZPosition(pan=pan, tilt=tilt, zoom=zoom)
  269. else:
  270. print(f"[PTZCamera] goto_exact_position FAILED!")
  271. return result
  272. def goto_preset(self, preset_id: int) -> bool:
  273. """
  274. 转到预置点
  275. Args:
  276. preset_id: 预置点ID
  277. Returns:
  278. 是否成功
  279. """
  280. return self.ptz_control(PTZCommand.POINT_GO, 0, preset_id, 0)
  281. def set_preset(self, preset_id: int) -> bool:
  282. """
  283. 设置预置点
  284. Args:
  285. preset_id: 预置点ID
  286. Returns:
  287. 是否成功
  288. """
  289. return self.ptz_control(PTZCommand.POINT_SET, 0, preset_id, 0)
  290. def clear_preset(self, preset_id: int) -> bool:
  291. """
  292. 清除预置点
  293. Args:
  294. preset_id: 预置点ID
  295. Returns:
  296. 是否成功
  297. """
  298. return self.ptz_control(PTZCommand.POINT_CLEAR, 0, preset_id, 0)
  299. def calculate_ptz_position(self, x_ratio: float, y_ratio: float,
  300. zoom: int = None) -> Tuple[float, float, int]:
  301. """
  302. 根据全景画面中的位置计算PTZ角度
  303. Args:
  304. x_ratio: X方向比例 (0-1)
  305. y_ratio: Y方向比例 (0-1)
  306. zoom: 变倍 (默认使用配置值)
  307. Returns:
  308. (pan, tilt, zoom) PTZ位置
  309. """
  310. if zoom is None:
  311. zoom = self.ptz_config['default_zoom']
  312. # 应用坐标偏移校准
  313. offset_x, offset_y = self.ptz_config['coordinate_offset']
  314. x_ratio = max(0, min(1, x_ratio + offset_x))
  315. y_ratio = max(0, min(1, y_ratio + offset_y))
  316. # 从配置获取视野参数
  317. pan_range = self.ptz_config.get('pan_range', (0, 180))
  318. tilt_range = self.ptz_config.get('tilt_range', (-45, 45))
  319. pan_center = self.ptz_config.get('pan_center', 90)
  320. tilt_center = self.ptz_config.get('tilt_center', 0)
  321. # 将画面比例转换为角度
  322. # x_ratio=0 对应 pan_range[0], x_ratio=1 对应 pan_range[1]
  323. pan = pan_range[0] + (pan_range[1] - pan_range[0]) * x_ratio
  324. # y_ratio=0.5 对应 tilt_center, y_ratio=0 对应 tilt_range[0], y_ratio=1 对应 tilt_range[1]
  325. tilt = tilt_range[0] + (tilt_range[1] - tilt_range[0]) * y_ratio
  326. # 应用方向修正
  327. # 1. 检查安装类型,吸顶安装需要反转tilt
  328. mount_type = self.ptz_config.get('mount_type', 'wall')
  329. tilt_flip = self.ptz_config.get('tilt_flip', False)
  330. # 吸顶安装时自动反转tilt方向
  331. if mount_type == 'ceiling' or tilt_flip:
  332. # 反转tilt:原来的正角度变负,负角度变正
  333. tilt = -tilt
  334. print(f"[PTZCamera] 吸顶安装,tilt方向修正: {-tilt} -> {tilt}")
  335. # 2. 应用pan方向修正
  336. pan_flip = self.ptz_config.get('pan_flip', False)
  337. if pan_flip:
  338. # 反转pan方向(180度偏移)
  339. pan = (pan + 180) % 360
  340. print(f"[PTZCamera] pan方向翻转: {(pan - 180) % 360} -> {pan}")
  341. return (pan, tilt, zoom)
  342. def move_to_target(self, x_ratio: float, y_ratio: float,
  343. zoom: int = None) -> bool:
  344. """
  345. 移动到目标位置
  346. Args:
  347. x_ratio: X方向比例 (0-1)
  348. y_ratio: Y方向比例 (0-1)
  349. zoom: 变倍
  350. Returns:
  351. 是否成功
  352. """
  353. pan, tilt, zoom = self.calculate_ptz_position(x_ratio, y_ratio, zoom)
  354. return self.goto_exact_position(pan, tilt, zoom)
  355. def track_target(self, x_ratio: float, y_ratio: float,
  356. zoom: int = None) -> bool:
  357. """
  358. 跟踪目标 - 与move_to_target相同,但可以添加跟踪特定逻辑
  359. Args:
  360. x_ratio: X方向比例
  361. y_ratio: Y方向比例
  362. zoom: 变倍
  363. Returns:
  364. 是否成功
  365. """
  366. return self.move_to_target(x_ratio, y_ratio, zoom)
  367. def get_current_position(self) -> PTZPosition:
  368. """获取当前位置"""
  369. with self.position_lock:
  370. return PTZPosition(
  371. pan=self.current_position.pan,
  372. tilt=self.current_position.tilt,
  373. zoom=self.current_position.zoom
  374. )
  375. def goto_and_confirm(self, pan: float, tilt: float, zoom: int,
  376. confirm_timeout: float = 1.0,
  377. get_frame_func=None) -> dict:
  378. """
  379. PTZ精确定位并确认到位
  380. Args:
  381. pan, tilt, zoom: 目标位置
  382. confirm_timeout: 确认超时秒数
  383. get_frame_func: 获取球机帧的函数,用于验证
  384. Returns:
  385. dict: {'success': bool, 'pan': float, 'tilt': float, 'zoom': int,
  386. 'frame_available': bool, 'elapsed_ms': float}
  387. """
  388. import time as _time
  389. start = _time.time()
  390. success = self.goto_exact_position(pan, tilt, zoom)
  391. result = {
  392. 'success': success,
  393. 'pan': pan,
  394. 'tilt': tilt,
  395. 'zoom': zoom,
  396. 'frame_available': False,
  397. 'elapsed_ms': (_time.time() - start) * 1000
  398. }
  399. if not success:
  400. return result
  401. # 等待球机物理移动到位
  402. time.sleep(0.2)
  403. # 如果有帧获取函数,验证球机画面
  404. if get_frame_func is not None:
  405. deadline = _time.time() + confirm_timeout
  406. while _time.time() < deadline:
  407. frame = get_frame_func()
  408. if frame is not None:
  409. result['frame_available'] = True
  410. break
  411. time.sleep(0.05)
  412. result['elapsed_ms'] = (_time.time() - start) * 1000
  413. return result
  414. def is_position_close(self, target_pan: float, target_tilt: float,
  415. threshold: float = 1.0) -> bool:
  416. """
  417. 检查当前位置是否接近目标位置
  418. Args:
  419. target_pan: 目标水平角度
  420. target_tilt: 目标垂直角度
  421. threshold: 角度容差(度)
  422. """
  423. current = self.get_current_position()
  424. pan_diff = abs(current.pan - target_pan)
  425. tilt_diff = abs(current.tilt - target_tilt)
  426. return pan_diff <= threshold and tilt_diff <= threshold
  427. class PTZController:
  428. """
  429. PTZ高级控制器
  430. 提供平滑移动、跟踪等功能
  431. """
  432. def __init__(self, ptz_camera: PTZCamera):
  433. """
  434. 初始化控制器
  435. Args:
  436. ptz_camera: PTZ摄像头实例
  437. """
  438. self.ptz = ptz_camera
  439. self.tracking = False
  440. self.tracking_thread = None
  441. self.target_position = None
  442. def smooth_move_to(self, pan: float, tilt: float, zoom: int,
  443. steps: int = 10, delay: float = 0.1) -> bool:
  444. """
  445. 平滑移动到目标位置
  446. Args:
  447. pan: 目标水平角度
  448. tilt: 目标垂直角度
  449. zoom: 目标变倍
  450. steps: 移动步数
  451. delay: 步间延迟
  452. Returns:
  453. 是否成功
  454. """
  455. current = self.ptz.get_current_position()
  456. # 计算步长
  457. pan_step = (pan - current.pan) / steps
  458. tilt_step = (tilt - current.tilt) / steps
  459. zoom_step = (zoom - current.zoom) / steps
  460. for i in range(1, steps + 1):
  461. current_pan = current.pan + pan_step * i
  462. current_tilt = current.tilt + tilt_step * i
  463. current_zoom = int(current.zoom + zoom_step * i)
  464. self.ptz.goto_exact_position(current_pan, current_tilt, current_zoom)
  465. time.sleep(delay)
  466. return True
  467. def start_tracking(self, get_target_func, update_interval: float = 0.1):
  468. """
  469. 开始跟踪
  470. Args:
  471. get_target_func: 获取目标位置的函数 (返回 x_ratio, y_ratio 或 None)
  472. update_interval: 更新间隔
  473. """
  474. self.tracking = True
  475. def tracking_worker():
  476. while self.tracking:
  477. try:
  478. target = get_target_func()
  479. if target:
  480. x_ratio, y_ratio = target
  481. self.ptz.track_target(x_ratio, y_ratio)
  482. time.sleep(update_interval)
  483. except Exception as e:
  484. print(f"跟踪错误: {e}")
  485. time.sleep(0.1)
  486. self.tracking_thread = threading.Thread(target=tracking_worker, daemon=True)
  487. self.tracking_thread.start()
  488. def stop_tracking(self):
  489. """停止跟踪"""
  490. self.tracking = False
  491. if self.tracking_thread:
  492. self.tracking_thread.join(timeout=1)
  493. self.tracking_thread = None
  494. def zoom_to_target_size(self, target_size: Tuple[int, int],
  495. frame_size: Tuple[int, int],
  496. min_zoom: int = 2, max_zoom: int = 20) -> int:
  497. """
  498. 根据目标大小计算合适的变倍
  499. Args:
  500. target_size: 目标尺寸 (width, height)
  501. frame_size: 画面尺寸 (width, height)
  502. min_zoom: 最小变倍
  503. max_zoom: 最大变倍
  504. Returns:
  505. 计算的变倍值
  506. """
  507. target_area = target_size[0] * target_size[1]
  508. frame_area = frame_size[0] * frame_size[1]
  509. # 目标占画面比例
  510. ratio = target_area / frame_area
  511. # 根据比例计算变倍 (目标占画面30%时变倍为1)
  512. if ratio > 0:
  513. ideal_zoom = math.sqrt(0.3 / ratio)
  514. zoom = int(max(min_zoom, min(max_zoom, ideal_zoom)))
  515. else:
  516. zoom = min_zoom
  517. return zoom