2019-03-22 07:57:34 +01:00
|
|
|
{ config, pkgs, ... }: with import <stockholm/lib>;
|
|
|
|
|
|
|
|
let
|
|
|
|
|
2019-06-22 02:10:29 +02:00
|
|
|
kcfg = config.krebs.syncthing;
|
|
|
|
scfg = config.services.syncthing;
|
2019-03-22 07:57:34 +01:00
|
|
|
|
|
|
|
devices = mapAttrsToList (name: peer: {
|
|
|
|
name = name;
|
|
|
|
deviceID = peer.id;
|
|
|
|
addresses = peer.addresses;
|
2019-06-22 02:10:29 +02:00
|
|
|
}) kcfg.peers;
|
2019-03-22 07:57:34 +01:00
|
|
|
|
2019-04-18 10:14:18 +02:00
|
|
|
folders = mapAttrsToList ( _: folder: {
|
2019-04-07 18:21:18 +02:00
|
|
|
inherit (folder) path id type;
|
2019-06-22 02:10:29 +02:00
|
|
|
devices = map (peer: { deviceId = kcfg.peers.${peer}.id; }) folder.peers;
|
2019-03-22 07:57:34 +01:00
|
|
|
rescanIntervalS = folder.rescanInterval;
|
2019-03-23 16:04:01 +01:00
|
|
|
fsWatcherEnabled = folder.watch;
|
|
|
|
fsWatcherDelayS = folder.watchDelay;
|
2019-06-21 23:56:07 +02:00
|
|
|
ignoreDelete = folder.ignoreDelete;
|
2019-03-23 16:04:01 +01:00
|
|
|
ignorePerms = folder.ignorePerms;
|
2019-06-22 02:10:29 +02:00
|
|
|
}) kcfg.folders;
|
2019-03-22 07:57:34 +01:00
|
|
|
|
|
|
|
getApiKey = pkgs.writeDash "getAPIKey" ''
|
|
|
|
${pkgs.libxml2}/bin/xmllint \
|
|
|
|
--xpath 'string(configuration/gui/apikey)'\
|
2019-06-25 19:21:20 +02:00
|
|
|
${scfg.configDir}/config.xml
|
2019-03-22 07:57:34 +01:00
|
|
|
'';
|
|
|
|
|
|
|
|
updateConfig = pkgs.writeDash "merge-syncthing-config" ''
|
|
|
|
set -efu
|
2019-06-22 01:36:50 +02:00
|
|
|
|
|
|
|
# XXX this assumes the GUI address to be "IPv4 address and port"
|
|
|
|
host=${shell.escape (elemAt (splitString ":" scfg.guiAddress) 0)}
|
|
|
|
port=${shell.escape (elemAt (splitString ":" scfg.guiAddress) 1)}
|
|
|
|
|
2019-03-23 16:04:22 +01:00
|
|
|
# wait for service to restart
|
2019-06-22 01:36:50 +02:00
|
|
|
${pkgs.untilport}/bin/untilport "$host" "$port"
|
|
|
|
|
2019-03-22 07:57:34 +01:00
|
|
|
API_KEY=$(${getApiKey})
|
2019-06-22 01:36:50 +02:00
|
|
|
|
|
|
|
_curl() {
|
|
|
|
${pkgs.curl}/bin/curl \
|
|
|
|
-Ss \
|
|
|
|
-H "X-API-Key: $API_KEY" \
|
|
|
|
"http://$host:$port/rest""$@"
|
|
|
|
}
|
|
|
|
|
|
|
|
old_config=$(_curl /system/config)
|
2019-06-25 19:21:20 +02:00
|
|
|
new_config=${shell.escape (toJSON {
|
2019-06-22 01:36:50 +02:00
|
|
|
inherit devices folders;
|
|
|
|
})}
|
|
|
|
new_config=$(${pkgs.jq}/bin/jq -en \
|
|
|
|
--argjson old_config "$old_config" \
|
2019-06-25 19:21:20 +02:00
|
|
|
--argjson new_config "$new_config" \
|
2019-06-22 01:36:50 +02:00
|
|
|
'
|
2019-06-25 19:21:20 +02:00
|
|
|
$old_config * $new_config
|
|
|
|
${optionalString (!kcfg.overridePeers) ''
|
|
|
|
* { devices: $old_config.devices }
|
|
|
|
''}
|
|
|
|
${optionalString (!kcfg.overrideFolders) ''
|
|
|
|
* { folders: $old_config.folders }
|
|
|
|
''}
|
2019-06-22 01:36:50 +02:00
|
|
|
'
|
|
|
|
)
|
|
|
|
echo $new_config | _curl /system/config -d @-
|
|
|
|
_curl /system/restart -X POST
|
2019-03-22 07:57:34 +01:00
|
|
|
'';
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
options.krebs.syncthing = {
|
|
|
|
|
|
|
|
enable = mkEnableOption "syncthing-init";
|
|
|
|
|
|
|
|
cert = mkOption {
|
|
|
|
type = types.nullOr types.absolute-pathname;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
key = mkOption {
|
|
|
|
type = types.nullOr types.absolute-pathname;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
|
2019-05-29 15:18:24 +02:00
|
|
|
overridePeers = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to delete the peers which are not configured via the peers option
|
|
|
|
'';
|
|
|
|
};
|
2019-03-22 07:57:34 +01:00
|
|
|
peers = mkOption {
|
|
|
|
default = {};
|
|
|
|
type = types.attrsOf (types.submodule ({
|
|
|
|
options = {
|
|
|
|
|
|
|
|
# TODO make into addr + port submodule
|
|
|
|
addresses = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
default = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
#TODO check
|
|
|
|
id = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
2019-05-29 15:18:24 +02:00
|
|
|
overrideFolders = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Whether to delete the folders which are not configured via the peers option
|
|
|
|
'';
|
|
|
|
};
|
2019-03-22 07:57:34 +01:00
|
|
|
folders = mkOption {
|
2019-04-18 10:14:18 +02:00
|
|
|
default = {};
|
|
|
|
type = types.attrsOf (types.submodule ({ config, ... }: {
|
2019-03-22 07:57:34 +01:00
|
|
|
options = {
|
|
|
|
|
|
|
|
path = mkOption {
|
|
|
|
type = types.absolute-pathname;
|
2019-04-18 10:14:18 +02:00
|
|
|
default = config._module.args.name;
|
2019-03-22 07:57:34 +01:00
|
|
|
};
|
|
|
|
|
2019-04-07 18:21:18 +02:00
|
|
|
id = mkOption {
|
|
|
|
type = types.str;
|
2019-04-18 10:14:18 +02:00
|
|
|
default = config._module.args.name;
|
2019-04-07 18:21:18 +02:00
|
|
|
};
|
|
|
|
|
2019-03-22 07:57:34 +01:00
|
|
|
peers = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
default = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
rescanInterval = mkOption {
|
|
|
|
type = types.int;
|
2019-03-23 16:04:50 +01:00
|
|
|
default = 3600;
|
2019-03-22 07:57:34 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
type = mkOption {
|
|
|
|
type = types.enum [ "sendreceive" "sendonly" "receiveonly" ];
|
|
|
|
default = "sendreceive";
|
|
|
|
};
|
|
|
|
|
2019-03-23 16:04:01 +01:00
|
|
|
watch = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
watchDelay = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 10;
|
|
|
|
};
|
|
|
|
|
2019-06-21 23:56:07 +02:00
|
|
|
ignoreDelete = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
};
|
|
|
|
|
2019-03-23 16:04:01 +01:00
|
|
|
ignorePerms = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
};
|
|
|
|
|
2019-03-22 07:57:34 +01:00
|
|
|
};
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2019-06-22 02:10:29 +02:00
|
|
|
config = mkIf kcfg.enable {
|
2019-03-22 07:57:34 +01:00
|
|
|
|
2019-06-22 02:10:29 +02:00
|
|
|
systemd.services.syncthing = mkIf (kcfg.cert != null || kcfg.key != null) {
|
2019-07-15 18:26:30 +02:00
|
|
|
serviceConfig.PermissionsStartOnly = mkDefault true;
|
2019-03-22 07:57:34 +01:00
|
|
|
preStart = ''
|
2019-06-22 02:10:29 +02:00
|
|
|
${optionalString (kcfg.cert != null) ''
|
2019-06-25 19:21:20 +02:00
|
|
|
cp ${toString kcfg.cert} ${scfg.configDir}/cert.pem
|
|
|
|
chown ${scfg.user}:${scfg.group} ${scfg.configDir}/cert.pem
|
|
|
|
chmod 400 ${scfg.configDir}/cert.pem
|
2019-04-09 16:52:17 +02:00
|
|
|
''}
|
2019-06-22 02:10:29 +02:00
|
|
|
${optionalString (kcfg.key != null) ''
|
2019-06-25 19:21:20 +02:00
|
|
|
cp ${toString kcfg.key} ${scfg.configDir}/key.pem
|
|
|
|
chown ${scfg.user}:${scfg.group} ${scfg.configDir}/key.pem
|
|
|
|
chmod 400 ${scfg.configDir}/key.pem
|
2019-04-09 16:52:17 +02:00
|
|
|
''}
|
2019-03-22 07:57:34 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
systemd.services.syncthing-init = {
|
|
|
|
after = [ "syncthing.service" ];
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
|
|
|
|
serviceConfig = {
|
2019-06-22 02:10:29 +02:00
|
|
|
User = scfg.user;
|
2019-03-22 07:57:34 +01:00
|
|
|
RemainAfterExit = true;
|
|
|
|
Type = "oneshot";
|
|
|
|
ExecStart = updateConfig;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|