Skip to content

Mydyns implements a HTTP API to update a dynamic DNS zone

License

Notifications You must be signed in to change notification settings

longsleep/mydyns

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

55 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Mydyns

Mydyns implements a HTTP API to update a dynamic DNS zone by adding or removing A and AAAA records from a DNS zone. Mydyns uses the nsupdate utility to submit Dynamic DNS Update requests as defined in RFC 2136 to a name server.

Build requirements

  • Go >= 1.13

Runtime requirements

  • nsupdate (Found in dnsutils provided with BIND)

Building

$ make

Configuration of users and hosts

Mydyns requires a users database and a hosts database. Both are simple text files.

Users database users.db

The users database can be managed with htpasswd from Apache. Make sure to use SHA for password hashing.

$ htpasswd -c -s users.db myuser

Hosts database hosts.db

The hosts database is a simple text file listing one host per line. In addition, the hosts have to be mapped to users. Users are added after the host, followed by a colon. Multiple users should be comma-separated.

somehost:usera,userb
otherhost:userc

Security database security.db

The security database is a simple text file listing one user with the current security code for this user. The entry is optional. The security code can be used to expire all existing tokens for this user. Tokens must always have the current security code, else they are not valid and useless. Change or set the security code, if a token becomes stolen. All tokens of a single user use the same security code.

usera:current security code
userb:supercode

DNS configuration and key

Mydyns sends updates to an upstream Bind DNS server using the nsupdate utility to send Dynamic DNS Update requests to a name server. This requires authentication, so you need to generate a DNSSec key which is used to connect to the DNS server and allows the update.

$ dnssec-keygen -a HMAC-SHA256 -b 256 -n HOST your.dns.zone

This creates a public and private key. Add the public key to allow updates to your DNS zone, and use the private key file when starting mydynsd.

Tokens

Mydyns uses tokens to authenticate /update requests for hosts. The token contains a HMAC of the user and the host. The secret for creating the HMAC is read from a file passed via the --secret parameter. You shoud generate the file with some random data.

$ dd if=/dev/urandom of=secret.key bs=1 count=32

The length of the key should be 32 or 64 bytes.

Startup

$ ./mydynsd \
	--server=your.name.server \
	--key=dnssec.key.private \
	--zone=your.dns.zone \
	--users=users.db \
	--hosts=hosts.db \
	--security=security.db \
	--secret=secret.key \
	--listen=127.0.0.1:8040 \
	--ttl=60

While the server is running, you can send the HUP signal to make it reload the database files for users, hosts and security. All other changes require a full restart.

HTTP API

The server provides HTTP API endpoints.

/token

The first one is /token which is used to generate an update token for a host. The /token endpoint requires HTTP Basic authentication to provide the user and password. When successfully authenticated and the user is listed in the hosts database for the provides hostname, the token is returned. This token value can then be used to use /update for that hostname.

$ curl -u user:password https://yourserver/token?hostname=myhost

/update

To send an update request, use the /update endpoint with the token parameter. When no further parameters are passed, it will set the IP address where the request came from for the hostname encoded in the token. You can also pass the IP address manually with the myip parameter. For compatibility reasons, the value auto and the address parameter are also supported. To only return the current IP without changing anything, pass the check parameter.

$ curl https://yourserver/update?token=tokenvalue

There is an update script example in the scripts directory which you can use to run from cron or similar. Also check the extra directory for some ideas on how to run the daemon as an upstart service.

Expose service to the Internet

Mydyns runs on the local interface by default. If you want to expose the service to the public Internet, you should run it behind a transparent proxy like Nginx to provide TLS encryption. For auto-detection of the remote IP addresses to work, make sure that the proxy injects the remote IP address in the X-Real-IP HTTP request header.

Nginx example

location ~* /(token|update)$ {
	proxy_pass http://127.0.0.1:8040;
	proxy_set_header Host $http_host;
	proxy_set_header X-Real-IP $remote_addr;
}

Docker

The Dockerfile can be used to build a Docker images to run mydynsd in a container.

Building Docker container

Running this will build you a minimal Docker image including the nsupdate utility. As the image is minimal, it is using a static build of mydynsd to avoid system dependencies.

$ sudo docker build -t longsleep/mydynsd -f Dockerfile .

Running Docker container

Running it for the first time will set up the location of the configuration and the port of your choice. Make sure to put all the configuration files and keys into a single folder and mount this folder as /data into Docker. Pass all the parameters to your files relatively to the /data directory of the container. Create all the files before running the container for the first time.

$ sudo docker run --rm=true -p=127.0.0.1:8040:8040 -v=/mnt/mydyns:/data --sig-proxy=true -it longsleep/mydynsd /app/mydynsd --server=your.name.server --key=/data/dnssec.key.private --zone=your.dns.zone --users=/data/users.db --hosts=/data/hosts.db --security=/data/security.db --secret=/data/secret.key --listen=0.0.0.0:8040 --ttl=60 --log=/data/mydynsd.log

This runs a Mydyns in a temporary container in foreground. To stop it, just press CTRL+C. To run the container as a system service, check out the upstart example in extras/mydynsd-container.upstart. When running properly with upstart, doing a reload mydynsd-container will reload the databases, without interrupting the service.

-- Simon Eisenmann - mailto:[email protected]