2022-04-03 19:01:21 +01:00
|
|
|
{ lib, pkgs, config, systems, ... }:
|
2022-02-19 22:55:53 +00:00
|
|
|
let
|
2022-04-03 19:01:21 +01:00
|
|
|
inherit (builtins) head attrNames;
|
2022-05-16 00:05:02 +01:00
|
|
|
inherit (lib) mkMerge mkIf mkDefault optionalAttrs mapAttrs' optionalString;
|
2022-02-19 22:55:53 +00:00
|
|
|
inherit (lib.my) mkOpt' mkBoolOpt';
|
|
|
|
|
|
|
|
cfg = config.my.deploy;
|
2022-04-03 19:01:21 +01:00
|
|
|
|
2022-06-06 13:09:31 +01:00
|
|
|
keepGensOpt = with lib.types; mkOpt' ints.unsigned 10
|
|
|
|
"Number of generations to keep when cleaning up old deployments (0 to disable deletion on deployment).";
|
|
|
|
keepGensSnippet = p: n: optionalString (n > 0) ''
|
2022-06-12 23:21:57 +01:00
|
|
|
${config.nix.package}/bin/nix-env -p "${p}" --delete-generations +${toString n}
|
2022-06-06 13:09:31 +01:00
|
|
|
'';
|
|
|
|
|
2022-05-16 00:05:02 +01:00
|
|
|
# Based on https://github.com/serokell/deploy-rs/blob/master/flake.nix
|
2023-01-08 17:32:10 +00:00
|
|
|
nixosActivate = cfg': base: (pkgs.deploy-rs.lib.activate.custom // {
|
|
|
|
dryActivate = "$PROFILE/bin/switch-to-configuration dry-activate";
|
2023-12-28 18:33:55 +00:00
|
|
|
boot = ''
|
|
|
|
$PROFILE/bin/switch-to-configuration boot
|
|
|
|
|
|
|
|
${keepGensSnippet "$PROFILE" cfg'.keepGenerations}
|
|
|
|
'';
|
2023-01-08 17:32:10 +00:00
|
|
|
}) base.config.system.build.toplevel ''
|
2022-05-16 00:05:02 +01:00
|
|
|
# work around https://github.com/NixOS/nixpkgs/issues/73404
|
|
|
|
cd /tmp
|
|
|
|
|
2023-12-28 18:33:55 +00:00
|
|
|
"$PROFILE"/bin/switch-to-configuration switch
|
2022-05-16 00:05:02 +01:00
|
|
|
|
|
|
|
# https://github.com/serokell/deploy-rs/issues/31
|
|
|
|
${with base.config.boot.loader;
|
2023-12-28 18:33:55 +00:00
|
|
|
optionalString systemd-boot.enable
|
2022-05-16 00:05:02 +01:00
|
|
|
"sed -i '/^default /d' ${efi.efiSysMountPoint}/loader/loader.conf"}
|
2022-06-06 13:09:31 +01:00
|
|
|
|
|
|
|
${keepGensSnippet "$PROFILE" cfg'.keepGenerations}
|
2022-05-16 00:05:02 +01:00
|
|
|
'';
|
|
|
|
|
2022-06-10 22:14:42 +01:00
|
|
|
systemdUtil = pkgs.writeShellApplication {
|
|
|
|
name = "systemd-util.sh";
|
|
|
|
text = ''
|
|
|
|
svcActionWatch() {
|
|
|
|
action="$1"
|
|
|
|
shift
|
|
|
|
unit="$1"
|
|
|
|
shift
|
|
|
|
|
|
|
|
journalctl -o cat --no-pager -n 0 -f -u "$unit" &
|
|
|
|
jPid=$!
|
|
|
|
cleanup() {
|
2023-01-08 17:32:10 +00:00
|
|
|
# shellcheck disable=SC2317
|
|
|
|
kill "$jPid"
|
2022-06-10 22:14:42 +01:00
|
|
|
}
|
|
|
|
trap cleanup EXIT
|
|
|
|
|
|
|
|
systemctl "$@" "$action" "$unit"
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2022-04-18 15:46:38 +01:00
|
|
|
ctrProfiles = optionalAttrs cfg.generate.containers.enable (mapAttrs' (n: c:
|
|
|
|
let
|
|
|
|
ctrConfig = systems."${n}".configuration.config;
|
|
|
|
in
|
|
|
|
{
|
2022-04-03 19:01:21 +01:00
|
|
|
name = "container-${n}";
|
|
|
|
value = {
|
2023-12-28 18:33:55 +00:00
|
|
|
path = (pkgs.deploy-rs.lib.activate.custom // {
|
|
|
|
boot = ''
|
|
|
|
echo "Next systemd-nspawn@${n}.service restart / reload will load config"
|
|
|
|
'';
|
|
|
|
}) ctrConfig.my.buildAs.container ''
|
2022-06-10 22:14:42 +01:00
|
|
|
source ${systemdUtil}/bin/systemd-util.sh
|
2022-06-06 13:09:31 +01:00
|
|
|
${if c.hotReload then ''
|
2022-06-11 17:04:38 +01:00
|
|
|
if (! systemctl show -p ActiveState systemd-nspawn@${n} | grep -q "ActiveState=active") || \
|
|
|
|
systemctl show -p StatusText systemd-nspawn@${n} | grep -q "Dummy container"; then
|
2022-05-31 21:25:51 +01:00
|
|
|
action=restart
|
|
|
|
else
|
|
|
|
action=reload
|
|
|
|
fi
|
|
|
|
|
2022-06-10 22:14:42 +01:00
|
|
|
svcActionWatch "$action" systemd-nspawn@${n}
|
2022-06-06 13:09:31 +01:00
|
|
|
'' else ''
|
2022-06-10 22:14:42 +01:00
|
|
|
svcActionWatch restart systemd-nspawn@${n}
|
2022-06-06 13:09:31 +01:00
|
|
|
''}
|
|
|
|
|
|
|
|
${keepGensSnippet "$PROFILE" cfg.generate.containers.keepGenerations}
|
|
|
|
'';
|
2022-04-03 19:01:21 +01:00
|
|
|
profilePath = "/nix/var/nix/profiles/per-container/${n}/system";
|
|
|
|
|
|
|
|
user = "root";
|
|
|
|
};
|
|
|
|
}) config.my.containers.instances);
|
2022-02-19 22:55:53 +00:00
|
|
|
in
|
|
|
|
{
|
2022-02-20 20:16:49 +00:00
|
|
|
options.my.deploy = with lib.types; {
|
2022-02-19 22:55:53 +00:00
|
|
|
authorizedKeys = {
|
|
|
|
keys = mkOpt' (listOf singleLineStr) [ ] "SSH public keys to add to the default deployment user.";
|
2023-11-02 13:41:50 +00:00
|
|
|
keyFiles = mkOpt' (listOf path) [ lib.my.c.sshKeyFiles.deploy ] "SSH public key files to add to the default deployment user.";
|
2022-02-19 22:55:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enable = mkBoolOpt' true "Whether to expose deploy-rs configuration for this system.";
|
2022-02-20 20:16:49 +00:00
|
|
|
inherit (lib.my.deploy-rs) node;
|
2022-02-19 22:55:53 +00:00
|
|
|
|
|
|
|
generate = {
|
2022-05-16 00:05:02 +01:00
|
|
|
system = {
|
|
|
|
enable = mkBoolOpt' true "Whether to generate a deploy-rs profile for this system's config.";
|
|
|
|
mode = mkOpt' str "switch" "switch-to-configuration mode.";
|
2022-06-06 13:09:31 +01:00
|
|
|
keepGenerations = keepGensOpt;
|
|
|
|
};
|
|
|
|
containers = {
|
|
|
|
enable = mkBoolOpt' true "Whether to generate deploy-rs profiles for this system's containers.";
|
|
|
|
keepGenerations = keepGensOpt;
|
2022-05-16 00:05:02 +01:00
|
|
|
};
|
2022-02-19 22:55:53 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkMerge [
|
|
|
|
{
|
2022-06-06 13:25:05 +01:00
|
|
|
my.deploy.enable = mkIf (config.my.build.isDevVM || config.boot.isContainer) false;
|
2022-02-20 20:16:49 +00:00
|
|
|
}
|
|
|
|
(mkIf cfg.enable {
|
|
|
|
my.deploy.node = {
|
|
|
|
hostname = mkDefault config.networking.fqdn;
|
2022-04-03 19:01:21 +01:00
|
|
|
profilesOrder = [ "system" ] ++ (attrNames ctrProfiles);
|
2022-02-20 20:16:49 +00:00
|
|
|
profiles = {
|
|
|
|
system = mkIf cfg.generate.system.enable {
|
2022-06-06 13:09:31 +01:00
|
|
|
path = nixosActivate cfg.generate.system { inherit config; };
|
2022-02-19 22:55:53 +00:00
|
|
|
|
2022-02-20 20:16:49 +00:00
|
|
|
user = "root";
|
2022-02-19 22:55:53 +00:00
|
|
|
};
|
2022-04-03 19:01:21 +01:00
|
|
|
} // ctrProfiles;
|
2022-02-20 20:16:49 +00:00
|
|
|
|
|
|
|
sshUser = "deploy";
|
|
|
|
user = mkDefault "root";
|
|
|
|
sudo = mkDefault (if config.security.doas.enable then "doas -u" else "sudo -u");
|
|
|
|
sshOpts = mkDefault [ "-p" (toString (head config.services.openssh.ports)) ];
|
2022-02-19 22:55:53 +00:00
|
|
|
};
|
2022-02-20 20:16:49 +00:00
|
|
|
|
2022-02-19 22:55:53 +00:00
|
|
|
users = {
|
|
|
|
users."${cfg.node.sshUser}" = {
|
|
|
|
isSystemUser = true;
|
|
|
|
group = cfg.node.sshUser;
|
|
|
|
extraGroups = mkDefault [ "wheel" ];
|
|
|
|
shell = pkgs.bash;
|
|
|
|
openssh.authorizedKeys = cfg.authorizedKeys;
|
|
|
|
};
|
|
|
|
groups."${cfg.node.sshUser}" = {};
|
|
|
|
};
|
|
|
|
})
|
|
|
|
];
|
|
|
|
}
|