BIND: Enabling TSIG for Zone Transfers

ISC BIND

My evening fun one day this week was to enable Transaction Signature (TSIG) capability for zone transfers on my home DNS servers. Yes, I know.

Why? Well, partly because it’s a good idea from a security perspective, and mainly because, well, it was there. Let’s face it, the security issues are probably limited when it comes to my home DNS servers, but that doesn’t stop me doing it for kicks. And since it’s pretty simple, let’s document it, shall we?

Home DNS

I can’t be bothered maintaining hosts files on multiple devices, but I really do like to have name resolution working, and given that I currently have around 50 devices on my home network this is perhaps less of stupid idea than it sounds. I also run IPv6, and the one thing I maintain about IPv6 is that you had better have good DNS, because you’ll never want to type an IPv6 address out even if you can remember it.

To that end, I currently have three DNS servers active, all running Ubuntu server. They are configured like this:

DNS Architecture

DNS1 is the zone master. DNS2 is set up as a secondary, slaved to DNS1. DNS3 is another secondary, but for fun, rather than getting zone transfers from DNS1, it instead gets a transfer from DNS2. There’s no particular reason for that in my environment other than that I wanted to configure it that way.

Configuring TSIG On The Master DNS Server

Generating Keys

Go to the server that is the master for the zone(s) for which you want to use with TSIG. TSIG needs a key to be generated, and for that we’ll use dnssec-keygen, which is a tool (included with BIND) that generates DNSSEC and TSIG keys. To tell dnssec-keygen that we’re generating a host key rather than a DNSSEC zone key we use the ‘-n HOST’ argument, and in this case we’ll call it “tsigkey”, but it really doesn’t matter too much what you call it:

dnssec-keygen -a HMAC-SHA512 -b 512 -n HOST -r /dev/urandom tsigkey

Running this command will generate two files with names based on ‘tsigkey’:

-rw-rw-r-- 1 john john 116 2013-06-05 20:10 Ktsigkey.+165+49047.key
-rw------- 1 john john 232 2013-06-05 20:10 Ktsigkey.+165+49047.private

If you look inside the .private file, it looks like this:

john@dns1:~$ cat Ktsigkey.+165+49047.private
Private-key-format: v1.3
Algorithm: 165 (HMAC_SHA512)
Key: Ok1qR5IW1ajVka5cHPEJQIXfLyx5V3PSkFBROAzOn21JumDq6nIpoj6H8rfj5Uo+Ok55ZWQ0Wgrf302fDscHLw==
[...]

The only bit we care about is the big string of characters after the Key: label. You’ll want to copy that text, because next we put it in a BIND configuration file. Edit a new file (in this case I use vi) named as you please – in this case I’m mimicking the name of the other bind configuration files, but you certainly don’t have to:

sudo vi /etc/bind/named.conf.tsigkeys

..and complete it as follows. Make sure the key is in quotes, and look out for the semicolons! I’ve called this key “my-tsig” because that’s the kind of thing I’m sure you’re expecting!

key "my-tsig" {
 algorithm HMAC-SHA512;
 secret "Ok1qR5IW1ajVka5cHPEJQIXfLyx5V3PSkFBROAzOn21JumDq6nIpoj6H8rfj5Uo+Ok55ZWQ0Wgrf302fDscHLw==";
};

Save the named.conf.tsigkeys file.

Tell BIND About The Keys

Next we’ll hook the key definition file into BIND. Edit the named.conf file:

sudo vi /etc/bind/named.conf

…and add in the new file we just created at the end:

include "/etc/bind/named.conf.options";
include "/etc/bind/named.conf.local";
include "/etc/bind/named.conf.default-zones";
// ADD THE LINE BELOW
include "/etc/bind/named.conf.tsigkeys";

So now when BIND loads, it will also include the key definition for “my-tsig”.

Secure Specific Zones

We need to tell BIND which zones we want to apply the TSIG key to. In this case, I’m choosing to apply it only to one zone out of a few that are in my configuration, so I’m going to add the key information within the zone configuration under the allow-transfer element:

zone "mysecuredzone.com" {
 type master;
 file "/etc/bind/zones/db.mysecuredzone.com";
 allow-transfer {
  key "my-tsig";
 };
};

Note that by putting the word ‘key’ in front of the name, it tells BIND that this is a TSIG key rather than an ACL name. You’ll also note that I don’t have to have the IP address of the secondary (slave) servers listed, because what you’re saying here is that any server that has the right key will be allowed to perform zone transfer. In fact if you do add in the IP address of your secondary, you will be allowing that server to do non-TSIG transfers, which is probably not what you had intended.

Now restart BIND:

sudo /etc/init.d/bind9 restart

You should see [OK] when the service restarts. If you get a ‘fail’, then check the syslog to see what the error was.

* Stopping domain name service... bind9 [ OK ]
* Starting domain name service... bind9 [ OK ]

managed-keys-bind

Take a look in syslog anyway. You may well see an error like this one:

Jun  5 21:13:37 dns1 named[28543]: managed-keys-zone ./IN: loading from master file managed-keys.bind failed: file not found

This isn’t fatal, but it’s annoying and the fix is simple:

sudo touch /var/cache/bind/managed-keys.bind

The master server for my “mysecureddomain.com” domain is now ready to rock and roll. We just need to configure the secondary, so let’s do that next.

Configuring The Secondary Server

On the secondary server, create the same keyfile you created on the master:

