2023-06-23 06:22:34 +02:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.shb.home-assistant;
|
2023-07-16 00:09:54 +02:00
|
|
|
|
2024-01-12 08:22:46 +01:00
|
|
|
contracts = pkgs.callPackage ../contracts {};
|
2024-03-01 00:34:53 +01:00
|
|
|
shblib = pkgs.callPackage ../../lib {};
|
2024-01-12 08:22:46 +01:00
|
|
|
|
2023-07-16 00:09:54 +02:00
|
|
|
fqdn = "${cfg.subdomain}.${cfg.domain}";
|
2023-07-20 08:19:08 +02:00
|
|
|
|
|
|
|
ldap_auth_script_repo = pkgs.fetchFromGitHub {
|
|
|
|
owner = "lldap";
|
|
|
|
repo = "lldap";
|
|
|
|
rev = "7d1f5abc137821c500de99c94f7579761fc949d8";
|
|
|
|
sha256 = "sha256-8D+7ww70Ja6Qwdfa+7MpjAAHewtCWNf/tuTAExoUrg0=";
|
|
|
|
};
|
|
|
|
|
|
|
|
ldap_auth_script = pkgs.writeShellScriptBin "ldap_auth.sh" ''
|
|
|
|
export PATH=${pkgs.gnused}/bin:${pkgs.curl}/bin:${pkgs.jq}/bin
|
|
|
|
exec ${pkgs.bash}/bin/bash ${ldap_auth_script_repo}/example_configs/lldap-ha-auth.sh $@
|
|
|
|
'';
|
2024-03-01 00:34:53 +01:00
|
|
|
|
|
|
|
# Filter secrets from config. Secrets are those of the form { source = <path>; }
|
|
|
|
secrets = lib.attrsets.filterAttrs (k: v: builtins.isAttrs v) cfg.config;
|
|
|
|
|
|
|
|
nonSecrets = (lib.attrsets.filterAttrs (k: v: !(builtins.isAttrs v)) cfg.config);
|
|
|
|
|
|
|
|
configWithSecretsIncludes =
|
|
|
|
nonSecrets
|
|
|
|
// (lib.attrsets.mapAttrs (k: v: "!secret ${k}") secrets);
|
2023-06-23 06:22:34 +02:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options.shb.home-assistant = {
|
|
|
|
enable = lib.mkEnableOption "selfhostblocks.home-assistant";
|
|
|
|
|
|
|
|
subdomain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Subdomain under which home-assistant will be served.";
|
|
|
|
example = "ha";
|
|
|
|
};
|
|
|
|
|
2023-07-03 09:26:57 +02:00
|
|
|
domain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "domain under which home-assistant will be served.";
|
|
|
|
example = "mydomain.com";
|
|
|
|
};
|
|
|
|
|
2024-01-12 08:22:46 +01:00
|
|
|
ssl = lib.mkOption {
|
|
|
|
description = "Path to SSL files";
|
|
|
|
type = lib.types.nullOr contracts.ssl.certs;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
2024-03-01 00:34:53 +01:00
|
|
|
config = lib.mkOption {
|
|
|
|
description = "See all available settings at https://www.home-assistant.io/docs/configuration/basic/";
|
|
|
|
type = lib.types.submodule {
|
|
|
|
freeformType = lib.types.attrsOf lib.types.str;
|
|
|
|
options = {
|
|
|
|
name = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str shblib.secretFileType ];
|
|
|
|
description = "Name of the Home Assistant instance.";
|
|
|
|
};
|
|
|
|
country = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str shblib.secretFileType ];
|
|
|
|
description = "Two letter country code where this instance is located.";
|
|
|
|
};
|
|
|
|
latitude = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str shblib.secretFileType ];
|
|
|
|
description = "Latitude where this instance is located.";
|
|
|
|
};
|
|
|
|
longitude = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str shblib.secretFileType ];
|
|
|
|
description = "Longitude where this instance is located.";
|
|
|
|
};
|
|
|
|
time_zone = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str shblib.secretFileType ];
|
|
|
|
description = "Timezone of this instance.";
|
|
|
|
example = "America/Los_Angeles";
|
|
|
|
};
|
|
|
|
unit_system = lib.mkOption {
|
|
|
|
type = lib.types.oneOf [ lib.types.str (lib.types.enum [ "metric" "us_customary" ]) ];
|
|
|
|
description = "Timezone of this instance.";
|
|
|
|
example = "America/Los_Angeles";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-01-07 23:42:07 +01:00
|
|
|
ldap = lib.mkOption {
|
|
|
|
description = ''
|
|
|
|
LDAP Integration App. [Manual](https://docs.nextcloud.com/server/latest/admin_manual/configuration_user/user_auth_ldap.html)
|
|
|
|
|
|
|
|
Enabling this app will create a new LDAP configuration or update one that exists with
|
|
|
|
the given host.
|
|
|
|
'';
|
|
|
|
default = {};
|
|
|
|
type = lib.types.submodule {
|
|
|
|
options = {
|
|
|
|
enable = lib.mkEnableOption "LDAP app.";
|
|
|
|
|
|
|
|
host = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = ''
|
|
|
|
Host serving the LDAP server.
|
|
|
|
|
|
|
|
|
|
|
|
If set, the Home Assistant auth will be disabled. To keep it, set
|
|
|
|
`keepDefaultAuth` to `true`.
|
|
|
|
'';
|
|
|
|
default = "127.0.0.1";
|
|
|
|
};
|
|
|
|
|
|
|
|
port = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
|
|
|
description = ''
|
|
|
|
Port of the service serving the LDAP server.
|
|
|
|
'';
|
|
|
|
default = 389;
|
|
|
|
};
|
|
|
|
|
|
|
|
userGroup = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
description = "Group users must belong to to be able to login to Nextcloud.";
|
|
|
|
default = "homeassistant_user";
|
|
|
|
};
|
|
|
|
|
|
|
|
keepDefaultAuth = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
|
|
|
description = ''
|
|
|
|
Keep Home Assistant auth active, even if LDAP is configured. Usually, you want to enable
|
|
|
|
this to transfer existing users to LDAP and then you can disabled it.
|
|
|
|
'';
|
|
|
|
default = false;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2023-07-20 08:19:08 +02:00
|
|
|
};
|
|
|
|
|
2023-06-23 06:22:34 +02:00
|
|
|
backupCfg = lib.mkOption {
|
|
|
|
type = lib.types.anything;
|
|
|
|
description = "Backup configuration for home-assistant";
|
|
|
|
default = {};
|
|
|
|
example = {
|
|
|
|
backend = "restic";
|
|
|
|
repositories = [];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = lib.mkIf cfg.enable {
|
|
|
|
services.home-assistant = {
|
|
|
|
enable = true;
|
|
|
|
# Find them at https://github.com/NixOS/nixpkgs/blob/master/pkgs/servers/home-assistant/component-packages.nix
|
|
|
|
extraComponents = [
|
|
|
|
# Components required to complete the onboarding
|
|
|
|
"met"
|
|
|
|
"radio_browser"
|
|
|
|
];
|
|
|
|
configDir = "/var/lib/hass";
|
|
|
|
# If you can't find a component in component-packages.nix, you can add them manually with something similar to:
|
|
|
|
# extraPackages = python3Packages: [
|
|
|
|
# (python3Packages.simplisafe-python.overrideAttrs (old: rec {
|
|
|
|
# pname = "simplisafe-python";
|
|
|
|
# version = "5b003a9fa1abd00f0e9a0b99d3ee57c4c7c16bda";
|
|
|
|
# format = "pyproject";
|
|
|
|
|
|
|
|
# src = pkgs.fetchFromGitHub {
|
|
|
|
# owner = "bachya";
|
|
|
|
# repo = pname;
|
|
|
|
# rev = "${version}";
|
|
|
|
# hash = "sha256-Ij2e0QGYLjENi/yhFBQ+8qWEJp86cgwC9E27PQ5xNno=";
|
|
|
|
# };
|
|
|
|
# }))
|
|
|
|
# ];
|
|
|
|
config = {
|
|
|
|
# Includes dependencies for a basic setup
|
|
|
|
# https://www.home-assistant.io/integrations/default_config/
|
|
|
|
default_config = {};
|
|
|
|
http = {
|
2023-07-16 00:09:54 +02:00
|
|
|
use_x_forwarded_for = true;
|
|
|
|
server_host = "127.0.0.1";
|
|
|
|
server_port = 8123;
|
2023-06-23 06:22:34 +02:00
|
|
|
trusted_proxies = "127.0.0.1";
|
|
|
|
};
|
|
|
|
logger.default = "info";
|
2024-03-01 00:34:53 +01:00
|
|
|
homeassistant = configWithSecretsIncludes // {
|
2023-07-03 09:26:57 +02:00
|
|
|
external_url = "https://${cfg.subdomain}.${cfg.domain}";
|
2024-01-07 23:42:07 +01:00
|
|
|
auth_providers =
|
|
|
|
(lib.optionals (!cfg.ldap.enable || cfg.ldap.keepDefaultAuth) [
|
|
|
|
{
|
|
|
|
type = "homeassistant";
|
|
|
|
}
|
|
|
|
])
|
|
|
|
++ (lib.optionals cfg.ldap.enable [
|
|
|
|
{
|
|
|
|
type = "command_line";
|
|
|
|
command = ldap_auth_script + "/bin/ldap_auth.sh";
|
|
|
|
args = [ "http://${cfg.ldap.host}:${toString cfg.ldap.port}" cfg.ldap.userGroup ];
|
|
|
|
meta = true;
|
|
|
|
}
|
|
|
|
]);
|
2023-06-23 06:22:34 +02:00
|
|
|
};
|
|
|
|
"automation ui" = "!include automations.yaml";
|
|
|
|
"scene ui" = "!include scenes.yaml";
|
|
|
|
"script ui" = "!include scripts.yaml";
|
|
|
|
|
|
|
|
"automation manual" = [
|
|
|
|
{
|
|
|
|
alias = "Create Backup on Schedule";
|
|
|
|
trigger = [
|
|
|
|
{
|
|
|
|
platform = "time_pattern";
|
|
|
|
minutes = "5";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
action = [
|
|
|
|
{
|
|
|
|
service = "shell_command.delete_backups";
|
|
|
|
data = {};
|
|
|
|
}
|
|
|
|
{
|
|
|
|
service = "backup.create";
|
|
|
|
data = {};
|
|
|
|
}
|
|
|
|
];
|
|
|
|
mode = "single";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
shell_command = {
|
|
|
|
delete_backups = "find ${config.services.home-assistant.configDir}/backups -type f -delete";
|
|
|
|
};
|
2024-01-22 08:38:57 +01:00
|
|
|
|
|
|
|
conversation.intents = {
|
|
|
|
TellJoke = [
|
|
|
|
"Tell [me] (a joke|something funny|a dad joke)"
|
|
|
|
"Raconte [moi] (une blague)"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
sensor = [
|
|
|
|
{
|
|
|
|
name = "random_joke";
|
|
|
|
platform = "rest";
|
|
|
|
json_attributes = ["joke" "id" "status"];
|
|
|
|
value_template = "{{ value_json.joke }}";
|
|
|
|
resource = "https://icanhazdadjoke.com/";
|
|
|
|
scan_interval = "3600";
|
|
|
|
headers.Accept = "application/json";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
intent_script.TellJoke = {
|
|
|
|
speech.text = ''{{ state_attr("sensor.random_joke", "joke") }}'';
|
|
|
|
action = {
|
|
|
|
service = "homeassistant.update_entity";
|
|
|
|
entity_id = "sensor.random_joke";
|
|
|
|
};
|
|
|
|
};
|
2023-06-23 06:22:34 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2023-07-16 00:09:54 +02:00
|
|
|
services.nginx.virtualHosts."${fqdn}" = {
|
|
|
|
http2 = true;
|
2024-01-12 08:22:46 +01:00
|
|
|
|
|
|
|
forceSSL = !(isNull cfg.ssl);
|
|
|
|
sslCertificate = lib.mkIf (!(isNull cfg.ssl)) cfg.ssl.paths.cert;
|
|
|
|
sslCertificateKey = lib.mkIf (!(isNull cfg.ssl)) cfg.ssl.paths.key;
|
|
|
|
|
2023-07-16 00:09:54 +02:00
|
|
|
extraConfig = ''
|
|
|
|
proxy_buffering off;
|
|
|
|
'';
|
|
|
|
locations."/" = {
|
|
|
|
proxyPass = "http://${toString config.services.home-assistant.config.http.server_host}:${toString config.services.home-assistant.config.http.server_port}/";
|
|
|
|
proxyWebsockets = true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-01-07 23:42:07 +01:00
|
|
|
systemd.services.home-assistant.preStart = lib.mkIf cfg.ldap.enable (
|
2023-11-18 07:52:51 +01:00
|
|
|
let
|
|
|
|
onboarding = pkgs.writeText "onboarding" ''
|
2024-01-07 23:42:07 +01:00
|
|
|
{
|
|
|
|
"version": 4,
|
|
|
|
"minor_version": 1,
|
|
|
|
"key": "onboarding",
|
|
|
|
"data": {
|
|
|
|
"done": [
|
|
|
|
"user",
|
|
|
|
"core_config"
|
|
|
|
]
|
|
|
|
}
|
2023-11-18 07:52:51 +01:00
|
|
|
}
|
|
|
|
'';
|
2024-03-01 00:34:53 +01:00
|
|
|
storage = "${config.services.home-assistant.configDir}";
|
|
|
|
file = "${storage}/.storage/onboarding";
|
2023-11-18 07:52:51 +01:00
|
|
|
in
|
2024-01-07 23:42:07 +01:00
|
|
|
''
|
|
|
|
if ! -f ${file}; then
|
|
|
|
mkdir -p ${storage} && cp ${onboarding} ${file}
|
|
|
|
fi
|
2024-03-01 00:34:53 +01:00
|
|
|
'' + shblib.replaceSecrets {
|
|
|
|
userConfig = cfg.config;
|
|
|
|
resultPath = "${config.services.home-assistant.configDir}/secrets.yaml";
|
2024-04-11 08:52:24 +02:00
|
|
|
generator = name: value: lib.generators.toYAML {} value;
|
2024-03-01 00:34:53 +01:00
|
|
|
});
|
2023-06-23 06:22:34 +02:00
|
|
|
|
|
|
|
systemd.tmpfiles.rules = [
|
|
|
|
"f ${config.services.home-assistant.configDir}/automations.yaml 0755 hass hass"
|
|
|
|
"f ${config.services.home-assistant.configDir}/scenes.yaml 0755 hass hass"
|
|
|
|
"f ${config.services.home-assistant.configDir}/scripts.yaml 0755 hass hass"
|
|
|
|
];
|
|
|
|
|
|
|
|
shb.backup.instances.home-assistant = lib.mkIf (cfg.backupCfg != {}) (
|
|
|
|
cfg.backupCfg
|
|
|
|
// {
|
|
|
|
sourceDirectories = [
|
|
|
|
"${config.services.home-assistant.configDir}/backups"
|
|
|
|
];
|
|
|
|
|
|
|
|
# No need for backup hooks as we use an hourly automation job in home assistant directly with a cron job.
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2023-07-23 04:11:22 +02:00
|
|
|
# Adds the "backup" user to the "hass" group.
|
|
|
|
users.groups.hass = {
|
|
|
|
members = [ "backup" ];
|
2023-06-23 06:22:34 +02:00
|
|
|
};
|
|
|
|
|
2023-07-23 04:11:22 +02:00
|
|
|
# This allows the "backup" user, member of the "backup" group, to access what's inside the home
|
|
|
|
# folder, which is needed for accessing the "backups" folder. It allows to read (r), enter the
|
|
|
|
# directory (x) but not modify what's inside.
|
|
|
|
users.users.hass.homeMode = "0750";
|
|
|
|
|
2023-06-23 06:22:34 +02:00
|
|
|
systemd.services.home-assistant.serviceConfig = {
|
2023-07-23 04:11:22 +02:00
|
|
|
# This allows all members of the "hass" group to read files, list directories and enter
|
|
|
|
# directories created by the home-assistant service. This is needed for the "backup" user,
|
|
|
|
# member of the "hass" group, to backup what is inside the "backup/" folder.
|
2023-06-23 06:22:34 +02:00
|
|
|
UMask = lib.mkForce "0027";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|