frigate-container/frigate-container.nix

249 lines
6.4 KiB
Nix
Raw Permalink Normal View History

2023-11-26 14:43:45 -08:00
{ config, lib, pkgs, ... }:
with lib;
let
cfg = config.services.frigateContainer;
makeEnvFile = envVars:
let
envLines =
mapAttrsToList (var: val: ''${var}="${toString val}"'') envVars;
in pkgs.writeText "envFile" (concatStringsSep "\n" envLines);
hostSecrets = config.fudo.secrets.host-secrets."${config.instance.hostname}";
removeNewline = removeSuffix "\n";
2023-11-29 15:01:42 -08:00
flatMapAttrs' = f: attrs: listToAttrs (mapConcat f attrs);
2023-11-26 17:38:26 -08:00
frigateCfg = let
content = builtins.toJSON {
mqtt = {
enabled = true;
inherit (cfg.mqtt) host port user;
password = "{FRIGATE_MQTT_PASSWORD}";
2023-11-26 14:43:45 -08:00
};
2023-11-26 17:38:26 -08:00
logger.default = cfg.log-level;
2023-11-26 21:19:49 -08:00
ffmpeg.hwaccel_args = optional (cfg.hwaccel != null) cfg.hwaccel;
2023-11-26 17:38:26 -08:00
cameras = mapAttrs' (_: camOpts:
nameValuePair camOpts.name {
2023-11-29 16:09:19 -08:00
birdseye.mode = if camOpts.default then "continuous" else "objects";
2023-11-26 17:38:26 -08:00
ffmpeg.inputs = [
{
path = camOpts.streams.high;
roles = [ "record" ];
}
{
path = camOpts.streams.low;
roles = [ "detect" ];
}
];
}) cfg.cameras;
2023-11-29 15:18:24 -08:00
go2rtc.streams = mapAttrs'
(_: camOpts: nameValuePair camOpts.name [ camOpts.streams.low ])
2023-11-29 15:01:42 -08:00
cfg.cameras;
2023-11-26 17:38:26 -08:00
detectors = cfg.detectors;
record = {
enabled = true;
retain = {
days = cfg.retention.default;
mode = "motion";
};
events.retain = {
default = cfg.retention.events;
mode = "active_objects";
objects = cfg.retention.objects;
};
2023-11-26 14:43:45 -08:00
};
2023-11-29 15:59:19 -08:00
birdseye = {
enabled = true;
mode = "continuous";
};
2023-11-26 14:43:45 -08:00
};
2023-11-26 17:38:26 -08:00
in pkgs.writeText "frigate.yaml" content;
2023-11-26 14:43:45 -08:00
in {
options.services.frigateContainer = with types; {
enable = mkEnableOption "Enable Frigate CCTV in a container.";
state-directory = mkOption {
type = str;
description = "Path at which to store Frigate recordings & data.";
};
log-level = mkOption {
type = str;
description = "Level at which to output Frigate logs.";
default = "error";
};
images = {
frigate = mkOption {
type = str;
description = "Frigate Docker image to run.";
};
};
2023-11-26 21:19:49 -08:00
hwaccel = mkOption {
type = nullOr str;
description = "Hardware acceleration driver.";
default = null;
};
2023-11-26 14:43:45 -08:00
retention = {
default = mkOption {
type = int;
description = "Retention time for all motion, in days.";
default = 7;
};
events = mkOption {
type = int;
description = "Retention time for all detected objects, in days.";
default = 14;
};
objects = mkOption {
type = attrsOf int;
description = "Map of object type to retention time in days.";
default = {
person = 60;
dog = 30;
cat = 30;
};
};
};
ports = {
frigate = mkOption {
type = port;
description = "Port on which to listen for Frigate web traffic.";
default = 5000;
};
rtsp = mkOption {
type = port;
description = "Port on which to listen for Frigate RTSP traffic.";
default = 8554;
};
webrtc = mkOption {
type = port;
description = "Port on which to listen for Frigate WebRTC traffic.";
default = 8555;
};
};
devices = mkOption {
type = listOf str;
description = "List of devices to pass to the container (for hw accel).";
default = [ ];
};
camera-password-file = mkOption {
type = str;
description =
"Path on build host to file containing the camera password.";
};
2023-11-26 16:36:31 -08:00
detectors = mkOption {
type = attrsOf (attrsOf str);
2023-11-26 16:36:31 -08:00
default = { };
};
2023-11-26 14:43:45 -08:00
cameras = let
cameraOpts = { name, ... }: {
options = {
name = mkOption {
type = str;
description = "Camera name.";
default = name;
};
2023-11-29 16:09:19 -08:00
default = mkEnableOption "Make camera the primary birds-eye view.";
2023-11-26 14:43:45 -08:00
streams = {
low = mkOption {
type = str;
description = "URL of the low-quality stream.";
};
high = mkOption {
type = str;
description = "URL of the high-quality stream.";
};
};
};
};
in mkOption {
type = attrsOf (submodule cameraOpts);
description = "Cameras for Frigate CCTV to use.";
default = { };
};
mqtt = {
host = mkOption {
type = str;
description = "Hostname of the MQTT server.";
};
port = mkOption {
type = port;
description = "Port on which to contact the MQTT server.";
};
2023-11-26 15:33:28 -08:00
user = mkOption {
2023-11-26 14:43:45 -08:00
type = str;
description = "User as which to connect to server.";
};
password-file = mkOption {
type = str;
description =
2023-11-26 14:43:45 -08:00
"File containing password with which to authenticate to MQTT server.";
};
};
};
config = mkIf cfg.enable {
fudo.secrets.host-secrets."${config.instance.hostname}" = {
frigateEnv = {
source-file = let
camPasswd = removeNewline (readFile cfg.camera-password-file);
mqttPasswd = removeNewline (readFile cfg.mqtt.password-file);
2023-11-26 16:37:20 -08:00
in makeEnvFile {
2023-11-26 14:43:45 -08:00
FRIGATE_RTSP_PASSWORD = camPasswd;
FRIGATE_MQTT_PASSWORD = mqttPasswd;
};
target-file = "/run/frigate/camera.passwd";
};
};
2023-11-26 16:05:58 -08:00
virtualisation.arion.projects.frigate.settings = let
2023-11-26 14:43:45 -08:00
image = { pkgs, ... }: {
project.name = "frigate-cctv";
services = {
frigate.service = {
image = cfg.images.frigate;
hostname = "frigate";
restart = "always";
volumes = [
"${frigateCfg}:/config/config.yml"
"${cfg.state-directory}:/media/frigate"
];
# shm_size = cfg.shm-size;
2023-11-26 14:43:45 -08:00
devices = cfg.devices;
ports = [
2023-11-26 16:41:12 -08:00
"${toString cfg.ports.frigate}:5000"
"${toString cfg.ports.rtsp}:8554"
"${toString cfg.ports.webrtc}:8555/tcp"
"${toString cfg.ports.webrtc}:8555/udp"
2023-11-26 14:43:45 -08:00
];
2023-11-26 16:37:42 -08:00
env_file = [ hostSecrets.frigateEnv.target-file ];
2023-11-26 14:43:45 -08:00
};
2024-05-24 09:44:07 -07:00
# TODO: add metrics exporter
2023-11-26 14:43:45 -08:00
};
};
2023-11-26 16:05:58 -08:00
in { imports = [ image ]; };
2023-11-26 14:43:45 -08:00
};
}