2014-04-14 15:26:48 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
2009-10-12 17:36:19 +01:00
|
|
|
|
2014-04-14 15:26:48 +01:00
|
|
|
with lib;
|
2009-03-06 12:27:13 +00:00
|
|
|
|
|
|
|
let
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
cfg = config.services.bind;
|
|
|
|
|
2021-07-11 14:11:33 +01:00
|
|
|
bindPkg = config.services.bind.package;
|
|
|
|
|
2012-07-26 23:07:25 +01:00
|
|
|
bindUser = "named";
|
|
|
|
|
2021-04-29 16:47:43 +01:00
|
|
|
bindZoneCoerce = list: builtins.listToAttrs (lib.forEach list (zone: { name = zone.name; value = zone; }));
|
|
|
|
|
|
|
|
bindZoneOptions = { name, config, ... }: {
|
|
|
|
options = {
|
|
|
|
name = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = name;
|
|
|
|
description = "Name of the zone.";
|
|
|
|
};
|
|
|
|
master = mkOption {
|
|
|
|
description = "Master=false means slave server";
|
|
|
|
type = types.bool;
|
|
|
|
};
|
|
|
|
file = mkOption {
|
|
|
|
type = types.either types.str types.path;
|
|
|
|
description = "Zone file resource records contain columns of data, separated by whitespace, that define the record.";
|
|
|
|
};
|
|
|
|
masters = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
description = "List of servers for inclusion in stub and secondary zones.";
|
|
|
|
};
|
|
|
|
slaves = mkOption {
|
|
|
|
type = types.listOf types.str;
|
|
|
|
description = "Addresses who may request zone transfers.";
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ ];
|
2021-04-29 16:47:43 +01:00
|
|
|
};
|
|
|
|
extraConfig = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
description = "Extra zone config to be appended at the end of the zone section.";
|
|
|
|
default = "";
|
|
|
|
};
|
2021-01-20 10:47:49 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
confFile = pkgs.writeText "named.conf"
|
|
|
|
''
|
2017-06-17 20:05:07 +01:00
|
|
|
include "/etc/bind/rndc.key";
|
|
|
|
controls {
|
|
|
|
inet 127.0.0.1 allow {localhost;} keys {"rndc-key";};
|
|
|
|
};
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
acl cachenetworks { ${concatMapStrings (entry: " ${entry}; ") cfg.cacheNetworks} };
|
|
|
|
acl badnetworks { ${concatMapStrings (entry: " ${entry}; ") cfg.blockedNetworks} };
|
|
|
|
|
|
|
|
options {
|
2017-06-10 11:19:07 +01:00
|
|
|
listen-on { ${concatMapStrings (entry: " ${entry}; ") cfg.listenOn} };
|
|
|
|
listen-on-v6 { ${concatMapStrings (entry: " ${entry}; ") cfg.listenOnIpv6} };
|
2009-10-12 17:36:19 +01:00
|
|
|
allow-query { cachenetworks; };
|
|
|
|
blackhole { badnetworks; };
|
2022-01-01 14:33:51 +00:00
|
|
|
forward ${cfg.forward};
|
2013-08-26 20:36:01 +01:00
|
|
|
forwarders { ${concatMapStrings (entry: " ${entry}; ") cfg.forwarders} };
|
2021-07-17 07:03:20 +01:00
|
|
|
directory "${cfg.directory}";
|
2018-12-19 21:39:28 +00:00
|
|
|
pid-file "/run/named/named.pid";
|
2018-05-07 02:05:30 +01:00
|
|
|
${cfg.extraOptions}
|
2009-03-06 12:27:13 +00:00
|
|
|
};
|
2009-10-12 17:36:19 +01:00
|
|
|
|
2015-09-29 10:50:39 +01:00
|
|
|
${cfg.extraConfig}
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
${ concatMapStrings
|
2019-07-20 16:36:05 +01:00
|
|
|
({ name, file, master ? true, slaves ? [], masters ? [], extraConfig ? "" }:
|
2011-09-14 19:20:50 +01:00
|
|
|
''
|
2009-10-12 17:36:19 +01:00
|
|
|
zone "${name}" {
|
|
|
|
type ${if master then "master" else "slave"};
|
|
|
|
file "${file}";
|
|
|
|
${ if master then
|
|
|
|
''
|
|
|
|
allow-transfer {
|
|
|
|
${concatMapStrings (ip: "${ip};\n") slaves}
|
|
|
|
};
|
|
|
|
''
|
|
|
|
else
|
|
|
|
''
|
|
|
|
masters {
|
|
|
|
${concatMapStrings (ip: "${ip};\n") masters}
|
|
|
|
};
|
|
|
|
''
|
|
|
|
}
|
|
|
|
allow-query { any; };
|
2019-07-20 16:36:05 +01:00
|
|
|
${extraConfig}
|
2009-10-12 17:36:19 +01:00
|
|
|
};
|
|
|
|
'')
|
2021-04-29 16:47:43 +01:00
|
|
|
(attrValues cfg.zones) }
|
2009-10-12 17:36:19 +01:00
|
|
|
'';
|
|
|
|
|
2009-03-06 12:27:13 +00:00
|
|
|
in
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
{
|
2009-03-06 12:27:13 +00:00
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
services.bind = {
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2020-04-20 19:05:26 +01:00
|
|
|
enable = mkEnableOption "BIND domain name server";
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2021-07-11 14:11:33 +01:00
|
|
|
|
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.bind;
|
2021-10-03 17:06:03 +01:00
|
|
|
defaultText = literalExpression "pkgs.bind";
|
2021-07-11 14:11:33 +01:00
|
|
|
description = "The BIND package to use.";
|
|
|
|
};
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
cacheNetworks = mkOption {
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ "127.0.0.0/24" ];
|
2021-01-20 10:47:49 +00:00
|
|
|
type = types.listOf types.str;
|
2009-10-12 17:36:19 +01:00
|
|
|
description = "
|
2019-10-23 16:50:47 +01:00
|
|
|
What networks are allowed to use us as a resolver. Note
|
|
|
|
that this is for recursive queries -- all networks are
|
|
|
|
allowed to query zones configured with the `zones` option.
|
|
|
|
It is recommended that you limit cacheNetworks to avoid your
|
|
|
|
server being used for DNS amplification attacks.
|
2009-10-12 17:36:19 +01:00
|
|
|
";
|
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
blockedNetworks = mkOption {
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ ];
|
2021-01-20 10:47:49 +00:00
|
|
|
type = types.listOf types.str;
|
2009-10-12 17:36:19 +01:00
|
|
|
description = "
|
|
|
|
What networks are just blocked.
|
|
|
|
";
|
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2012-06-10 15:50:44 +01:00
|
|
|
ipv4Only = mkOption {
|
|
|
|
default = false;
|
2021-01-20 10:47:49 +00:00
|
|
|
type = types.bool;
|
2012-06-10 15:50:44 +01:00
|
|
|
description = "
|
2013-08-26 20:36:01 +01:00
|
|
|
Only use ipv4, even if the host supports ipv6.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
|
|
|
forwarders = mkOption {
|
|
|
|
default = config.networking.nameservers;
|
2021-11-26 00:16:05 +00:00
|
|
|
defaultText = literalExpression "config.networking.nameservers";
|
2021-01-20 10:47:49 +00:00
|
|
|
type = types.listOf types.str;
|
2013-08-26 20:36:01 +01:00
|
|
|
description = "
|
|
|
|
List of servers we should forward requests to.
|
2012-06-10 15:50:44 +01:00
|
|
|
";
|
|
|
|
};
|
|
|
|
|
2022-01-01 14:33:51 +00:00
|
|
|
forward = mkOption {
|
|
|
|
default = "first";
|
|
|
|
type = types.enum ["first" "only"];
|
|
|
|
description = "
|
|
|
|
Whether to forward 'first' (try forwarding but lookup directly if forwarding fails) or 'only'.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
2017-06-10 11:19:07 +01:00
|
|
|
listenOn = mkOption {
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ "any" ];
|
2017-06-10 11:19:07 +01:00
|
|
|
type = types.listOf types.str;
|
|
|
|
description = "
|
|
|
|
Interfaces to listen on.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
|
|
|
listenOnIpv6 = mkOption {
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ "any" ];
|
2017-06-10 11:19:07 +01:00
|
|
|
type = types.listOf types.str;
|
|
|
|
description = "
|
|
|
|
Ipv6 interfaces to listen on.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
2021-07-17 07:03:20 +01:00
|
|
|
directory = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "/run/named";
|
|
|
|
description = "Working directory of BIND.";
|
|
|
|
};
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
zones = mkOption {
|
2021-05-08 22:13:58 +01:00
|
|
|
default = [ ];
|
2021-04-29 16:47:43 +01:00
|
|
|
type = with types; coercedTo (listOf attrs) bindZoneCoerce (attrsOf (types.submodule bindZoneOptions));
|
2009-10-12 17:36:19 +01:00
|
|
|
description = "
|
|
|
|
List of zones we claim authority over.
|
|
|
|
";
|
2021-04-29 16:47:43 +01:00
|
|
|
example = {
|
|
|
|
"example.com" = {
|
|
|
|
master = false;
|
|
|
|
file = "/var/dns/example.com";
|
2021-05-08 22:13:58 +01:00
|
|
|
masters = [ "192.168.0.1" ];
|
|
|
|
slaves = [ ];
|
2021-04-29 16:47:43 +01:00
|
|
|
extraConfig = "";
|
|
|
|
};
|
|
|
|
};
|
2009-10-12 17:36:19 +01:00
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2015-09-29 10:50:39 +01:00
|
|
|
extraConfig = mkOption {
|
2016-10-23 18:33:41 +01:00
|
|
|
type = types.lines;
|
2015-09-29 10:50:39 +01:00
|
|
|
default = "";
|
|
|
|
description = "
|
|
|
|
Extra lines to be added verbatim to the generated named configuration file.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
2018-05-07 02:05:30 +01:00
|
|
|
extraOptions = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
description = ''
|
|
|
|
Extra lines to be added verbatim to the options section of the
|
|
|
|
generated named configuration file.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2011-11-03 19:13:52 +00:00
|
|
|
configFile = mkOption {
|
2016-01-17 18:34:55 +00:00
|
|
|
type = types.path;
|
2011-11-03 19:13:52 +00:00
|
|
|
default = confFile;
|
2021-10-03 17:06:03 +01:00
|
|
|
defaultText = literalExpression "confFile";
|
2011-11-03 19:13:52 +00:00
|
|
|
description = "
|
|
|
|
Overridable config file to use for named. By default, that
|
|
|
|
generated by nixos.
|
|
|
|
";
|
|
|
|
};
|
|
|
|
|
2008-07-01 13:15:56 +01:00
|
|
|
};
|
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2008-07-01 13:15:56 +01:00
|
|
|
|
2009-10-12 17:36:19 +01:00
|
|
|
###### implementation
|
2008-07-01 13:15:56 +01:00
|
|
|
|
2019-07-15 18:18:49 +01:00
|
|
|
config = mkIf cfg.enable {
|
|
|
|
|
|
|
|
networking.resolvconf.useLocalResolver = mkDefault true;
|
2009-03-06 12:27:13 +00:00
|
|
|
|
2019-09-14 18:51:29 +01:00
|
|
|
users.users.${bindUser} =
|
2021-05-08 22:13:58 +01:00
|
|
|
{
|
2021-08-08 13:00:00 +01:00
|
|
|
group = bindUser;
|
2012-07-26 23:07:25 +01:00
|
|
|
description = "BIND daemon user";
|
2021-08-08 13:00:00 +01:00
|
|
|
isSystemUser = true;
|
2012-07-26 23:07:25 +01:00
|
|
|
};
|
2021-08-08 13:00:00 +01:00
|
|
|
users.groups.${bindUser} = {};
|
2012-07-26 23:07:25 +01:00
|
|
|
|
2016-01-06 06:50:18 +00:00
|
|
|
systemd.services.bind = {
|
2016-09-08 21:19:43 +01:00
|
|
|
description = "BIND Domain Name Server";
|
2016-09-10 19:14:09 +01:00
|
|
|
after = [ "network.target" ];
|
2016-01-06 06:50:18 +00:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
2009-03-06 12:27:13 +00:00
|
|
|
|
2016-01-06 06:50:18 +00:00
|
|
|
preStart = ''
|
2017-06-17 20:05:07 +01:00
|
|
|
mkdir -m 0755 -p /etc/bind
|
|
|
|
if ! [ -f "/etc/bind/rndc.key" ]; then
|
2021-07-11 14:11:33 +01:00
|
|
|
${bindPkg.out}/sbin/rndc-confgen -c /etc/bind/rndc.key -u ${bindUser} -a -A hmac-sha256 2>/dev/null
|
2017-06-17 20:05:07 +01:00
|
|
|
fi
|
|
|
|
|
2018-12-19 21:39:28 +00:00
|
|
|
${pkgs.coreutils}/bin/mkdir -p /run/named
|
|
|
|
chown ${bindUser} /run/named
|
2021-07-17 07:03:20 +01:00
|
|
|
|
|
|
|
${pkgs.coreutils}/bin/mkdir -p ${cfg.directory}
|
|
|
|
chown ${bindUser} ${cfg.directory}
|
2016-01-06 06:50:18 +00:00
|
|
|
'';
|
2008-07-01 13:15:56 +01:00
|
|
|
|
2017-06-17 20:05:07 +01:00
|
|
|
serviceConfig = {
|
2021-07-11 14:11:33 +01:00
|
|
|
ExecStart = "${bindPkg.out}/sbin/named -u ${bindUser} ${optionalString cfg.ipv4Only "-4"} -c ${cfg.configFile} -f";
|
|
|
|
ExecReload = "${bindPkg.out}/sbin/rndc -k '/etc/bind/rndc.key' reload";
|
|
|
|
ExecStop = "${bindPkg.out}/sbin/rndc -k '/etc/bind/rndc.key' stop";
|
2017-06-17 20:05:07 +01:00
|
|
|
};
|
|
|
|
|
2016-09-08 21:19:43 +01:00
|
|
|
unitConfig.Documentation = "man:named(8)";
|
2016-01-06 06:50:18 +00:00
|
|
|
};
|
2009-03-06 12:27:13 +00:00
|
|
|
};
|
|
|
|
}
|