{ 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; }; }; }; }