nix-da/options/services/rsync/default.nix
2024-10-08 18:14:23 +02:00

167 lines
4.1 KiB
Nix

{
config,
lib,
pkgs,
...
}:
let
inherit (lib)
mkIf
mkEnableOption
mkOption
stringToCharacters
head
concatStringsSep
;
inherit (lib.types)
listOf
nonEmptyStr
package
enum
str
;
cfg = config.modules.services.rsync-backup;
in
{
# Used to back up multiple source destinations to a target destination regularly,
# with possibility to enable incremental backups for versioning.
# TODO: systemd unit
# - [x] script taking configurable args
# - [x] sync from/to paths
# - [ ] support
# - [ ] nfs
# - [ ] sftp
# - [x] ssh
# - [x] local
# - retention limit for incremental backups
options.modules.services.rsync-backup = {
enable = mkEnableOption "rsync backups";
sources = mkOption {
type = listOf nonEmptyStr;
default = [ ]; # TODO: validate if at least 1 element
apply = map (s: "\"${s}\"");
description = ''
A list of absolute paths to the files or folders that should get synchronized.
It isn't necessary to wrap them in escaped double quotation marks.
'';
};
exclude = mkOption {
type = str;
default = "";
};
target = mkOption {
type = nonEmptyStr;
description = ''
The target to where the data should be backed up.
If it is a directory, it will be created automatically by rsync.
Append a : to the string if it is an ssh target and ensure that passwordless access is available.
'';
};
targetType = mkOption {
type = enum [
"local"
"ssh"
];
default = if head (stringToCharacters cfg.target) == "/" then "local" else "ssh";
description = ''
Guesses if the given target is local or ssh, very rudimentary therefore it might be necessary to specify manually (or rewrite the logic).
'';
};
package = mkOption {
type = package;
default = pkgs.rsync;
description = ''
The rsync package to use.
'';
};
flags = mkOption {
type = listOf str;
default = [
"--archive"
"--verbose"
];
};
flagsFinal = mkOption {
type = listOf str;
default = cfg.flags ++ [
(mkIf cfg.incremental.enable "--delete")
(mkIf (cfg.exclude != "") "--exclude=\"${cfg.exclude}\"")
(mkIf cfg.incremental.enable "--backup-dir=\"${cfg.incremental.finalIncrementPath}\"")
];
apply = concatStringsSep " ";
};
interval = mkOption {
type = nonEmptyStr;
default = "1d";
};
incremental = {
enable = mkEnableOption "incremental backups";
format = mkOption {
type = nonEmptyStr;
default = "%Y-%m-%d-%-H-%M-%S-$RANDOM";
description = ''
The increment folder name. Refer to `man date` for specifics about the format.
Omit the leading +.
'';
};
finalIncrementPath = mkOption {
type = nonEmptyStr;
default = "${cfg.target}/increment/$(date +${cfg.incremental.format})/";
};
# TODO: add retention
};
unitName = mkOption {
type = nonEmptyStr;
default = "rsync-backup";
};
};
config = mkIf cfg.enable {
systemd.timers.${cfg.unitName} = {
enable = true;
unitConfig = {
Description = "Triggers rsync-backup regularly.";
};
timerConfig = {
Unit = "${cfg.unitName}.service";
OnBootSec = cfg.interval;
};
};
systemd.services.${cfg.unitName} = {
script =
let
destination = if cfg.incremental.enable then "${cfg.target}/backup" else cfg.target;
in
''
${cfg.package}/bin/rsync \
${cfg.flagsFinal} \
${concatStringsSep " " cfg.sources} \
"${destination}"
'';
requires = mkIf (cfg.targetType == "ssh") [ "network.target" ];
unitConfig = {
Description = "Backs up files from a source location to a specified destination.";
};
serviceConfig = {
Type = "simple";
Restart = "on-failure";
RestartSec = 300;
};
};
};
}