home-manager/modules/services/unison.nix

134 lines
3.4 KiB
Nix
Raw Normal View History

2019-07-15 01:33:13 +02:00
{ config, lib, pkgs, ... }:
with lib;
let
cfg = config.services.unison;
pairOf = t:
2020-02-02 00:39:17 +01:00
let list = types.addCheck (types.listOf t) (l: length l == 2);
in list // { description = list.description + " of length 2"; };
2019-07-15 01:33:13 +02:00
pairOptions = {
options = {
stateDirectory = mkOption {
type = types.path;
default = "${config.xdg.dataHome}/unison";
defaultText = "$XDG_DATA_HOME/unison";
description = ''
2019-07-15 01:33:13 +02:00
Unison state directory to use.
'';
};
commandOptions = mkOption rec {
type = with types; attrsOf (either str (listOf str));
2019-07-15 01:33:13 +02:00
apply = mergeAttrs default;
default = {
repeat = "watch";
sshcmd = "${pkgs.openssh}/bin/ssh";
ui = "text";
auto = "true";
batch = "true";
log = "false"; # don't log to file, handled by systemd
};
description = ''
2019-07-15 01:33:13 +02:00
Additional command line options as a dictionary to pass to the
`unison` program.
Use a list of strings to declare the same option multiple times.
2019-07-15 01:33:13 +02:00
See
{manpage}`unison(1)`
2019-07-15 01:33:13 +02:00
for a list of available options.
'';
};
roots = mkOption {
type = pairOf types.str;
example = literalExpression ''
2019-07-15 01:33:13 +02:00
[
"/home/user/documents"
"ssh://remote/documents"
]
'';
description = ''
2019-07-15 01:33:13 +02:00
Pair of roots to synchronise.
'';
};
};
};
serialiseArg = key: val:
concatStringsSep " "
(forEach (toList val) (x: escapeShellArg "-${key}=${escape [ "=" ] x}"));
2019-07-15 01:33:13 +02:00
2020-02-02 00:39:17 +01:00
serialiseArgs = args: concatStringsSep " " (mapAttrsToList serialiseArg args);
2019-07-15 01:33:13 +02:00
unitName = name: "unison-pair-${name}";
2019-07-15 01:33:13 +02:00
makeDefs = gen:
mapAttrs' (name: pairCfg: nameValuePair (unitName name) (gen name pairCfg))
2020-02-02 00:39:17 +01:00
cfg.pairs;
2019-07-15 01:33:13 +02:00
2020-02-02 00:39:17 +01:00
in {
meta.maintainers = with maintainers; [ euxane ];
2019-07-15 01:33:13 +02:00
options.services.unison = {
enable = mkEnableOption "Unison synchronisation";
2019-07-15 01:33:13 +02:00
package = mkPackageOption pkgs "unison" {
example = "pkgs.unison.override { enableX11 = false; }";
};
2019-07-15 01:33:13 +02:00
pairs = mkOption {
type = with types; attrsOf (submodule pairOptions);
default = { };
example = literalExpression ''
2019-07-15 01:33:13 +02:00
{
"my-documents" = {
roots = [
"/home/user/documents"
"ssh://remote/documents"
];
};
2019-07-15 01:33:13 +02:00
}
'';
description = ''
2019-07-15 01:33:13 +02:00
Unison root pairs to keep synchronised.
'';
};
};
config = mkIf cfg.enable {
assertions = [
(lib.hm.assertions.assertPlatform "services.unison" pkgs
lib.platforms.linux)
];
2019-07-15 01:33:13 +02:00
systemd.user.services = makeDefs (name: pairCfg: {
Unit.Description = "Unison pair sync (${name})";
2019-07-15 01:33:13 +02:00
Service = {
CPUSchedulingPolicy = "idle";
IOSchedulingClass = "idle";
Environment = [ "UNISON='${toString pairCfg.stateDirectory}'" ];
ExecStart = ''
${cfg.package}/bin/unison \
2019-07-15 01:33:13 +02:00
${serialiseArgs pairCfg.commandOptions} \
${strings.concatMapStringsSep " " escapeShellArg pairCfg.roots}
'';
};
});
2019-07-15 01:33:13 +02:00
systemd.user.timers = makeDefs (name: pairCfg: {
Unit.Description = "Unison pair sync auto-restart (${name})";
Install.WantedBy = [ "timers.target" ];
Timer = {
Unit = "${unitName name}.service";
OnActiveSec = 1;
OnUnitInactiveSec = 60;
};
2019-07-15 01:33:13 +02:00
});
};
}