Make ddns behave more standard conform, like SOA records

This commit is contained in:
Philipp Böhm 2014-09-21 17:27:26 +02:00
parent 7dbcc66298
commit c3641cfdda
3 changed files with 79 additions and 29 deletions

View File

@ -50,11 +50,11 @@ host.
$ sudo vim /etc/powerdns/pdns.d/pipe.conf $ sudo vim /etc/powerdns/pdns.d/pipe.conf
`pipe.conf` should have the following content. Please adjust the path of `ddns` `pipe.conf` should have the following content. Please adjust the path of `ddns`
and the supplied domain name `--domain=sub.example.com`: and the values supplied to `--domain` and `--soa_fqdn`:
launch=pipe launch=pipe
pipebackend-abi-version=1 pipebackend-abi-version=1
pipe-command=/home/user/gocode/bin/ddns --domain=sub.example.com backend pipe-command=/home/user/gocode/bin/ddns --soa_fqdn=dns.example.com --domain=sub.example.com backend
Then restart `pdns`: Then restart `pdns`:

View File

@ -5,60 +5,95 @@ import (
"fmt" "fmt"
"os" "os"
"strings" "strings"
"time"
) )
type responder func()
func respondWithFAIL() {
fmt.Printf("FAIL\n")
}
func respondWithEND() {
fmt.Printf("END\n")
}
// This function implements the PowerDNS-Pipe-Backend protocol and generates // This function implements the PowerDNS-Pipe-Backend protocol and generates
// the response data it possible // the response data it possible
func RunBackend(conn *RedisConnection) { func RunBackend(conn *RedisConnection) {
bio := bufio.NewReader(os.Stdin)
// handshake with PowerDNS // handshake with PowerDNS
_, _, _ = bio.ReadLine()
fmt.Printf("OK\tDDNS Go Backend\n") fmt.Printf("OK\tDDNS Go Backend\n")
bio := bufio.NewReader(os.Stdin)
for { for {
line, _, err := bio.ReadLine() line, _, err := bio.ReadLine()
if err != nil {
respondWithFAIL()
continue
}
HandleErr(err) HandleRequest(string(line), conn)()
HandleRequest(string(line), conn)
} }
} }
func HandleRequest(line string, conn *RedisConnection) { func HandleRequest(line string, conn *RedisConnection) responder {
defer fmt.Printf("END\n")
if Verbose { if Verbose {
fmt.Printf("LOG\t'%s'\n", line) fmt.Printf("LOG\t'%s'\n", line)
} }
parts := strings.Split(line, "\t") parts := strings.Split(line, "\t")
if len(parts) != 6 { if len(parts) != 6 {
return return respondWithFAIL
} }
query_name := parts[1] query_name := parts[1]
query_class := parts[2] query_class := parts[2]
// query_type := parts[3] // TODO Handle SOA Requests query_type := parts[3]
query_id := parts[4] query_id := parts[4]
// get the host part of the fqdn var response, record string
// pi.d.example.org -> pi record = query_type
switch query_type {
case "SOA":
response = fmt.Sprintf("%s. hostmaster.example.com. %d 1800 3600 7200 5",
DdnsSoaFqdn, getSoaSerial())
case "NS":
response = DdnsSoaFqdn
case "A":
case "ANY":
// get the host part of the fqdn: pi.d.example.org -> pi
hostname := "" hostname := ""
if strings.HasSuffix(query_name, DdnsDomain) { if strings.HasSuffix(query_name, DdnsDomain) {
hostname = query_name[:len(query_name)-len(DdnsDomain)] hostname = query_name[:len(query_name)-len(DdnsDomain)]
} }
if hostname == "" || !conn.HostExist(hostname) { if hostname == "" || !conn.HostExist(hostname) {
return return respondWithFAIL
} }
host := conn.GetHost(hostname) host := conn.GetHost(hostname)
response = host.Ip
record := "A" record = "A"
if !host.IsIPv4() { if !host.IsIPv4() {
record = "AAAA" record = "AAAA"
} }
fmt.Printf("DATA\t%s\t%s\t%s\t10\t%s\t%s\n", default:
query_name, query_class, record, query_id, host.Ip) return respondWithFAIL
}
fmt.Printf("DATA\t%s\t%s\t%s\t10\t%s\t%s\n",
query_name, query_class, record, query_id, response)
return respondWithEND
}
func getSoaSerial() int64 {
// return current time in milliseconds
return time.Now().UnixNano()
} }

23
ddns.go
View File

@ -12,10 +12,16 @@ func HandleErr(err error) {
} }
} }
const (
CmdBackend string = "backend"
CmdWeb string = "web"
)
var ( var (
DdnsDomain string DdnsDomain string
DdnsWebListenSocket string DdnsWebListenSocket string
DdnsRedisHost string DdnsRedisHost string
DdnsSoaFqdn string
Verbose bool Verbose bool
) )
@ -29,28 +35,37 @@ func init() {
flag.StringVar(&DdnsRedisHost, "redis", ":6379", flag.StringVar(&DdnsRedisHost, "redis", ":6379",
"The Redis socket that should be used") "The Redis socket that should be used")
flag.StringVar(&DdnsSoaFqdn, "soa_fqdn", "",
"The FQDN of the DNS server which is returned as a SOA record")
flag.BoolVar(&Verbose, "verbose", false, flag.BoolVar(&Verbose, "verbose", false,
"Be more verbose") "Be more verbose")
} }
func ValidateCommandArgs() { func ValidateCommandArgs(cmd string) {
if DdnsDomain == "" { if DdnsDomain == "" {
log.Fatal("You have to supply the domain via --domain=DOMAIN") log.Fatal("You have to supply the domain via --domain=DOMAIN")
} else if !strings.HasPrefix(DdnsDomain, ".") { } else if !strings.HasPrefix(DdnsDomain, ".") {
// get the domain in the right format // get the domain in the right format
DdnsDomain = "." + DdnsDomain DdnsDomain = "." + DdnsDomain
} }
if cmd == CmdBackend {
if DdnsSoaFqdn == "" {
log.Fatal("You have to supply the server FQDN via --soa_fqdn=FQDN")
}
}
} }
func PrepareForExecution() string { func PrepareForExecution() string {
flag.Parse() flag.Parse()
ValidateCommandArgs()
if len(flag.Args()) != 1 { if len(flag.Args()) != 1 {
usage() usage()
} }
cmd := flag.Args()[0] cmd := flag.Args()[0]
ValidateCommandArgs(cmd)
return cmd return cmd
} }
@ -61,10 +76,10 @@ func main() {
defer conn.Close() defer conn.Close()
switch cmd { switch cmd {
case "backend": case CmdBackend:
log.Printf("Starting PDNS Backend\n") log.Printf("Starting PDNS Backend\n")
RunBackend(conn) RunBackend(conn)
case "web": case CmdWeb:
log.Printf("Starting Web Service\n") log.Printf("Starting Web Service\n")
RunWebService(conn) RunWebService(conn)
default: default: