{ config, lib, pkgs, ... }: with lib; let cfg = config.services.teslaMateContainer; hostSecrets = config.fudo.secrets.host-secrets."${config.instance.hostname}"; makeEnvFile = envVars: let envLines = mapAttrsToList (var: val: ''${var}="${toString val}"'') envVars; in pkgs.writeText "envFile" (concatStringsSep "\n" envLines); makeTeslaMateImage = { teslaMateImage, postgresImage, grafanaImage , teslaMateEnvFile, postgresEnvFile, grafanaEnvFile, teslaMateUid , postgresUid, grafanaUid, stateDirectory, ... }: { pkgs, ... }: { project.name = "teslamate"; services = { teslamate = { image = teslaMateImage; restart = "always"; volumes = [ "${stateDirectory}/import:/opt/app/import" ]; ports = [ "4000:4000" ]; user = [ "${teslaMateUid}:${teslaMateUid}" ]; env_file = [ teslaMateEnvFile ]; cap_drop = "all"; }; postgres = { image = postgresImage; restart = "always"; volumes = [ "${stateDirectory}/postgres:/var/lib/postgresql/data" ]; env_file = [ postgresEnvFile ]; user = [ "${postgresUid}:${postgresUid}" ]; }; grafana = { image = grafanaImage; restart = "always"; volumes = [ "${stateDirectory}/grafana:/var/lib/grafana" ]; env_file = [ grafanaEnvFile ]; user = [ "${grafanaUid}:${grafanaUid}" ]; ports = [ "3000:3000" ]; }; }; }; in { options.services.teslaMateContainer = with types; { enable = mkEnableOption "Enable TeslaMate in a PodMan container."; images = { tesla-mate = mkOption { type = str; description = "Docker image to use for Tesla Mate."; default = "teslamate/teslamate:latest"; }; postgres = mkOption { type = str; description = "Docker image to use for Postgres DB."; default = "postgres:15"; }; grafana = mkOption { type = str; description = "Docker image to use for Grafana."; default = "teslamate/grafana:latest"; }; }; mqtt = { host = mkOption { type = str; description = "Hostname of the MQTT server."; }; port = mkOption { type = port; description = "Port of the MQTT server"; default = 1883; }; user = mkOption { type = str; description = "User as which to connect to the MQTT server."; default = "tesla-mate"; }; password = mkOption { type = str; description = "Password with which to authenticate with the MQTT server."; }; }; }; config = mkIf cfg.enable { fudo.secrets.host-secrets."${config.instance.hostname}" = let teslaMateDbPass = readFile (pkgs.lib.passwd.stablerandom-passwd-file "teslaMateDbPasswd" config.instance.build-seed); teslaMateEncryptionKey = readFile (pkgs.lib.passwd.stablerandom-passwd-file "teslaMateEncryptionKey" config.instance.build-seed); in { teslaMateEnv = { source-file = makeEnvFile { ENCRYPTION_KEY = teslaMateEncryptionKey; DATABASE_USER = "teslamate"; DATABASE_PASS = teslaMateDbPass; DATABASE_HOST = "postgres"; MQTT_HOST = cfg.mqtt.host; MQTT_PORT = cfg.mqtt.port; MQTT_USERNAME = cfg.mqtt.user; MQTT_PASSWORD = cfg.mqtt.password; }; target-file = "/run/tesla-mate/tesla-mate.env"; }; teslaMatePostgresEnv = { source-file = makeEnvFile { POSTGRES_USER = "teslamate"; POSTGRES_PASSWORD = teslaMateDbPass; POSTGRES_DB = "teslamate"; }; target-file = "/run/tesla-mate/postgres.env"; }; teslaMateGrafanaEnv = { source-file = makeEnvFile { DATABASE_USER = "teslamate"; DATABASE_PASS = teslaMateDbPass; DATABASE_NAME = "teslamate"; DATABASE_HOST = "postgres"; }; target-file = "/run/tesla-mate/grafana.env"; }; }; users.users = { tesla-mate = { isSystemUser = true; group = "tesla-mate"; uid = 720; }; tesla-mate-postgres = { isSystemUser = true; group = "tesla-mate"; uid = 721; }; tesla-mate-grafana = { isSystemUser = true; group = "tesla-mate"; uid = 722; }; }; virtualisation.arion.projects.teslamate.settings = let teslaMateImage = makeTeslaMateImage { teslaMateImage = cfg.images.tesla-mate; postgresImage = cfg.images.postgres; grafanaImage = cfg.images.grafana; stateDirectory = cfg.state-directory; teslaMateEnvFile = hostSecrets.teslaMateEnv.target-file; postgresEnvFile = hostSecrets.teslaMatePostgresEnv.target-file; grafanaEnvFile = hostSecrets.teslaMateGrafanaEnv.target-file; teslaMateUid = config.users.users.tesla-mate.uid; teslaMatePostgresUid = config.users.users.tesla-mate-postgres.uid; teslaMateGrafanaUid = config.users.users.tesla-mate-grafana.uid; }; in { imports = [ teslaMateImage ]; }; }; }