sudo vi /etc/bind/named.conf.tsigkeys

…and copy in the same content, but this time also add an entry for your master server telling it which key to use. In this example I’m using an example IP of 192.168.2.254 as my master server’s IP:

key "my-tsig" {
 algorithm HMAC-SHA512;
 secret "Ok1qR5IW1ajVka5cHPEJQIXfLyx5V3PSkFBROAzOn21JumDq6nIpoj6H8rfj5Uo+Ok55ZWQ0Wgrf302fDscHLw==";
};

server 192.168.2.254 {
 keys { my-tsig; };
};

Then go hook it into BIND in the same way as on the master, by editing named.conf:

sudo vi /etc/bind/named.conf

and adding this line:

include "/etc/bind/named.conf.tsigkeys";

Then restart BIND:

sudo /etc/init.d/bind9 restart

In theory, you’re now ready to roll.

Testing TSIG

Go to the secondary and try grabbing a zone transfer (put in your master dns server’s IP and choose the appropriate domain name):

dig @{master-dns-ip} mysecuredomain.com axfr

With any luck, you’ll get on error like this:

; <<>> DiG 9.7.3 <<>> @dns1 mysecuredomain.com axfr
; (1 server found)
;; global options: +cmd
; Transfer failed.

This is a good thing because it means a non-TSIG transfer is being denied! Now let’s try the same thing but this time force dig to use the TSIG key (which means using sudo):

sudo dig @{master-dns-ip} mysecuredomain.com axfr -k /etc/bind/named.conf.tsigkeys

Hopefully you will now see a zone transfer scroll by! If you do, TSIG is working.

Slave of a Slave

It’s a bit meta, but in my case dns3 is a slave to dns2 (which is a slave to dns1). From dns3’s perspective, dns2 is its master. To that end, dns2 needs to be configured a little bit like a master too.

On DNS3, we follow the same steps we followed for DNS2 above.

Then on the DNS2 slave, we configure each secured zone in named.conf.local with the allow-transfer element:

allow-transfer {
 key "my-tsig";
};

…just like we did on the master (DNS1). DNS3’s configuration tells it that DNS2 is the master, so it will be looking in the right place for updates. Restart BIND on DNS2 and DNS3 so that the changes take effect, and we can now repeat the TSIG testing above, this time issuing the dig commands on DNS3, and using DNS2’s IP address as the target.

And That’s It!

It really was that simple. You too can go completely overboard on your home network.

Coming soon: forcing visitors to my home to use guest WiFi, two factor authentication and a VPN client if they want to access my network…

 

TMTOWTDI

Echoing the perl mantra that there’s more than one way to do it, it should be noted that this is just one way to configure things. If you work with BIND much you’ll know that configuration can go in multiple places and be abstracted to quite some degree through server definitions, acls, and so forth. So please understand this is just one way to do it that works in my small home environment, and may not be the ideal way to do it in, say, a large enterprise!

 

Updated 2013/6/10: Oops – I forgot to include the master server configuration on the secondary server’s named.conf.tsigkeys configuration. Added description and code. Also fixed reversed directory names in touch command.

14 Comments on BIND: Enabling TSIG for Zone Transfers

  1. You are probably the only person I know using TSIG on their home network, or that has 3 DNS servers in their home network. I’d guess that your home network is probably more advanced/secure/larger than the majority of small business networks.

    Cheers!

    • That’s probably a good thing, I would imagine! You know the problem though – geek uses home network as their personal lab…

  2. Good information and easy to follow. Had to use a bit different syntax to get the zone transfer to work in our environment but the article still got me in the right direction.

  3. I just read your tutorial and it really helped me!
    But I think there is a crucial point missing:
    One should ensure that the permissions for the file named.conf.tsigkeys are matching those of the Ktsigkey.+165+49047.private file (nobody except root can read it, since it contains the private key):

    chmod 600 /etc/bind/named.conf.tsigkeys

  4. why do you use private keys and not public key in your conf file?

    john@dns1:~$ cat Ktsigkey.+165+49047.private

    instead of

    john@dns1:~$ cat Ktsigkey.+165+49047.key

    • Good question; I should probably have included a little bit more background on this before diving in.

      The TSIG keys in this case are being used as a shared secret between any given pair of BIND servers, so we’re really just using the dnssec-keygen tool as a handy way to generate a nice secure shared secret that’s already encoded in Base64 as required by BIND. ISC (authors of BIND) use this mechanism themselves in their documentation, e.g. http://ftp.isc.org/isc/bind/9.6.3/doc/arm/Bv9ARM.ch04.html#id2571082

      You could achieve a similar result at the command line, so long as the key is a multiple of 4 characters long (if I understand the documentation correctly):

      #echo -n "thisismysupersecretsharedkey" | openssl base64
      dGhpc2lzbXlzdXBlcnNlY3JldHNoYXJlZGtleQ==
      

      …but why put yourself out creating a likely not-very-random key when you can just let something else do it for you?

      As the tool name “dnssec-keygen” suggests, it’s also able to create and sign public/private keypairs for use with dnssec to secure zones, but that’s not how we’re using it here.

2 Trackbacks & Pingbacks

  1. Back to BIND (with OpenDNSSEC) - FOSS Notes
  2. Back to BIND (with OpenDNSSEC) – FOSS Notes

Leave a Reply

Your email address will not be published.


*


 

This site uses Akismet to reduce spam. Learn how your comment data is processed.