2014-08-24 01:33:31 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.services.tinc;
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
|
|
services.tinc = {
|
|
|
|
|
|
|
|
networks = mkOption {
|
|
|
|
default = { };
|
2016-11-16 07:32:02 +00:00
|
|
|
type = with types; attrsOf (submodule {
|
2016-09-11 10:37:46 +01:00
|
|
|
options = {
|
|
|
|
|
|
|
|
extraConfig = mkOption {
|
|
|
|
default = "";
|
|
|
|
type = types.lines;
|
|
|
|
description = ''
|
|
|
|
Extra lines to add to the tinc service configuration file.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
name = mkOption {
|
|
|
|
default = null;
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
description = ''
|
|
|
|
The name of the node which is used as an identifier when communicating
|
|
|
|
with the remote nodes in the mesh. If null then the hostname of the system
|
2017-05-27 16:31:25 +01:00
|
|
|
is used to derive a name (note that tinc may replace non-alphanumeric characters in
|
|
|
|
hostnames by underscores).
|
2016-09-11 10:37:46 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
ed25519PrivateKeyFile = mkOption {
|
|
|
|
default = null;
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
description = ''
|
|
|
|
Path of the private ed25519 keyfile.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
debugLevel = mkOption {
|
|
|
|
default = 0;
|
|
|
|
type = types.addCheck types.int (l: l >= 0 && l <= 5);
|
|
|
|
description = ''
|
|
|
|
The amount of debugging information to add to the log. 0 means little
|
|
|
|
logging while 5 is the most logging. <command>man tincd</command> for
|
|
|
|
more details.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
hosts = mkOption {
|
|
|
|
default = { };
|
2016-11-16 07:32:02 +00:00
|
|
|
type = types.attrsOf types.lines;
|
2016-09-11 10:37:46 +01:00
|
|
|
description = ''
|
|
|
|
The name of the host in the network as well as the configuration for that host.
|
|
|
|
This name should only contain alphanumerics and underscores.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
interfaceType = mkOption {
|
|
|
|
default = "tun";
|
2016-11-04 04:04:17 +00:00
|
|
|
type = types.enum [ "tun" "tap" ];
|
2016-09-11 10:37:46 +01:00
|
|
|
description = ''
|
|
|
|
The type of virtual interface used for the network connection
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
listenAddress = mkOption {
|
|
|
|
default = null;
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
description = ''
|
2017-07-14 11:47:15 +01:00
|
|
|
The ip address to listen on for incoming connections.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
bindToAddress = mkOption {
|
|
|
|
default = null;
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
description = ''
|
|
|
|
The ip address to bind to (both listen on and send packets from).
|
2016-09-11 10:37:46 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.tinc_pre;
|
|
|
|
defaultText = "pkgs.tinc_pre";
|
|
|
|
description = ''
|
|
|
|
The package to use for the tinc daemon's binary.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
chroot = mkOption {
|
|
|
|
default = true;
|
|
|
|
type = types.bool;
|
|
|
|
description = ''
|
|
|
|
Change process root directory to the directory where the config file is located (/etc/tinc/netname/), for added security.
|
|
|
|
The chroot is performed after all the initialization is done, after writing pid files and opening network sockets.
|
|
|
|
|
|
|
|
Note that tinc can't run scripts anymore (such as tinc-down or host-up), unless it is setup to be runnable inside chroot environment.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2014-08-24 01:33:31 +01:00
|
|
|
description = ''
|
|
|
|
Defines the tinc networks which will be started.
|
|
|
|
Each network invokes a different daemon.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf (cfg.networks != { }) {
|
|
|
|
|
|
|
|
environment.etc = fold (a: b: a // b) { }
|
|
|
|
(flip mapAttrsToList cfg.networks (network: data:
|
|
|
|
flip mapAttrs' data.hosts (host: text: nameValuePair
|
|
|
|
("tinc/${network}/hosts/${host}")
|
2017-07-15 09:58:07 +01:00
|
|
|
({ mode = "0644"; user = "tinc.${network}"; inherit text; })
|
2014-08-24 01:33:31 +01:00
|
|
|
) // {
|
|
|
|
"tinc/${network}/tinc.conf" = {
|
|
|
|
mode = "0444";
|
|
|
|
text = ''
|
|
|
|
Name = ${if data.name == null then "$HOST" else data.name}
|
|
|
|
DeviceType = ${data.interfaceType}
|
2015-11-09 17:21:30 +00:00
|
|
|
${optionalString (data.ed25519PrivateKeyFile != null) "Ed25519PrivateKeyFile = ${data.ed25519PrivateKeyFile}"}
|
2017-07-14 11:47:15 +01:00
|
|
|
${optionalString (data.listenAddress != null) "ListenAddress = ${data.listenAddress}"}
|
|
|
|
${optionalString (data.bindToAddress != null) "BindToAddress = ${data.bindToAddress}"}
|
2014-08-24 01:33:31 +01:00
|
|
|
Interface = tinc.${network}
|
|
|
|
${data.extraConfig}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
}
|
|
|
|
));
|
|
|
|
|
|
|
|
networking.interfaces = flip mapAttrs' cfg.networks (network: data: nameValuePair
|
|
|
|
("tinc.${network}")
|
|
|
|
({
|
|
|
|
virtual = true;
|
|
|
|
virtualType = "${data.interfaceType}";
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
systemd.services = flip mapAttrs' cfg.networks (network: data: nameValuePair
|
|
|
|
("tinc.${network}")
|
|
|
|
({
|
|
|
|
description = "Tinc Daemon - ${network}";
|
2016-09-10 19:18:51 +01:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "network.target" ];
|
2014-08-24 01:33:31 +01:00
|
|
|
path = [ data.package ];
|
|
|
|
serviceConfig = {
|
|
|
|
Type = "simple";
|
2017-07-27 14:17:13 +01:00
|
|
|
Restart = "always";
|
|
|
|
RestartSec = "3";
|
2017-09-27 16:02:23 +01:00
|
|
|
ExecStart = "${data.package}/bin/tincd -D -U tinc.${network} -n ${network} ${optionalString (data.chroot) "-R"} --pidfile /run/tinc.${network}.pid -d ${toString data.debugLevel}";
|
2014-08-24 01:33:31 +01:00
|
|
|
};
|
|
|
|
preStart = ''
|
|
|
|
mkdir -p /etc/tinc/${network}/hosts
|
2017-08-09 23:07:01 +01:00
|
|
|
chown tinc.${network} /etc/tinc/${network}/hosts
|
2014-08-24 01:33:31 +01:00
|
|
|
|
2015-02-06 07:37:20 +00:00
|
|
|
# Determine how we should generate our keys
|
|
|
|
if type tinc >/dev/null 2>&1; then
|
|
|
|
# Tinc 1.1+ uses the tinc helper application for key generation
|
2015-11-09 17:21:30 +00:00
|
|
|
${if data.ed25519PrivateKeyFile != null then " # Keyfile managed by nix" else ''
|
2015-02-06 07:37:20 +00:00
|
|
|
# Prefer ED25519 keys (only in 1.1+)
|
|
|
|
[ -f "/etc/tinc/${network}/ed25519_key.priv" ] || tinc -n ${network} generate-ed25519-keys
|
2015-11-09 17:21:30 +00:00
|
|
|
''}
|
2015-02-06 07:37:20 +00:00
|
|
|
# Otherwise use RSA keys
|
|
|
|
[ -f "/etc/tinc/${network}/rsa_key.priv" ] || tinc -n ${network} generate-rsa-keys 4096
|
|
|
|
else
|
|
|
|
# Tinc 1.0 uses the tincd application
|
|
|
|
[ -f "/etc/tinc/${network}/rsa_key.priv" ] || tincd -n ${network} -K 4096
|
|
|
|
fi
|
2014-08-24 01:33:31 +01:00
|
|
|
'';
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2017-07-21 14:43:44 +01:00
|
|
|
environment.systemPackages = let
|
|
|
|
cli-wrappers = pkgs.stdenv.mkDerivation {
|
|
|
|
name = "tinc-cli-wrappers";
|
|
|
|
buildInputs = [ pkgs.makeWrapper ];
|
|
|
|
buildCommand = ''
|
|
|
|
mkdir -p $out/bin
|
2017-09-16 21:05:25 +01:00
|
|
|
${concatStringsSep "\n" (mapAttrsToList (network: data:
|
|
|
|
optionalString (versionAtLeast data.package.version "1.1pre") ''
|
|
|
|
makeWrapper ${data.package}/bin/tinc "$out/bin/tinc.${network}" \
|
|
|
|
--add-flags "--pidfile=/run/tinc.${network}.pid"
|
2017-07-21 14:43:44 +01:00
|
|
|
'') cfg.networks)}
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
in [ cli-wrappers ];
|
|
|
|
|
2014-08-24 01:33:31 +01:00
|
|
|
users.extraUsers = flip mapAttrs' cfg.networks (network: _:
|
|
|
|
nameValuePair ("tinc.${network}") ({
|
|
|
|
description = "Tinc daemon user for ${network}";
|
2015-05-22 04:11:13 +01:00
|
|
|
isSystemUser = true;
|
2014-08-24 01:33:31 +01:00
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|