2019-10-27 03:37:30 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.syncoid;
|
2020-02-10 00:40:52 +00:00
|
|
|
|
|
|
|
# Extract pool names of local datasets (ones that don't contain "@") that
|
|
|
|
# have the specified type (either "source" or "target")
|
|
|
|
getPools = type: unique (map (d: head (builtins.match "([^/]+).*" d)) (
|
|
|
|
# Filter local datasets
|
|
|
|
filter (d: !hasInfix "@" d)
|
|
|
|
# Get datasets of the specified type
|
|
|
|
(catAttrs type (attrValues cfg.commands))
|
|
|
|
));
|
2019-10-27 03:37:30 +00:00
|
|
|
in {
|
|
|
|
|
|
|
|
# Interface
|
|
|
|
|
|
|
|
options.services.syncoid = {
|
|
|
|
enable = mkEnableOption "Syncoid ZFS synchronization service";
|
|
|
|
|
|
|
|
interval = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "hourly";
|
|
|
|
example = "*-*-* *:15:00";
|
|
|
|
description = ''
|
|
|
|
Run syncoid at this interval. The default is to run hourly.
|
|
|
|
|
|
|
|
The format is described in
|
|
|
|
<citerefentry><refentrytitle>systemd.time</refentrytitle>
|
|
|
|
<manvolnum>7</manvolnum></citerefentry>.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
user = mkOption {
|
|
|
|
type = types.str;
|
2020-02-10 00:40:52 +00:00
|
|
|
default = "syncoid";
|
2019-10-27 03:37:30 +00:00
|
|
|
example = "backup";
|
|
|
|
description = ''
|
2020-02-10 00:40:52 +00:00
|
|
|
The user for the service. ZFS privilege delegation will be
|
|
|
|
automatically configured for any local pools used by syncoid if this
|
|
|
|
option is set to a user other than root. The user will be given the
|
|
|
|
"hold" and "send" privileges on any pool that has datasets being sent
|
|
|
|
and the "create", "mount", "receive", and "rollback" privileges on
|
|
|
|
any pool that has datasets being received.
|
2019-10-27 03:37:30 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2020-02-10 00:40:52 +00:00
|
|
|
group = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "syncoid";
|
|
|
|
example = "backup";
|
|
|
|
description = "The group for the service.";
|
|
|
|
};
|
|
|
|
|
2019-10-27 03:37:30 +00:00
|
|
|
sshKey = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
# Prevent key from being copied to store
|
|
|
|
apply = mapNullable toString;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
SSH private key file to use to login to the remote system. Can be
|
|
|
|
overridden in individual commands.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
commonArgs = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
default = [];
|
|
|
|
example = [ "--no-sync-snap" ];
|
|
|
|
description = ''
|
|
|
|
Arguments to add to every syncoid command, unless disabled for that
|
|
|
|
command. See
|
|
|
|
<link xlink:href="https://github.com/jimsalterjrs/sanoid/#syncoid-command-line-options"/>
|
|
|
|
for available options.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
commands = mkOption {
|
|
|
|
type = types.attrsOf (types.submodule ({ name, ... }: {
|
|
|
|
options = {
|
|
|
|
source = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
example = "pool/dataset";
|
|
|
|
description = ''
|
|
|
|
Source ZFS dataset. Can be either local or remote. Defaults to
|
|
|
|
the attribute name.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
target = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
example = "user@server:pool/dataset";
|
|
|
|
description = ''
|
|
|
|
Target ZFS dataset. Can be either local
|
|
|
|
(<replaceable>pool/dataset</replaceable>) or remote
|
|
|
|
(<replaceable>user@server:pool/dataset</replaceable>).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
recursive = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Whether to also transfer child datasets.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
sshKey = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
# Prevent key from being copied to store
|
|
|
|
apply = mapNullable toString;
|
|
|
|
description = ''
|
|
|
|
SSH private key file to use to login to the remote system.
|
|
|
|
Defaults to <option>services.syncoid.sshKey</option> option.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
sendOptions = mkOption {
|
|
|
|
type = types.separatedString " ";
|
|
|
|
default = "";
|
|
|
|
example = "Lc e";
|
|
|
|
description = ''
|
|
|
|
Advanced options to pass to zfs send. Options are specified
|
|
|
|
without their leading dashes and separated by spaces.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
recvOptions = mkOption {
|
|
|
|
type = types.separatedString " ";
|
|
|
|
default = "";
|
|
|
|
example = "ux recordsize o compression=lz4";
|
|
|
|
description = ''
|
|
|
|
Advanced options to pass to zfs recv. Options are specified
|
|
|
|
without their leading dashes and separated by spaces.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
useCommonArgs = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to add the configured common arguments to this command.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
extraArgs = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
default = [];
|
|
|
|
example = [ "--sshport 2222" ];
|
|
|
|
description = "Extra syncoid arguments for this command.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
config = {
|
|
|
|
source = mkDefault name;
|
|
|
|
sshKey = mkDefault cfg.sshKey;
|
|
|
|
};
|
|
|
|
}));
|
|
|
|
default = {};
|
2020-04-02 06:39:04 +01:00
|
|
|
example = literalExample ''
|
|
|
|
{
|
|
|
|
"pool/test".target = "root@target:pool/test";
|
|
|
|
}
|
|
|
|
'';
|
2019-10-27 03:37:30 +00:00
|
|
|
description = "Syncoid commands to run.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
# Implementation
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2020-02-10 00:40:52 +00:00
|
|
|
users = {
|
|
|
|
users = mkIf (cfg.user == "syncoid") {
|
|
|
|
syncoid = {
|
|
|
|
group = cfg.group;
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
groups = mkIf (cfg.group == "syncoid") {
|
|
|
|
syncoid = {};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2019-10-27 03:37:30 +00:00
|
|
|
systemd.services.syncoid = {
|
|
|
|
description = "Syncoid ZFS synchronization service";
|
|
|
|
script = concatMapStringsSep "\n" (c: lib.escapeShellArgs
|
|
|
|
([ "${pkgs.sanoid}/bin/syncoid" ]
|
|
|
|
++ (optionals c.useCommonArgs cfg.commonArgs)
|
|
|
|
++ (optional c.recursive "-r")
|
|
|
|
++ (optionals (c.sshKey != null) [ "--sshkey" c.sshKey ])
|
|
|
|
++ c.extraArgs
|
|
|
|
++ [ "--sendoptions" c.sendOptions
|
|
|
|
"--recvoptions" c.recvOptions
|
2020-02-10 00:40:52 +00:00
|
|
|
"--no-privilege-elevation"
|
2019-10-27 03:37:30 +00:00
|
|
|
c.source c.target
|
|
|
|
])) (attrValues cfg.commands);
|
|
|
|
after = [ "zfs.target" ];
|
2020-02-10 00:40:52 +00:00
|
|
|
serviceConfig = {
|
2020-11-20 21:33:16 +00:00
|
|
|
ExecStartPre = let
|
|
|
|
allowCmd = permissions: pool: lib.escapeShellArgs [
|
|
|
|
"+/run/booted-system/sw/bin/zfs" "allow"
|
|
|
|
cfg.user (concatStringsSep "," permissions) pool
|
|
|
|
];
|
|
|
|
in
|
|
|
|
(map (allowCmd [ "hold" "send" "snapshot" "destroy" ]) (getPools "source")) ++
|
|
|
|
(map (allowCmd [ "create" "mount" "receive" "rollback" ]) (getPools "target"));
|
2020-02-10 00:40:52 +00:00
|
|
|
User = cfg.user;
|
|
|
|
Group = cfg.group;
|
|
|
|
};
|
2019-10-27 03:37:30 +00:00
|
|
|
startAt = cfg.interval;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
meta.maintainers = with maintainers; [ lopsided98 ];
|
|
|
|
}
|