2023-07-17 00:01:45 -07:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.shb.ldap;
|
|
|
|
|
2024-01-11 23:22:46 -08:00
|
|
|
contracts = pkgs.callPackage ../contracts {};
|
|
|
|
|
2023-07-17 00:01:45 -07:00
|
|
|
fqdn = "${cfg.subdomain}.${cfg.domain}";
|
2024-05-17 15:11:25 -07:00
|
|
|
|
|
|
|
lldap-cli-auth = pkgs.callPackage ({ stdenvNoCC, makeWrapper, lldap-cli }: stdenvNoCC.mkDerivation {
|
|
|
|
name = "lldap-cli";
|
|
|
|
|
|
|
|
src = lldap-cli;
|
|
|
|
|
|
|
|
nativeBuildInputs = [
|
|
|
|
makeWrapper
|
|
|
|
];
|
|
|
|
|
|
|
|
# No quotes around the value for LLDAP_PASSWORD because we want the value to not be enclosed in quotes.
|
|
|
|
installPhase = ''
|
|
|
|
makeWrapper ${pkgs.lldap-cli}/bin/lldap-cli $out/bin/lldap-cli \
|
|
|
|
--set LLDAP_USERNAME "admin" \
|
|
|
|
--set LLDAP_PASSWORD $(cat ${cfg.ldapUserPasswordFile}) \
|
|
|
|
--set LLDAP_HTTPURL "http://${config.services.lldap.settings.http_host}:${toString config.services.lldap.settings.http_port}"
|
|
|
|
'';
|
|
|
|
}) {};
|
2023-07-17 00:01:45 -07:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options.shb.ldap = {
|
2023-11-30 22:08:38 -08:00
|
|
|
enable = lib.mkEnableOption "the LDAP service";
|
2023-07-17 00:01:45 -07:00
|
|
|
|
|
|
|
dcdomain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2023-11-30 22:08:38 -08:00
|
|
|
description = "dc domain to serve.";
|
2023-07-17 00:01:45 -07:00
|
|
|
example = "dc=mydomain,dc=com";
|
|
|
|
};
|
|
|
|
|
|
|
|
subdomain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2023-11-30 22:08:38 -08:00
|
|
|
description = "Subdomain under which the LDAP service will be served.";
|
2023-07-17 00:01:45 -07:00
|
|
|
example = "grafana";
|
|
|
|
};
|
|
|
|
|
|
|
|
domain = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2023-11-30 22:08:38 -08:00
|
|
|
description = "Domain under which the LDAP service will be served.";
|
2023-07-17 00:01:45 -07:00
|
|
|
example = "mydomain.com";
|
|
|
|
};
|
|
|
|
|
2023-10-17 13:41:33 -07:00
|
|
|
ldapPort = lib.mkOption {
|
2023-10-14 21:17:59 -07:00
|
|
|
type = lib.types.port;
|
|
|
|
description = "Port on which the server listens for the LDAP protocol.";
|
|
|
|
default = 3890;
|
|
|
|
};
|
|
|
|
|
2024-01-11 23:22:46 -08:00
|
|
|
ssl = lib.mkOption {
|
|
|
|
description = "Path to SSL files";
|
|
|
|
type = lib.types.nullOr contracts.ssl.certs;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
2023-11-30 22:08:38 -08:00
|
|
|
webUIListenPort = lib.mkOption {
|
2023-10-14 21:17:59 -07:00
|
|
|
type = lib.types.port;
|
|
|
|
description = "Port on which the web UI is exposed.";
|
|
|
|
default = 17170;
|
|
|
|
};
|
|
|
|
|
2023-11-30 22:08:38 -08:00
|
|
|
ldapUserPasswordFile = lib.mkOption {
|
2023-07-17 00:01:45 -07:00
|
|
|
type = lib.types.path;
|
2023-11-30 22:08:38 -08:00
|
|
|
description = "File containing the LDAP admin user password.";
|
2023-07-17 00:01:45 -07:00
|
|
|
};
|
2023-07-30 17:44:50 -07:00
|
|
|
|
2023-11-30 22:08:38 -08:00
|
|
|
jwtSecretFile = lib.mkOption {
|
|
|
|
type = lib.types.path;
|
|
|
|
description = "File containing the JWT secret.";
|
|
|
|
};
|
|
|
|
|
|
|
|
restrictAccessIPRange = lib.mkOption {
|
2023-11-17 22:52:29 -08:00
|
|
|
type = lib.types.nullOr lib.types.str;
|
2023-11-30 22:08:38 -08:00
|
|
|
description = "Set a local network range to restrict access to the UI to only those IPs.";
|
2023-07-30 17:44:50 -07:00
|
|
|
example = "192.168.1.1/24";
|
2023-11-17 22:52:29 -08:00
|
|
|
default = null;
|
2023-07-30 17:44:50 -07:00
|
|
|
};
|
2023-11-30 22:08:38 -08:00
|
|
|
|
|
|
|
debug = lib.mkOption {
|
|
|
|
description = "Enable debug logging.";
|
|
|
|
type = lib.types.bool;
|
|
|
|
default = false;
|
|
|
|
};
|
2023-07-17 00:01:45 -07:00
|
|
|
|
2024-05-17 15:11:25 -07:00
|
|
|
groups = lib.mkOption {
|
|
|
|
description = "LDAP Groups to manage declaratively.";
|
|
|
|
default = {};
|
|
|
|
example = lib.literalExpression ''
|
|
|
|
{
|
|
|
|
family = {};
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
type = lib.types.attrsOf (lib.types.submodule {
|
|
|
|
options = {};
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
users = lib.mkOption {
|
|
|
|
description = "LDAP Users to manage declaratively.";
|
|
|
|
default = {};
|
|
|
|
type = lib.types.attrsOf (lib.types.submodule {
|
|
|
|
options = {
|
|
|
|
email = lib.mkOption {
|
|
|
|
description = "Email address.";
|
|
|
|
type = lib.types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
displayName = lib.mkOption {
|
|
|
|
description = "Display name.";
|
|
|
|
type = lib.types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
firstName = lib.mkOption {
|
|
|
|
description = "First name.";
|
|
|
|
type = lib.types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
lastName = lib.mkOption {
|
|
|
|
description = "Last name.";
|
|
|
|
type = lib.types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
groups = lib.mkOption {
|
|
|
|
description = "Groups this user is member of. The group must exist.";
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
default = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
passwordFile = lib.mkOption {
|
|
|
|
type = lib.types.path;
|
|
|
|
description = "File containing the user's password.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
2023-07-17 00:01:45 -07:00
|
|
|
|
|
|
|
config = lib.mkIf cfg.enable {
|
|
|
|
services.nginx = {
|
|
|
|
enable = true;
|
|
|
|
|
|
|
|
virtualHosts.${fqdn} = {
|
2024-01-11 23:22:46 -08: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-17 00:01:45 -07:00
|
|
|
locations."/" = {
|
|
|
|
extraConfig = ''
|
|
|
|
proxy_set_header Host $host;
|
2023-11-30 22:08:38 -08:00
|
|
|
'' + (if isNull cfg.restrictAccessIPRange then "" else ''
|
|
|
|
allow ${cfg.restrictAccessIPRange};
|
2023-07-30 17:44:50 -07:00
|
|
|
deny all;
|
2023-11-17 22:52:29 -08:00
|
|
|
'');
|
2023-07-17 00:01:45 -07:00
|
|
|
proxyPass = "http://${toString config.services.lldap.settings.http_host}:${toString config.services.lldap.settings.http_port}/";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
users.users.lldap = {
|
|
|
|
name = "lldap";
|
|
|
|
group = "lldap";
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
users.groups.lldap = {
|
|
|
|
members = [ "backup" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
services.lldap = {
|
|
|
|
enable = true;
|
|
|
|
|
|
|
|
environment = {
|
2023-11-30 22:08:38 -08:00
|
|
|
LLDAP_JWT_SECRET_FILE = toString cfg.jwtSecretFile;
|
|
|
|
LLDAP_LDAP_USER_PASS_FILE = toString cfg.ldapUserPasswordFile;
|
2023-07-19 23:19:08 -07:00
|
|
|
|
2023-11-30 22:08:38 -08:00
|
|
|
RUST_LOG = lib.mkIf cfg.debug "debug";
|
2023-07-17 00:01:45 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
settings = {
|
|
|
|
http_url = "https://${fqdn}";
|
|
|
|
http_host = "127.0.0.1";
|
2023-11-30 22:08:38 -08:00
|
|
|
http_port = cfg.webUIListenPort;
|
2023-07-17 00:01:45 -07:00
|
|
|
|
|
|
|
ldap_host = "127.0.0.1";
|
2023-10-17 13:41:33 -07:00
|
|
|
ldap_port = cfg.ldapPort;
|
2023-07-17 00:01:45 -07:00
|
|
|
|
|
|
|
ldap_base_dn = cfg.dcdomain;
|
2023-07-19 23:19:08 -07:00
|
|
|
|
2023-11-30 22:08:38 -08:00
|
|
|
verbose = cfg.debug;
|
2023-07-17 00:01:45 -07:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-05-17 15:11:25 -07:00
|
|
|
environment.systemPackages = [
|
|
|
|
lldap-cli-auth
|
|
|
|
];
|
|
|
|
|
|
|
|
# $ lldap-cli schema attribute user list
|
|
|
|
#
|
|
|
|
# Name Type Is list Is visible Is editable
|
|
|
|
# ---- ---- ------- ---------- -----------
|
|
|
|
# avatar JpegPhoto false true true
|
|
|
|
# creation_date DateTime false true false
|
|
|
|
# display_name String false true true
|
|
|
|
# first_name String false true true
|
|
|
|
# last_name String false true true
|
|
|
|
# mail String false true true
|
|
|
|
# user_id String false true false
|
|
|
|
# uuid String false true false
|
|
|
|
|
|
|
|
|
|
|
|
# $ lldap-cli schema attribute group list
|
|
|
|
#
|
|
|
|
# Name Type Is list Is visible Is editable
|
|
|
|
# ---- ---- ------- ---------- -----------
|
|
|
|
# creation_date DateTime false true false
|
|
|
|
# display_name String false true true
|
|
|
|
# group_id Integer false true false
|
|
|
|
# uuid String false true false
|
|
|
|
|
|
|
|
systemd.services.lldap.postStart =
|
|
|
|
let
|
|
|
|
configFile = (pkgs.formats.toml {}).generate "lldap_config.toml" config.services.lldap.settings;
|
|
|
|
|
|
|
|
login = [''
|
|
|
|
set -euo pipefail
|
|
|
|
|
|
|
|
sleep 3
|
|
|
|
|
|
|
|
export LLDAP_USERNAME=admin
|
|
|
|
export LLDAP_PASSWORD=$(cat ${cfg.ldapUserPasswordFile})
|
|
|
|
export LLDAP_HTTPURL=http://${config.services.lldap.settings.http_host}:${toString config.services.lldap.settings.http_port}
|
|
|
|
|
|
|
|
eval $(${pkgs.lldap-cli}/bin/lldap-cli login)
|
|
|
|
|
|
|
|
set -x
|
|
|
|
''];
|
|
|
|
|
|
|
|
deleteGroups = [''
|
|
|
|
allUids=(${lib.concatStringsSep " " (
|
|
|
|
(lib.mapAttrsToList (uid: g: uid) cfg.groups)
|
|
|
|
++ [ "lldap_admin" "lldap_password_manager" "lldap_strict_readonly" ])
|
|
|
|
})
|
|
|
|
echo All managed groups are: $allUids
|
|
|
|
echo Other groups will be deleted
|
|
|
|
for uid in $(${pkgs.lldap-cli}/bin/lldap-cli group list); do
|
|
|
|
if [[ ! " ''${allUids[*]} " =~ [[:space:]]''${uid}[[:space:]] ]]; then
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli group del $uid
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
''];
|
|
|
|
|
|
|
|
createGroups = lib.mapAttrsToList (uid: g: ''
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli group add ${uid}
|
|
|
|
'') cfg.groups;
|
|
|
|
|
|
|
|
deleteUsers = [''
|
|
|
|
allUids=(${lib.concatStringsSep " " (
|
|
|
|
(lib.mapAttrsToList (uid: u: uid) cfg.users)
|
|
|
|
++ [ "admin" ])
|
|
|
|
})
|
|
|
|
for uid in $(${pkgs.lldap-cli}/bin/lldap-cli user list uid); do
|
|
|
|
if [[ ! " ''${allUids[*]} " =~ [[:space:]]''${uid}[[:space:]] ]]; then
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user del $uid
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
''];
|
|
|
|
|
|
|
|
createUsers = lib.mapAttrsToList (uid: u: ''
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user add ${uid} "${u.email}"
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user update set ${uid} password "$(cat ${u.passwordFile})"
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user update set ${uid} mail "${u.email}"
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user update set ${uid} display_name "${u.displayName}"
|
|
|
|
# ${pkgs.lldap-cli}/bin/lldap-cli user update set ${uid} first_name "${u.firstName}"
|
|
|
|
# ${pkgs.lldap-cli}/bin/lldap-cli user update set ${uid} last_name "${u.lastName}"
|
|
|
|
'') cfg.users;
|
|
|
|
|
|
|
|
addToGroups = lib.mapAttrsToList (uid: u: lib.concatMapStringsSep "\n" (g: ''
|
|
|
|
${pkgs.lldap-cli}/bin/lldap-cli user group add \
|
|
|
|
${uid} \
|
|
|
|
${g}
|
|
|
|
'') u.groups) cfg.users;
|
|
|
|
in
|
|
|
|
lib.concatStringsSep "\n\n" (
|
|
|
|
login
|
|
|
|
++ deleteGroups
|
|
|
|
++ createGroups
|
|
|
|
++ deleteUsers
|
|
|
|
++ createUsers
|
|
|
|
++ addToGroups
|
|
|
|
);
|
|
|
|
|
2023-07-17 00:01:45 -07:00
|
|
|
shb.backup.instances.lldap = {
|
|
|
|
sourceDirectories = [
|
|
|
|
"/var/lib/lldap"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|