feat(rsync-backup): add an rsync module with support for incremental and simple backups
Currently running tests on brontes to validate the module
This commit is contained in:
parent
d6da8c55c9
commit
fd3bcb601e
3 changed files with 177 additions and 0 deletions
|
@ -40,6 +40,15 @@
|
||||||
enable = true;
|
enable = true;
|
||||||
tags = [ "client" ];
|
tags = [ "client" ];
|
||||||
};
|
};
|
||||||
|
rsync-backup = {
|
||||||
|
enable = true;
|
||||||
|
sources = [
|
||||||
|
"/home/ny/Downloads"
|
||||||
|
"/home/ny/notes"
|
||||||
|
];
|
||||||
|
target = "/home/ny/test";
|
||||||
|
incremental.enable = true;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
media.enableAll = true;
|
media.enableAll = true;
|
||||||
|
|
|
@ -2,5 +2,6 @@
|
||||||
imports = [
|
imports = [
|
||||||
./nysh.nix
|
./nysh.nix
|
||||||
./tailscale.nix
|
./tailscale.nix
|
||||||
|
./rsync
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
167
options/services/rsync/default.nix
Normal file
167
options/services/rsync/default.nix
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
{
|
||||||
|
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;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue