mirror of
https://github.com/blakeblackshear/frigate.git
synced 2025-09-05 17:51:36 +02:00
Autotracking improvements (#19873)
* Use asyncio lock when checking camera status get_camera_status() can be called during normal autotracking movement and from routine camera_maintenance(). Some cameras cause one of the status calls to hang, which then subsequently hangs autotracking. A lock serializes access and prevents the hang. * use while loop in camera_maintenance for status check some cameras seem to take a little bit to update their status, don't assume the first call shows the motor has stopped
This commit is contained in:
parent
e9dc30235b
commit
f7ed8b4cab
@ -1462,7 +1462,7 @@ class PtzAutoTracker:
|
|||||||
if not self.autotracker_init[camera]:
|
if not self.autotracker_init[camera]:
|
||||||
self._autotracker_setup(self.config.cameras[camera], camera)
|
self._autotracker_setup(self.config.cameras[camera], camera)
|
||||||
# regularly update camera status
|
# regularly update camera status
|
||||||
if not self.ptz_metrics[camera].motor_stopped.is_set():
|
while not self.ptz_metrics[camera].motor_stopped.is_set():
|
||||||
await self.onvif.get_camera_status(camera)
|
await self.onvif.get_camera_status(camera)
|
||||||
|
|
||||||
# return to preset if tracking is over
|
# return to preset if tracking is over
|
||||||
@ -1491,7 +1491,7 @@ class PtzAutoTracker:
|
|||||||
)
|
)
|
||||||
|
|
||||||
# update stored zoom level from preset
|
# update stored zoom level from preset
|
||||||
if not self.ptz_metrics[camera].motor_stopped.is_set():
|
while not self.ptz_metrics[camera].motor_stopped.is_set():
|
||||||
await self.onvif.get_camera_status(camera)
|
await self.onvif.get_camera_status(camera)
|
||||||
|
|
||||||
self.ptz_metrics[camera].tracking_active.clear()
|
self.ptz_metrics[camera].tracking_active.clear()
|
||||||
|
@ -48,6 +48,8 @@ class OnvifController:
|
|||||||
self.config = config
|
self.config = config
|
||||||
self.ptz_metrics = ptz_metrics
|
self.ptz_metrics = ptz_metrics
|
||||||
|
|
||||||
|
self.status_locks: dict[str, asyncio.Lock] = {}
|
||||||
|
|
||||||
# Create a dedicated event loop and run it in a separate thread
|
# Create a dedicated event loop and run it in a separate thread
|
||||||
self.loop = asyncio.new_event_loop()
|
self.loop = asyncio.new_event_loop()
|
||||||
self.loop_thread = threading.Thread(target=self._run_event_loop, daemon=True)
|
self.loop_thread = threading.Thread(target=self._run_event_loop, daemon=True)
|
||||||
@ -59,6 +61,7 @@ class OnvifController:
|
|||||||
continue
|
continue
|
||||||
if cam.onvif.host:
|
if cam.onvif.host:
|
||||||
self.camera_configs[cam_name] = cam
|
self.camera_configs[cam_name] = cam
|
||||||
|
self.status_locks[cam_name] = asyncio.Lock()
|
||||||
|
|
||||||
asyncio.run_coroutine_threadsafe(self._init_cameras(), self.loop)
|
asyncio.run_coroutine_threadsafe(self._init_cameras(), self.loop)
|
||||||
|
|
||||||
@ -764,105 +767,110 @@ class OnvifController:
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
async def get_camera_status(self, camera_name: str) -> None:
|
async def get_camera_status(self, camera_name: str) -> None:
|
||||||
if camera_name not in self.cams.keys():
|
async with self.status_locks[camera_name]:
|
||||||
logger.error(f"ONVIF is not configured for {camera_name}")
|
if camera_name not in self.cams.keys():
|
||||||
return
|
logger.error(f"ONVIF is not configured for {camera_name}")
|
||||||
|
|
||||||
if not self.cams[camera_name]["init"]:
|
|
||||||
if not await self._init_onvif(camera_name):
|
|
||||||
return
|
return
|
||||||
|
|
||||||
status_request = self.cams[camera_name]["status_request"]
|
if not self.cams[camera_name]["init"]:
|
||||||
try:
|
if not await self._init_onvif(camera_name):
|
||||||
status = await self.cams[camera_name]["ptz"].GetStatus(status_request)
|
return
|
||||||
except Exception:
|
|
||||||
pass # We're unsupported, that'll be reported in the next check.
|
|
||||||
|
|
||||||
try:
|
status_request = self.cams[camera_name]["status_request"]
|
||||||
pan_tilt_status = getattr(status.MoveStatus, "PanTilt", None)
|
try:
|
||||||
zoom_status = getattr(status.MoveStatus, "Zoom", None)
|
status = await self.cams[camera_name]["ptz"].GetStatus(status_request)
|
||||||
|
except Exception:
|
||||||
|
pass # We're unsupported, that'll be reported in the next check.
|
||||||
|
|
||||||
# if it's not an attribute, see if MoveStatus even exists in the status result
|
try:
|
||||||
if pan_tilt_status is None:
|
pan_tilt_status = getattr(status.MoveStatus, "PanTilt", None)
|
||||||
pan_tilt_status = getattr(status, "MoveStatus", None)
|
zoom_status = getattr(status.MoveStatus, "Zoom", None)
|
||||||
|
|
||||||
# we're unsupported
|
# if it's not an attribute, see if MoveStatus even exists in the status result
|
||||||
if pan_tilt_status is None or pan_tilt_status not in [
|
if pan_tilt_status is None:
|
||||||
"IDLE",
|
pan_tilt_status = getattr(status, "MoveStatus", None)
|
||||||
"MOVING",
|
|
||||||
]:
|
# we're unsupported
|
||||||
raise Exception
|
if pan_tilt_status is None or pan_tilt_status not in [
|
||||||
except Exception:
|
"IDLE",
|
||||||
logger.warning(
|
"MOVING",
|
||||||
f"Camera {camera_name} does not support the ONVIF GetStatus method. Autotracking will not function correctly and must be disabled in your config."
|
]:
|
||||||
|
raise Exception
|
||||||
|
except Exception:
|
||||||
|
logger.warning(
|
||||||
|
f"Camera {camera_name} does not support the ONVIF GetStatus method. Autotracking will not function correctly and must be disabled in your config."
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
logger.debug(
|
||||||
|
f"{camera_name}: Pan/tilt status: {pan_tilt_status}, Zoom status: {zoom_status}"
|
||||||
)
|
)
|
||||||
return
|
|
||||||
|
|
||||||
logger.debug(
|
if pan_tilt_status == "IDLE" and (
|
||||||
f"{camera_name}: Pan/tilt status: {pan_tilt_status}, Zoom status: {zoom_status}"
|
zoom_status is None or zoom_status == "IDLE"
|
||||||
)
|
):
|
||||||
|
self.cams[camera_name]["active"] = False
|
||||||
|
if not self.ptz_metrics[camera_name].motor_stopped.is_set():
|
||||||
|
self.ptz_metrics[camera_name].motor_stopped.set()
|
||||||
|
|
||||||
if pan_tilt_status == "IDLE" and (zoom_status is None or zoom_status == "IDLE"):
|
logger.debug(
|
||||||
self.cams[camera_name]["active"] = False
|
f"{camera_name}: PTZ stop time: {self.ptz_metrics[camera_name].frame_time.value}"
|
||||||
if not self.ptz_metrics[camera_name].motor_stopped.is_set():
|
)
|
||||||
self.ptz_metrics[camera_name].motor_stopped.set()
|
|
||||||
|
|
||||||
|
self.ptz_metrics[camera_name].stop_time.value = self.ptz_metrics[
|
||||||
|
camera_name
|
||||||
|
].frame_time.value
|
||||||
|
else:
|
||||||
|
self.cams[camera_name]["active"] = True
|
||||||
|
if self.ptz_metrics[camera_name].motor_stopped.is_set():
|
||||||
|
self.ptz_metrics[camera_name].motor_stopped.clear()
|
||||||
|
|
||||||
|
logger.debug(
|
||||||
|
f"{camera_name}: PTZ start time: {self.ptz_metrics[camera_name].frame_time.value}"
|
||||||
|
)
|
||||||
|
|
||||||
|
self.ptz_metrics[camera_name].start_time.value = self.ptz_metrics[
|
||||||
|
camera_name
|
||||||
|
].frame_time.value
|
||||||
|
self.ptz_metrics[camera_name].stop_time.value = 0
|
||||||
|
|
||||||
|
if (
|
||||||
|
self.config.cameras[camera_name].onvif.autotracking.zooming
|
||||||
|
!= ZoomingModeEnum.disabled
|
||||||
|
):
|
||||||
|
# store absolute zoom level as 0 to 1 interpolated from the values of the camera
|
||||||
|
self.ptz_metrics[camera_name].zoom_level.value = numpy.interp(
|
||||||
|
round(status.Position.Zoom.x, 2),
|
||||||
|
[
|
||||||
|
self.cams[camera_name]["absolute_zoom_range"]["XRange"]["Min"],
|
||||||
|
self.cams[camera_name]["absolute_zoom_range"]["XRange"]["Max"],
|
||||||
|
],
|
||||||
|
[0, 1],
|
||||||
|
)
|
||||||
logger.debug(
|
logger.debug(
|
||||||
f"{camera_name}: PTZ stop time: {self.ptz_metrics[camera_name].frame_time.value}"
|
f"{camera_name}: Camera zoom level: {self.ptz_metrics[camera_name].zoom_level.value}"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# some hikvision cams won't update MoveStatus, so warn if it hasn't changed
|
||||||
|
if (
|
||||||
|
not self.ptz_metrics[camera_name].motor_stopped.is_set()
|
||||||
|
and not self.ptz_metrics[camera_name].reset.is_set()
|
||||||
|
and self.ptz_metrics[camera_name].start_time.value != 0
|
||||||
|
and self.ptz_metrics[camera_name].frame_time.value
|
||||||
|
> (self.ptz_metrics[camera_name].start_time.value + 10)
|
||||||
|
and self.ptz_metrics[camera_name].stop_time.value == 0
|
||||||
|
):
|
||||||
|
logger.debug(
|
||||||
|
f"Start time: {self.ptz_metrics[camera_name].start_time.value}, Stop time: {self.ptz_metrics[camera_name].stop_time.value}, Frame time: {self.ptz_metrics[camera_name].frame_time.value}"
|
||||||
|
)
|
||||||
|
# set the stop time so we don't come back into this again and spam the logs
|
||||||
self.ptz_metrics[camera_name].stop_time.value = self.ptz_metrics[
|
self.ptz_metrics[camera_name].stop_time.value = self.ptz_metrics[
|
||||||
camera_name
|
camera_name
|
||||||
].frame_time.value
|
].frame_time.value
|
||||||
else:
|
logger.warning(
|
||||||
self.cams[camera_name]["active"] = True
|
f"Camera {camera_name} is still in ONVIF 'MOVING' status."
|
||||||
if self.ptz_metrics[camera_name].motor_stopped.is_set():
|
|
||||||
self.ptz_metrics[camera_name].motor_stopped.clear()
|
|
||||||
|
|
||||||
logger.debug(
|
|
||||||
f"{camera_name}: PTZ start time: {self.ptz_metrics[camera_name].frame_time.value}"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
self.ptz_metrics[camera_name].start_time.value = self.ptz_metrics[
|
|
||||||
camera_name
|
|
||||||
].frame_time.value
|
|
||||||
self.ptz_metrics[camera_name].stop_time.value = 0
|
|
||||||
|
|
||||||
if (
|
|
||||||
self.config.cameras[camera_name].onvif.autotracking.zooming
|
|
||||||
!= ZoomingModeEnum.disabled
|
|
||||||
):
|
|
||||||
# store absolute zoom level as 0 to 1 interpolated from the values of the camera
|
|
||||||
self.ptz_metrics[camera_name].zoom_level.value = numpy.interp(
|
|
||||||
round(status.Position.Zoom.x, 2),
|
|
||||||
[
|
|
||||||
self.cams[camera_name]["absolute_zoom_range"]["XRange"]["Min"],
|
|
||||||
self.cams[camera_name]["absolute_zoom_range"]["XRange"]["Max"],
|
|
||||||
],
|
|
||||||
[0, 1],
|
|
||||||
)
|
|
||||||
logger.debug(
|
|
||||||
f"{camera_name}: Camera zoom level: {self.ptz_metrics[camera_name].zoom_level.value}"
|
|
||||||
)
|
|
||||||
|
|
||||||
# some hikvision cams won't update MoveStatus, so warn if it hasn't changed
|
|
||||||
if (
|
|
||||||
not self.ptz_metrics[camera_name].motor_stopped.is_set()
|
|
||||||
and not self.ptz_metrics[camera_name].reset.is_set()
|
|
||||||
and self.ptz_metrics[camera_name].start_time.value != 0
|
|
||||||
and self.ptz_metrics[camera_name].frame_time.value
|
|
||||||
> (self.ptz_metrics[camera_name].start_time.value + 10)
|
|
||||||
and self.ptz_metrics[camera_name].stop_time.value == 0
|
|
||||||
):
|
|
||||||
logger.debug(
|
|
||||||
f"Start time: {self.ptz_metrics[camera_name].start_time.value}, Stop time: {self.ptz_metrics[camera_name].stop_time.value}, Frame time: {self.ptz_metrics[camera_name].frame_time.value}"
|
|
||||||
)
|
|
||||||
# set the stop time so we don't come back into this again and spam the logs
|
|
||||||
self.ptz_metrics[camera_name].stop_time.value = self.ptz_metrics[
|
|
||||||
camera_name
|
|
||||||
].frame_time.value
|
|
||||||
logger.warning(f"Camera {camera_name} is still in ONVIF 'MOVING' status.")
|
|
||||||
|
|
||||||
def close(self) -> None:
|
def close(self) -> None:
|
||||||
"""Gracefully shut down the ONVIF controller."""
|
"""Gracefully shut down the ONVIF controller."""
|
||||||
if not hasattr(self, "loop") or self.loop.is_closed():
|
if not hasattr(self, "loop") or self.loop.is_closed():
|
||||||
|
Loading…
Reference in New Issue
Block a user