mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-03-10 10:33:11 +03:00
* use react-jsonschema-form for UI config * don't use properties wrapper when generating config i18n json * configure for full i18n support * section fields * add descriptions to all fields for i18n * motion i18n * fix nullable fields * sanitize internal fields * add switches widgets and use friendly names * fix nullable schema entries * ensure update_topic is added to api calls this needs further backend implementation to work correctly * add global sections, camera config overrides, and reset button * i18n * add reset logic to global config view * tweaks * fix sections and live validation * fix validation for schema objects that can be null * generic and custom per-field validation * improve generic error validation messages * remove show advanced fields switch * tweaks * use shadcn theme * fix array field template * i18n tweaks * remove collapsible around root section * deep merge schema for advanced fields * add array field item template and fix ffmpeg section * add missing i18n keys * tweaks * comment out api call for testing * add config groups as a separate i18n namespace * add descriptions to all pydantic fields * make titles more concise * new titles as i18n * update i18n config generation script to use json schema * tweaks * tweaks * rebase * clean up * form tweaks * add wildcards and fix object filter fields * add field template for additionalproperties schema objects * improve typing * add section description from schema and clarify global vs camera level descriptions * separate and consolidate global and camera i18n namespaces * clean up now obsolete namespaces * tweaks * refactor sections and overrides * add ability to render components before and after fields * fix titles * chore(sections): remove legacy single-section components replaced by template * refactor configs to use individual files with a template * fix review description * apply hidden fields after ui schema * move util * remove unused i18n * clean up error messages * fix fast refresh * add custom validation and use it for ffmpeg input roles * update nav tree * remove unused * re-add override and modified indicators * mark pending changes and add confirmation dialog for resets * fix red unsaved dot * tweaks * add docs links, readonly keys, and restart required per field * add special case and comments for global motion section * add section form special cases * combine review sections * tweaks * add audio labels endpoint * add audio label switches and input to filter list * fix type * remove key from config when resetting to default/global * don't show description for new key/val fields * tweaks * spacing tweaks * add activity indicator and scrollbar tweaks * add docs to filter fields * wording changes * fix global ffmpeg section * add review classification zones to review form * add backend endpoint and frontend widget for ffmpeg presets and manual args * improve wording * hide descriptions for additional properties arrays * add warning log about incorrectly nested model config * spacing and language tweaks * fix i18n keys * networking section docs and description * small wording tweaks * add layout grid field * refactor with shared utilities * field order * add individual detectors to schema add detector titles and descriptions (docstrings in pydantic are used for descriptions) and add i18n keys to globals * clean up detectors section and i18n * don't save model config back to yaml when saving detectors * add full detectors config to api model dump works around the way we use detector plugins so we can have the full detector config for the frontend * add restart button to toast when restart is required * add ui option to remove inner cards * fix buttons * section tweaks * don't zoom into text on mobile * make buttons sticky at bottom of sections * small tweaks * highlight label of changed fields * add null to enum list when unwrapping * refactor to shared utils and add save all button * add undo all button * add RJSF to dictionary * consolidate utils * preserve form data when changing cameras * add mono fonts * add popover to show what fields will be saved * fix mobile menu not re-rendering with unsaved dots * tweaks * fix logger and env vars config section saving use escaped periods in keys to retain them in the config file (eg "frigate.embeddings") * add timezone widget * role map field with validation * fix validation for model section * add another hidden field * add footer message for required restart * use rjsf for notifications view * fix config saving * add replace rules field * default column layout and add field sizing * clean up field template * refactor profile settings to match rjsf forms * tweaks * refactor frigate+ view and make tweaks to sections * show frigate+ model info in detection model settings when using a frigate+ model * update restartRequired for all fields * fix restart fields * tweaks and add ability enable disabled cameras more backend changes required * require restart when enabling camera that is disabled in config * disable save when form is invalid * refactor ffmpeg section for readability * change label * clean up camera inputs fields * misc tweaks to ffmpeg section - add raw paths endpoint to ensure credentials get saved - restart required tooltip * maintenance settings tweaks * don't mutate with lodash * fix description re-rendering for nullable object fields * hide reindex field * update rjsf * add frigate+ description to settings pane * disable save all when any section is invalid * show translated field name in validation error pane * clean up * remove unused * fix genai merge * fix genai
160 lines
5.1 KiB
Python
160 lines
5.1 KiB
Python
from enum import Enum
|
|
from typing import Union
|
|
|
|
from pydantic import Field, field_validator
|
|
|
|
from frigate.const import DEFAULT_FFMPEG_VERSION, INCLUDED_FFMPEG_VERSIONS
|
|
|
|
from ..base import FrigateBaseModel
|
|
from ..env import EnvString
|
|
|
|
__all__ = [
|
|
"CameraFfmpegConfig",
|
|
"CameraInput",
|
|
"CameraRoleEnum",
|
|
"FfmpegConfig",
|
|
"FfmpegOutputArgsConfig",
|
|
]
|
|
|
|
# Note: Setting threads to less than 2 caused several issues with recording segments
|
|
# https://github.com/blakeblackshear/frigate/issues/5659
|
|
FFMPEG_GLOBAL_ARGS_DEFAULT = ["-hide_banner", "-loglevel", "warning", "-threads", "2"]
|
|
FFMPEG_INPUT_ARGS_DEFAULT = "preset-rtsp-generic"
|
|
|
|
RECORD_FFMPEG_OUTPUT_ARGS_DEFAULT = "preset-record-generic-audio-aac"
|
|
DETECT_FFMPEG_OUTPUT_ARGS_DEFAULT = [
|
|
"-threads",
|
|
"2",
|
|
"-f",
|
|
"rawvideo",
|
|
"-pix_fmt",
|
|
"yuv420p",
|
|
]
|
|
|
|
|
|
class FfmpegOutputArgsConfig(FrigateBaseModel):
|
|
detect: Union[str, list[str]] = Field(
|
|
default=DETECT_FFMPEG_OUTPUT_ARGS_DEFAULT,
|
|
title="Detect output arguments",
|
|
description="Default output arguments for detect role streams.",
|
|
)
|
|
record: Union[str, list[str]] = Field(
|
|
default=RECORD_FFMPEG_OUTPUT_ARGS_DEFAULT,
|
|
title="Record output arguments",
|
|
description="Default output arguments for record role streams.",
|
|
)
|
|
|
|
|
|
class FfmpegConfig(FrigateBaseModel):
|
|
path: str = Field(
|
|
default="default",
|
|
title="FFmpeg path",
|
|
description='Path to the FFmpeg binary to use or a version alias ("5.0" or "7.0").',
|
|
)
|
|
global_args: Union[str, list[str]] = Field(
|
|
default=FFMPEG_GLOBAL_ARGS_DEFAULT,
|
|
title="FFmpeg global arguments",
|
|
description="Global arguments passed to FFmpeg processes.",
|
|
)
|
|
hwaccel_args: Union[str, list[str]] = Field(
|
|
default="auto",
|
|
title="Hardware acceleration arguments",
|
|
description="Hardware acceleration arguments for FFmpeg. Provider-specific presets are recommended.",
|
|
)
|
|
input_args: Union[str, list[str]] = Field(
|
|
default=FFMPEG_INPUT_ARGS_DEFAULT,
|
|
title="Input arguments",
|
|
description="Input arguments applied to FFmpeg input streams.",
|
|
)
|
|
output_args: FfmpegOutputArgsConfig = Field(
|
|
default_factory=FfmpegOutputArgsConfig,
|
|
title="Output arguments",
|
|
description="Default output arguments used for different FFmpeg roles such as detect and record.",
|
|
)
|
|
retry_interval: float = Field(
|
|
default=10.0,
|
|
title="FFmpeg retry time",
|
|
description="Seconds to wait before attempting to reconnect a camera stream after failure. Default is 10.",
|
|
gt=0.0,
|
|
)
|
|
apple_compatibility: bool = Field(
|
|
default=False,
|
|
title="Apple compatibility",
|
|
description="Enable HEVC tagging for better Apple player compatibility when recording H.265.",
|
|
)
|
|
gpu: int = Field(
|
|
default=0,
|
|
title="GPU index",
|
|
description="Default GPU index used for hardware acceleration if available.",
|
|
)
|
|
|
|
@property
|
|
def ffmpeg_path(self) -> str:
|
|
if self.path == "default":
|
|
return f"/usr/lib/ffmpeg/{DEFAULT_FFMPEG_VERSION}/bin/ffmpeg"
|
|
elif self.path in INCLUDED_FFMPEG_VERSIONS:
|
|
return f"/usr/lib/ffmpeg/{self.path}/bin/ffmpeg"
|
|
else:
|
|
return f"{self.path}/bin/ffmpeg"
|
|
|
|
@property
|
|
def ffprobe_path(self) -> str:
|
|
if self.path == "default":
|
|
return f"/usr/lib/ffmpeg/{DEFAULT_FFMPEG_VERSION}/bin/ffprobe"
|
|
elif self.path in INCLUDED_FFMPEG_VERSIONS:
|
|
return f"/usr/lib/ffmpeg/{self.path}/bin/ffprobe"
|
|
else:
|
|
return f"{self.path}/bin/ffprobe"
|
|
|
|
|
|
class CameraRoleEnum(str, Enum):
|
|
audio = "audio"
|
|
record = "record"
|
|
detect = "detect"
|
|
|
|
|
|
class CameraInput(FrigateBaseModel):
|
|
path: EnvString = Field(
|
|
title="Input path",
|
|
description="Camera input stream URL or path.",
|
|
)
|
|
roles: list[CameraRoleEnum] = Field(
|
|
title="Input roles",
|
|
description="Roles for this input stream.",
|
|
)
|
|
global_args: Union[str, list[str]] = Field(
|
|
default_factory=list,
|
|
title="FFmpeg global arguments",
|
|
description="FFmpeg global arguments for this input stream.",
|
|
)
|
|
hwaccel_args: Union[str, list[str]] = Field(
|
|
default_factory=list,
|
|
title="Hardware acceleration arguments",
|
|
description="Hardware acceleration arguments for this input stream.",
|
|
)
|
|
input_args: Union[str, list[str]] = Field(
|
|
default_factory=list,
|
|
title="Input arguments",
|
|
description="Input arguments specific to this stream.",
|
|
)
|
|
|
|
|
|
class CameraFfmpegConfig(FfmpegConfig):
|
|
inputs: list[CameraInput] = Field(
|
|
title="Camera inputs",
|
|
description="List of input stream definitions (paths and roles) for this camera.",
|
|
)
|
|
|
|
@field_validator("inputs")
|
|
@classmethod
|
|
def validate_roles(cls, v):
|
|
roles = [role for input in v for role in input.roles]
|
|
|
|
if len(roles) != len(set(roles)):
|
|
raise ValueError("Each input role may only be used once.")
|
|
|
|
if "detect" not in roles:
|
|
raise ValueError("The detect role is required.")
|
|
|
|
return v
|