2020-04-24 23:36:52 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.sanoid;
|
|
|
|
|
|
|
|
datasetSettingsType = with types;
|
|
|
|
(attrsOf (nullOr (oneOf [ str int bool (listOf str) ]))) // {
|
|
|
|
description = "dataset/template options";
|
|
|
|
};
|
|
|
|
|
|
|
|
commonOptions = {
|
|
|
|
hourly = mkOption {
|
|
|
|
description = "Number of hourly snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr ints.unsigned;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
daily = mkOption {
|
|
|
|
description = "Number of daily snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr ints.unsigned;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
monthly = mkOption {
|
|
|
|
description = "Number of monthly snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr ints.unsigned;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
yearly = mkOption {
|
|
|
|
description = "Number of yearly snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr ints.unsigned;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
autoprune = mkOption {
|
|
|
|
description = "Whether to automatically prune old snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr bool;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
autosnap = mkOption {
|
|
|
|
description = "Whether to automatically take snapshots.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = with types; nullOr bool;
|
|
|
|
default = null;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-07-03 03:11:41 +00:00
|
|
|
datasetOptions = rec {
|
|
|
|
use_template = mkOption {
|
2020-04-24 23:36:52 +00:00
|
|
|
description = "Names of the templates to use for this dataset.";
|
2021-07-03 03:11:41 +00:00
|
|
|
type = types.listOf (types.enum (attrNames cfg.templates));
|
2020-04-24 23:36:52 +00:00
|
|
|
default = [];
|
|
|
|
};
|
2021-07-03 03:11:41 +00:00
|
|
|
useTemplate = use_template;
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
recursive = mkOption {
|
|
|
|
description = "Whether to recursively snapshot dataset children.";
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
};
|
|
|
|
|
2021-07-03 03:11:41 +00:00
|
|
|
process_children_only = mkOption {
|
2020-04-24 23:36:52 +00:00
|
|
|
description = "Whether to only snapshot child datasets if recursing.";
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
};
|
2021-07-03 03:11:41 +00:00
|
|
|
processChildrenOnly = process_children_only;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
# Extract pool names from configured datasets
|
|
|
|
pools = unique (map (d: head (builtins.match "([^/]+).*" d)) (attrNames cfg.datasets));
|
|
|
|
|
|
|
|
configFile = let
|
|
|
|
mkValueString = v:
|
|
|
|
if builtins.isList v then concatStringsSep "," v
|
|
|
|
else generators.mkValueStringDefault {} v;
|
|
|
|
|
|
|
|
mkKeyValue = k: v: if v == null then ""
|
2021-07-03 03:11:41 +00:00
|
|
|
else if k == "processChildrenOnly" then ""
|
|
|
|
else if k == "useTemplate" then ""
|
2020-04-24 23:36:52 +00:00
|
|
|
else generators.mkKeyValueDefault { inherit mkValueString; } "=" k v;
|
|
|
|
in generators.toINI { inherit mkKeyValue; } cfg.settings;
|
|
|
|
|
|
|
|
in {
|
|
|
|
|
|
|
|
# Interface
|
|
|
|
|
|
|
|
options.services.sanoid = {
|
|
|
|
enable = mkEnableOption "Sanoid ZFS snapshotting service";
|
|
|
|
|
|
|
|
interval = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "hourly";
|
|
|
|
example = "daily";
|
|
|
|
description = ''
|
|
|
|
Run sanoid at this interval. The default is to run hourly.
|
|
|
|
|
|
|
|
The format is described in
|
|
|
|
<citerefentry><refentrytitle>systemd.time</refentrytitle>
|
|
|
|
<manvolnum>7</manvolnum></citerefentry>.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
datasets = mkOption {
|
2021-07-03 03:11:41 +00:00
|
|
|
type = types.attrsOf (types.submodule ({config, options, ...}: {
|
|
|
|
freeformType = datasetSettingsType;
|
2020-04-24 23:36:52 +00:00
|
|
|
options = commonOptions // datasetOptions;
|
2021-07-03 03:11:41 +00:00
|
|
|
config.use_template = mkAliasDefinitions (options.useTemplate or {});
|
|
|
|
config.process_children_only = mkAliasDefinitions (options.processChildrenOnly or {});
|
2020-04-24 23:36:52 +00:00
|
|
|
}));
|
|
|
|
default = {};
|
|
|
|
description = "Datasets to snapshot.";
|
|
|
|
};
|
|
|
|
|
|
|
|
templates = mkOption {
|
2021-07-03 03:11:41 +00:00
|
|
|
type = types.attrsOf (types.submodule {
|
|
|
|
freeformType = datasetSettingsType;
|
2020-04-24 23:36:52 +00:00
|
|
|
options = commonOptions;
|
2021-07-03 03:11:41 +00:00
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
default = {};
|
|
|
|
description = "Templates for datasets.";
|
|
|
|
};
|
|
|
|
|
|
|
|
settings = mkOption {
|
|
|
|
type = types.attrsOf datasetSettingsType;
|
|
|
|
description = ''
|
|
|
|
Free-form settings written directly to the config file. See
|
|
|
|
<link xlink:href="https://github.com/jimsalterjrs/sanoid/blob/master/sanoid.defaults.conf"/>
|
|
|
|
for allowed values.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
extraArgs = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
default = [];
|
|
|
|
example = [ "--verbose" "--readonly" "--debug" ];
|
|
|
|
description = ''
|
|
|
|
Extra arguments to pass to sanoid. See
|
|
|
|
<link xlink:href="https://github.com/jimsalterjrs/sanoid/#sanoid-command-line-options"/>
|
|
|
|
for allowed options.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
# Implementation
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
services.sanoid.settings = mkMerge [
|
2021-07-03 03:11:41 +00:00
|
|
|
(mapAttrs' (d: v: nameValuePair ("template_" + d) v) cfg.templates)
|
|
|
|
(mapAttrs (d: v: v) cfg.datasets)
|
2020-04-24 23:36:52 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
systemd.services.sanoid = {
|
|
|
|
description = "Sanoid snapshot service";
|
|
|
|
serviceConfig = {
|
|
|
|
ExecStartPre = map (pool: lib.escapeShellArgs [
|
|
|
|
"+/run/booted-system/sw/bin/zfs" "allow"
|
|
|
|
"sanoid" "snapshot,mount,destroy" pool
|
|
|
|
]) pools;
|
|
|
|
ExecStart = lib.escapeShellArgs ([
|
|
|
|
"${pkgs.sanoid}/bin/sanoid"
|
|
|
|
"--cron"
|
2021-07-03 03:11:41 +00:00
|
|
|
"--configdir" (pkgs.writeTextDir "sanoid.conf" configFile)
|
2020-04-24 23:36:52 +00:00
|
|
|
] ++ cfg.extraArgs);
|
|
|
|
ExecStopPost = map (pool: lib.escapeShellArgs [
|
|
|
|
"+/run/booted-system/sw/bin/zfs" "unallow" "sanoid" pool
|
|
|
|
]) pools;
|
|
|
|
User = "sanoid";
|
|
|
|
Group = "sanoid";
|
|
|
|
DynamicUser = true;
|
|
|
|
RuntimeDirectory = "sanoid";
|
|
|
|
CacheDirectory = "sanoid";
|
|
|
|
};
|
|
|
|
# Prevents missing snapshots during DST changes
|
|
|
|
environment.TZ = "UTC";
|
|
|
|
after = [ "zfs.target" ];
|
|
|
|
startAt = cfg.interval;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
meta.maintainers = with maintainers; [ lopsided98 ];
|
|
|
|
}
|