mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-06-25 22:01:51 +03:00
* ensure motion masks from source camera are copied to replay * stop polling debug_replay/status after live_ready * use vod for constructing replay clips
291 lines
9.0 KiB
Python
291 lines
9.0 KiB
Python
"""Debug replay API endpoints."""
|
|
|
|
import asyncio
|
|
import logging
|
|
from datetime import datetime
|
|
|
|
from fastapi import APIRouter, Depends, Request
|
|
from fastapi.responses import JSONResponse
|
|
from peewee import DoesNotExist
|
|
from pydantic import BaseModel, Field
|
|
|
|
from frigate.api.auth import require_role
|
|
from frigate.api.defs.tags import Tags
|
|
from frigate.jobs.debug_replay import (
|
|
ExportDebugReplaySource,
|
|
RecordingDebugReplaySource,
|
|
start_debug_replay_job,
|
|
)
|
|
from frigate.models import Export
|
|
from frigate.util.services import get_video_properties
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
router = APIRouter(tags=[Tags.app])
|
|
|
|
|
|
class DebugReplayStartBody(BaseModel):
|
|
"""Request body for starting a debug replay session."""
|
|
|
|
camera: str = Field(title="Source camera name")
|
|
start_time: float = Field(title="Start timestamp")
|
|
end_time: float = Field(title="End timestamp")
|
|
|
|
|
|
class DebugReplayStartFromExportBody(BaseModel):
|
|
"""Request body for starting a debug replay session from an export."""
|
|
|
|
export_id: str = Field(title="Export id")
|
|
|
|
|
|
class DebugReplayStartResponse(BaseModel):
|
|
"""Response for starting a debug replay session."""
|
|
|
|
success: bool
|
|
replay_camera: str
|
|
job_id: str
|
|
|
|
|
|
class DebugReplayStatusResponse(BaseModel):
|
|
"""Response for debug replay status.
|
|
|
|
Returns only session-presence fields. Startup progress and error
|
|
details flow through the job_state WebSocket topic via the
|
|
debug_replay job (see frigate.jobs.debug_replay); the
|
|
Replay page subscribes there with useJobStatus("debug_replay").
|
|
"""
|
|
|
|
active: bool
|
|
replay_camera: str | None = None
|
|
source_camera: str | None = None
|
|
start_time: float | None = None
|
|
end_time: float | None = None
|
|
live_ready: bool = False
|
|
|
|
|
|
class DebugReplayStopResponse(BaseModel):
|
|
"""Response for stopping a debug replay session."""
|
|
|
|
success: bool
|
|
|
|
|
|
@router.post(
|
|
"/debug_replay/start",
|
|
response_model=DebugReplayStartResponse,
|
|
status_code=202,
|
|
responses={
|
|
400: {"description": "Invalid camera, time range, or no recordings"},
|
|
409: {"description": "A replay session is already active"},
|
|
},
|
|
dependencies=[Depends(require_role(["admin"]))],
|
|
summary="Start debug replay",
|
|
description="Start a debug replay session from camera recordings. Returns "
|
|
"immediately while clip generation runs as a background job; subscribe "
|
|
"to the 'debug_replay' job_state WS topic to track progress.",
|
|
)
|
|
async def start_debug_replay(request: Request, body: DebugReplayStartBody):
|
|
"""Start a debug replay session asynchronously."""
|
|
replay_manager = request.app.replay_manager
|
|
internal_port = request.app.frigate_config.networking.listen.internal
|
|
if type(internal_port) is str:
|
|
internal_port = int(internal_port.split(":")[-1])
|
|
|
|
source = RecordingDebugReplaySource(
|
|
source_camera=body.camera,
|
|
start_ts=body.start_time,
|
|
end_ts=body.end_time,
|
|
internal_port=internal_port,
|
|
)
|
|
|
|
try:
|
|
job_id = await asyncio.to_thread(
|
|
start_debug_replay_job,
|
|
source=source,
|
|
frigate_config=request.app.frigate_config,
|
|
config_publisher=request.app.config_publisher,
|
|
replay_manager=replay_manager,
|
|
)
|
|
except RuntimeError:
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "A replay session is already active",
|
|
},
|
|
status_code=409,
|
|
)
|
|
except ValueError:
|
|
logger.exception("Rejected debug replay start request")
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "Invalid debug replay parameters",
|
|
},
|
|
status_code=400,
|
|
)
|
|
|
|
return JSONResponse(
|
|
content={
|
|
"success": True,
|
|
"replay_camera": replay_manager.replay_camera_name,
|
|
"job_id": job_id,
|
|
},
|
|
status_code=202,
|
|
)
|
|
|
|
|
|
@router.post(
|
|
"/debug_replay/start_from_export",
|
|
response_model=DebugReplayStartResponse,
|
|
status_code=202,
|
|
responses={
|
|
400: {"description": "Invalid export, time range, or no recordings"},
|
|
404: {"description": "Export not found"},
|
|
409: {"description": "A replay session is already active"},
|
|
},
|
|
dependencies=[Depends(require_role(["admin"]))],
|
|
summary="Start debug replay from an export",
|
|
description="Start a debug replay session covering an existing export's "
|
|
"time range. The end time is derived from the export's video duration.",
|
|
)
|
|
async def start_debug_replay_from_export(
|
|
request: Request, body: DebugReplayStartFromExportBody
|
|
):
|
|
"""Start a debug replay session from an existing export."""
|
|
try:
|
|
export: Export = Export.get(Export.id == body.export_id)
|
|
except DoesNotExist:
|
|
return JSONResponse(
|
|
content={"success": False, "message": "Export not found"},
|
|
status_code=404,
|
|
)
|
|
|
|
properties = await get_video_properties(
|
|
request.app.frigate_config.ffmpeg, export.video_path, get_duration=True
|
|
)
|
|
duration = properties.get("duration", -1)
|
|
|
|
if duration is None or duration <= 0:
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "Could not determine export duration",
|
|
},
|
|
status_code=400,
|
|
)
|
|
|
|
replay_manager = request.app.replay_manager
|
|
source = ExportDebugReplaySource(export=export, duration=float(duration))
|
|
|
|
try:
|
|
job_id = await asyncio.to_thread(
|
|
start_debug_replay_job,
|
|
source=source,
|
|
frigate_config=request.app.frigate_config,
|
|
config_publisher=request.app.config_publisher,
|
|
replay_manager=replay_manager,
|
|
)
|
|
except RuntimeError:
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "A replay session is already active",
|
|
},
|
|
status_code=409,
|
|
)
|
|
except ValueError:
|
|
logger.exception("Rejected debug replay start request")
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "Invalid debug replay parameters",
|
|
},
|
|
status_code=400,
|
|
)
|
|
|
|
return JSONResponse(
|
|
content={
|
|
"success": True,
|
|
"replay_camera": replay_manager.replay_camera_name,
|
|
"job_id": job_id,
|
|
},
|
|
status_code=202,
|
|
)
|
|
|
|
|
|
@router.get(
|
|
"/debug_replay/status",
|
|
response_model=DebugReplayStatusResponse,
|
|
dependencies=[Depends(require_role(["admin"]))],
|
|
summary="Get debug replay status",
|
|
description="Get the status of the current debug replay session.",
|
|
)
|
|
def get_debug_replay_status(request: Request):
|
|
"""Get the current replay session status."""
|
|
replay_manager = request.app.replay_manager
|
|
|
|
live_ready = False
|
|
replay_camera = replay_manager.replay_camera_name
|
|
|
|
if replay_manager.active and replay_camera:
|
|
frame_processor = request.app.detected_frames_processor
|
|
frame = (
|
|
frame_processor.get_current_frame(replay_camera)
|
|
if frame_processor is not None
|
|
else None
|
|
)
|
|
|
|
if frame is not None:
|
|
frame_time = frame_processor.get_current_frame_time(replay_camera)
|
|
camera_config = request.app.frigate_config.cameras.get(replay_camera)
|
|
retry_interval = 10.0
|
|
|
|
if camera_config is not None:
|
|
retry_interval = float(camera_config.ffmpeg.retry_interval or 10)
|
|
|
|
live_ready = datetime.now().timestamp() <= frame_time + retry_interval
|
|
|
|
return DebugReplayStatusResponse(
|
|
active=replay_manager.active,
|
|
replay_camera=replay_camera,
|
|
source_camera=replay_manager.source_camera,
|
|
start_time=replay_manager.start_ts,
|
|
end_time=replay_manager.end_ts,
|
|
live_ready=live_ready,
|
|
)
|
|
|
|
|
|
@router.post(
|
|
"/debug_replay/stop",
|
|
response_model=DebugReplayStopResponse,
|
|
dependencies=[Depends(require_role(["admin"]))],
|
|
summary="Stop debug replay",
|
|
description="Stop the active debug replay session and clean up all artifacts.",
|
|
)
|
|
async def stop_debug_replay(request: Request):
|
|
"""Stop the active replay session."""
|
|
replay_manager = request.app.replay_manager
|
|
|
|
if not replay_manager.active:
|
|
return JSONResponse(
|
|
content={"success": False, "message": "No active replay session"},
|
|
status_code=400,
|
|
)
|
|
|
|
try:
|
|
await asyncio.to_thread(
|
|
replay_manager.stop,
|
|
frigate_config=request.app.frigate_config,
|
|
config_publisher=request.app.config_publisher,
|
|
)
|
|
except (ValueError, RuntimeError, OSError) as e:
|
|
logger.error("Error stopping replay: %s", e)
|
|
return JSONResponse(
|
|
content={
|
|
"success": False,
|
|
"message": "Failed to stop replay session due to an internal error.",
|
|
},
|
|
status_code=500,
|
|
)
|
|
|
|
return DebugReplayStopResponse(success=True)
|