2017-09-27 17:30:49 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.traefik;
|
2019-12-31 00:41:18 +00:00
|
|
|
dynamicConfigFile = if cfg.dynamicConfigFile == null then
|
|
|
|
pkgs.runCommand "config.toml" {
|
|
|
|
buildInputs = [ pkgs.remarshal ];
|
|
|
|
preferLocalBuild = true;
|
|
|
|
} ''
|
|
|
|
remarshal -if json -of toml \
|
|
|
|
< ${
|
|
|
|
pkgs.writeText "dynamic_config.json"
|
|
|
|
(builtins.toJSON cfg.dynamicConfigOptions)
|
|
|
|
} \
|
|
|
|
> $out
|
|
|
|
''
|
|
|
|
else
|
|
|
|
cfg.dynamicConfigFile;
|
|
|
|
staticConfigFile = if cfg.staticConfigFile == null then
|
|
|
|
pkgs.runCommand "config.toml" {
|
|
|
|
buildInputs = [ pkgs.yj ];
|
|
|
|
preferLocalBuild = true;
|
|
|
|
} ''
|
|
|
|
yj -jt -i \
|
|
|
|
< ${
|
|
|
|
pkgs.writeText "static_config.json" (builtins.toJSON
|
|
|
|
(recursiveUpdate cfg.staticConfigOptions {
|
|
|
|
providers.file.filename = "${dynamicConfigFile}";
|
|
|
|
}))
|
|
|
|
} \
|
|
|
|
> $out
|
|
|
|
''
|
|
|
|
else
|
|
|
|
cfg.staticConfigFile;
|
2017-09-27 17:30:49 +01:00
|
|
|
in {
|
|
|
|
options.services.traefik = {
|
|
|
|
enable = mkEnableOption "Traefik web server";
|
|
|
|
|
2019-12-31 00:41:18 +00:00
|
|
|
staticConfigFile = mkOption {
|
2017-09-27 17:30:49 +01:00
|
|
|
default = null;
|
2019-12-31 00:41:18 +00:00
|
|
|
example = literalExample "/path/to/static_config.toml";
|
2017-09-27 17:30:49 +01:00
|
|
|
type = types.nullOr types.path;
|
2017-10-04 11:47:34 +01:00
|
|
|
description = ''
|
2019-12-31 00:41:18 +00:00
|
|
|
Path to traefik's static configuration to use.
|
|
|
|
(Using that option has precedence over <literal>staticConfigOptions</literal> and <literal>dynamicConfigOptions</literal>)
|
2017-10-04 11:47:34 +01:00
|
|
|
'';
|
2017-09-27 17:30:49 +01:00
|
|
|
};
|
2017-10-04 11:47:34 +01:00
|
|
|
|
2019-12-31 00:41:18 +00:00
|
|
|
staticConfigOptions = mkOption {
|
2017-09-27 17:30:49 +01:00
|
|
|
description = ''
|
2019-12-31 00:41:18 +00:00
|
|
|
Static configuration for Traefik.
|
2017-09-27 17:30:49 +01:00
|
|
|
'';
|
|
|
|
type = types.attrs;
|
2019-12-31 00:41:18 +00:00
|
|
|
default = { entryPoints.http.address = ":80"; };
|
2017-09-27 17:30:49 +01:00
|
|
|
example = {
|
2019-12-31 00:41:18 +00:00
|
|
|
entryPoints.web.address = ":8080";
|
2017-10-04 11:47:34 +01:00
|
|
|
entryPoints.http.address = ":80";
|
|
|
|
|
2019-12-31 00:41:18 +00:00
|
|
|
api = { };
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
dynamicConfigFile = mkOption {
|
|
|
|
default = null;
|
|
|
|
example = literalExample "/path/to/dynamic_config.toml";
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
description = ''
|
|
|
|
Path to traefik's dynamic configuration to use.
|
|
|
|
(Using that option has precedence over <literal>dynamicConfigOptions</literal>)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
dynamicConfigOptions = mkOption {
|
|
|
|
description = ''
|
|
|
|
Dynamic configuration for Traefik.
|
|
|
|
'';
|
|
|
|
type = types.attrs;
|
|
|
|
default = { };
|
|
|
|
example = {
|
|
|
|
http.routers.router1 = {
|
|
|
|
rule = "Host(`localhost`)";
|
|
|
|
service = "service1";
|
2017-09-27 17:30:49 +01:00
|
|
|
};
|
2019-12-31 00:41:18 +00:00
|
|
|
|
|
|
|
http.services.service1.loadBalancer.servers =
|
|
|
|
[{ url = "http://localhost:8080"; }];
|
2017-09-27 17:30:49 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
dataDir = mkOption {
|
|
|
|
default = "/var/lib/traefik";
|
|
|
|
type = types.path;
|
|
|
|
description = ''
|
2019-12-31 00:41:18 +00:00
|
|
|
Location for any persistent data traefik creates, ie. acme
|
2017-09-27 17:30:49 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2018-02-09 09:37:29 +00:00
|
|
|
group = mkOption {
|
|
|
|
default = "traefik";
|
2019-08-08 21:48:27 +01:00
|
|
|
type = types.str;
|
2018-02-09 09:37:29 +00:00
|
|
|
example = "docker";
|
|
|
|
description = ''
|
|
|
|
Set the group that traefik runs under.
|
|
|
|
For the docker backend this needs to be set to <literal>docker</literal> instead.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2017-09-27 17:30:49 +01:00
|
|
|
package = mkOption {
|
|
|
|
default = pkgs.traefik;
|
|
|
|
defaultText = "pkgs.traefik";
|
|
|
|
type = types.package;
|
|
|
|
description = "Traefik package to use.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2019-12-31 00:41:18 +00:00
|
|
|
systemd.tmpfiles.rules = [ "d '${cfg.dataDir}' 0700 traefik traefik - -" ];
|
2019-02-24 13:04:49 +00:00
|
|
|
|
2017-09-27 17:30:49 +01:00
|
|
|
systemd.services.traefik = {
|
|
|
|
description = "Traefik web server";
|
|
|
|
after = [ "network-online.target" ];
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
serviceConfig = {
|
2019-12-31 00:41:18 +00:00
|
|
|
ExecStart =
|
|
|
|
"${cfg.package}/bin/traefik --configfile=${staticConfigFile}";
|
2017-09-27 17:30:49 +01:00
|
|
|
Type = "simple";
|
|
|
|
User = "traefik";
|
2018-02-09 09:37:29 +00:00
|
|
|
Group = cfg.group;
|
2017-09-27 17:30:49 +01:00
|
|
|
Restart = "on-failure";
|
|
|
|
StartLimitInterval = 86400;
|
|
|
|
StartLimitBurst = 5;
|
|
|
|
AmbientCapabilities = "cap_net_bind_service";
|
|
|
|
CapabilityBoundingSet = "cap_net_bind_service";
|
|
|
|
NoNewPrivileges = true;
|
|
|
|
LimitNPROC = 64;
|
|
|
|
LimitNOFILE = 1048576;
|
|
|
|
PrivateTmp = true;
|
|
|
|
PrivateDevices = true;
|
|
|
|
ProtectHome = true;
|
|
|
|
ProtectSystem = "full";
|
|
|
|
ReadWriteDirectories = cfg.dataDir;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2018-06-30 00:58:35 +01:00
|
|
|
users.users.traefik = {
|
2017-09-27 17:30:49 +01:00
|
|
|
group = "traefik";
|
|
|
|
home = cfg.dataDir;
|
|
|
|
createHome = true;
|
2019-10-12 21:25:28 +01:00
|
|
|
isSystemUser = true;
|
2017-09-27 17:30:49 +01:00
|
|
|
};
|
|
|
|
|
2019-12-31 00:41:18 +00:00
|
|
|
users.groups.traefik = { };
|
2017-09-27 17:30:49 +01:00
|
|
|
};
|
|
|
|
}
|