mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-12 10:27:57 +08:00
328 lines
7.8 KiB
Go
328 lines
7.8 KiB
Go
// Generate zonefiles.
|
|
// This generates a zonefile that prioritizes beauty over efficiency.
|
|
package bind
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/miekg/dns/dnsutil"
|
|
)
|
|
|
|
type zoneGenData struct {
|
|
Origin string
|
|
DefaultTtl uint32
|
|
Records []dns.RR
|
|
}
|
|
|
|
func (z *zoneGenData) Len() int { return len(z.Records) }
|
|
func (z *zoneGenData) Swap(i, j int) { z.Records[i], z.Records[j] = z.Records[j], z.Records[i] }
|
|
func (z *zoneGenData) Less(i, j int) bool {
|
|
a, b := z.Records[i], z.Records[j]
|
|
compA, compB := dnsutil.AddOrigin(a.Header().Name, z.Origin+"."), dnsutil.AddOrigin(b.Header().Name, z.Origin+".")
|
|
if compA != compB {
|
|
if compA == z.Origin+"." {
|
|
compA = "@"
|
|
}
|
|
if compB == z.Origin+"." {
|
|
compB = "@"
|
|
}
|
|
return zoneLabelLess(compA, compB)
|
|
}
|
|
rrtypeA, rrtypeB := a.Header().Rrtype, b.Header().Rrtype
|
|
if rrtypeA != rrtypeB {
|
|
return zoneRrtypeLess(rrtypeA, rrtypeB)
|
|
}
|
|
switch rrtypeA { // #rtype_variations
|
|
case dns.TypeNS, dns.TypeTXT, dns.TypeTLSA:
|
|
// pass through.
|
|
case dns.TypeA:
|
|
ta2, tb2 := a.(*dns.A), b.(*dns.A)
|
|
ipa, ipb := ta2.A.To4(), tb2.A.To4()
|
|
if ipa == nil || ipb == nil {
|
|
log.Fatalf("should not happen: IPs are not 4 bytes: %#v %#v", ta2, tb2)
|
|
}
|
|
return bytes.Compare(ipa, ipb) == -1
|
|
case dns.TypeAAAA:
|
|
ta2, tb2 := a.(*dns.AAAA), b.(*dns.AAAA)
|
|
ipa, ipb := ta2.AAAA.To16(), tb2.AAAA.To16()
|
|
return bytes.Compare(ipa, ipb) == -1
|
|
case dns.TypeMX:
|
|
ta2, tb2 := a.(*dns.MX), b.(*dns.MX)
|
|
pa, pb := ta2.Preference, tb2.Preference
|
|
return pa < pb
|
|
case dns.TypeSRV:
|
|
ta2, tb2 := a.(*dns.SRV), b.(*dns.SRV)
|
|
pa, pb := ta2.Port, tb2.Port
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
pa, pb = ta2.Priority, tb2.Priority
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
pa, pb = ta2.Weight, tb2.Weight
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
case dns.TypePTR:
|
|
ta2, tb2 := a.(*dns.PTR), b.(*dns.PTR)
|
|
pa, pb := ta2.Ptr, tb2.Ptr
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
case dns.TypeCAA:
|
|
ta2, tb2 := a.(*dns.CAA), b.(*dns.CAA)
|
|
// sort by tag
|
|
pa, pb := ta2.Tag, tb2.Tag
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
// then flag
|
|
fa, fb := ta2.Flag, tb2.Flag
|
|
if fa != fb {
|
|
// flag set goes before ones without flag set
|
|
return fa > fb
|
|
}
|
|
default:
|
|
panic(fmt.Sprintf("zoneGenData Less: unimplemented rtype %v", dns.TypeToString[rrtypeA]))
|
|
// We panic so that we quickly find any switch statements
|
|
// that have not been updated for a new RR type.
|
|
}
|
|
return a.String() < b.String()
|
|
}
|
|
|
|
// mostCommonTtl returns the most common TTL in a set of records. If there is
|
|
// a tie, the highest TTL is selected. This makes the results consistent.
|
|
// NS records are not included in the analysis because Tom said so.
|
|
func mostCommonTtl(records []dns.RR) uint32 {
|
|
// Index the TTLs in use:
|
|
d := make(map[uint32]int)
|
|
for _, r := range records {
|
|
if r.Header().Rrtype != dns.TypeNS {
|
|
d[r.Header().Ttl]++
|
|
}
|
|
}
|
|
// Find the largest count:
|
|
var mc int
|
|
for _, value := range d {
|
|
if value > mc {
|
|
mc = value
|
|
}
|
|
}
|
|
// Find the largest key with that count:
|
|
var mk uint32
|
|
for key, value := range d {
|
|
if value == mc {
|
|
if key > mk {
|
|
mk = key
|
|
}
|
|
}
|
|
}
|
|
return mk
|
|
}
|
|
|
|
// WriteZoneFile writes a beautifully formatted zone file.
|
|
func WriteZoneFile(w io.Writer, records []dns.RR, origin string) error {
|
|
// This function prioritizes beauty over efficiency.
|
|
// * The zone records are sorted by label, grouped by subzones to
|
|
// be easy to read and pleasant to the eye.
|
|
// * Within a label, SOA and NS records are listed first.
|
|
// * MX records are sorted numericly by preference value.
|
|
// * SRV records are sorted numericly by port, then priority, then weight.
|
|
// * A records are sorted by IP address, not lexicographically.
|
|
// * Repeated labels are removed.
|
|
// * $TTL is used to eliminate clutter. The most common TTL value is used.
|
|
// * "@" is used instead of the apex domain name.
|
|
|
|
defaultTtl := mostCommonTtl(records)
|
|
|
|
z := &zoneGenData{
|
|
Origin: dnsutil.AddOrigin(origin, "."),
|
|
DefaultTtl: defaultTtl,
|
|
}
|
|
z.Records = nil
|
|
for _, r := range records {
|
|
z.Records = append(z.Records, r)
|
|
}
|
|
return z.generateZoneFileHelper(w)
|
|
}
|
|
|
|
// generateZoneFileHelper creates a pretty zonefile.
|
|
func (z *zoneGenData) generateZoneFileHelper(w io.Writer) error {
|
|
|
|
nameShortPrevious := ""
|
|
|
|
sort.Sort(z)
|
|
fmt.Fprintln(w, "$TTL", z.DefaultTtl)
|
|
for i, rr := range z.Records {
|
|
line := rr.String()
|
|
if line[0] == ';' {
|
|
continue
|
|
}
|
|
hdr := rr.Header()
|
|
|
|
items := strings.SplitN(line, "\t", 5)
|
|
if len(items) < 5 {
|
|
log.Fatalf("Too few items in: %v", line)
|
|
}
|
|
|
|
// items[0]: name
|
|
nameFqdn := hdr.Name
|
|
nameShort := dnsutil.TrimDomainName(nameFqdn, z.Origin)
|
|
name := nameShort
|
|
if i > 0 && nameShort == nameShortPrevious {
|
|
name = ""
|
|
} else {
|
|
name = nameShort
|
|
}
|
|
nameShortPrevious = nameShort
|
|
|
|
// items[1]: ttl
|
|
ttl := ""
|
|
if hdr.Ttl != z.DefaultTtl && hdr.Ttl != 0 {
|
|
ttl = items[1]
|
|
}
|
|
|
|
// items[2]: class
|
|
if hdr.Class != dns.ClassINET {
|
|
log.Fatalf("generateZoneFileHelper: Unimplemented class=%v", items[2])
|
|
}
|
|
|
|
// items[3]: type
|
|
typeStr := dns.TypeToString[hdr.Rrtype]
|
|
|
|
// items[4]: the remaining line
|
|
target := items[4]
|
|
//if typeStr == "TXT" {
|
|
// fmt.Printf("generateZoneFileHelper.go: target=%#v\n", target)
|
|
//}
|
|
|
|
fmt.Fprintln(w, formatLine([]int{10, 5, 2, 5, 0}, []string{name, ttl, "IN", typeStr, target}))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func formatLine(lengths []int, fields []string) string {
|
|
c := 0
|
|
result := ""
|
|
for i, length := range lengths {
|
|
item := fields[i]
|
|
for len(result) < c {
|
|
result += " "
|
|
}
|
|
if item != "" {
|
|
result += item + " "
|
|
}
|
|
c += length + 1
|
|
}
|
|
return strings.TrimRight(result, " ")
|
|
}
|
|
|
|
func isNumeric(s string) bool {
|
|
_, err := strconv.ParseFloat(s, 64)
|
|
return err == nil
|
|
}
|
|
|
|
func zoneLabelLess(a, b string) bool {
|
|
// Compare two zone labels for the purpose of sorting the RRs in a Zone.
|
|
|
|
// If they are equal, we are done. All other code is simplified
|
|
// because we can assume a!=b.
|
|
if a == b {
|
|
return false
|
|
}
|
|
|
|
// Sort @ at the top, then *, then everything else lexigraphically.
|
|
// i.e. @ always is less. * is is less than everything but @.
|
|
if a == "@" {
|
|
return true
|
|
}
|
|
if b == "@" {
|
|
return false
|
|
}
|
|
if a == "*" {
|
|
return true
|
|
}
|
|
if b == "*" {
|
|
return false
|
|
}
|
|
|
|
// Split into elements and match up last elements to first. Compare the
|
|
// first non-equal elements.
|
|
|
|
as := strings.Split(a, ".")
|
|
bs := strings.Split(b, ".")
|
|
ia := len(as) - 1
|
|
ib := len(bs) - 1
|
|
|
|
var min int
|
|
if ia < ib {
|
|
min = len(as) - 1
|
|
} else {
|
|
min = len(bs) - 1
|
|
}
|
|
|
|
// Skip the matching highest elements, then compare the next item.
|
|
for i, j := ia, ib; min >= 0; i, j, min = i-1, j-1, min-1 {
|
|
// Compare as[i] < bs[j]
|
|
// Sort @ at the top, then *, then everything else.
|
|
// i.e. @ always is less. * is is less than everything but @.
|
|
// If both are numeric, compare as integers, otherwise as strings.
|
|
|
|
if as[i] != bs[j] {
|
|
|
|
// If the first element is *, it is always less.
|
|
if i == 0 && as[i] == "*" {
|
|
return true
|
|
}
|
|
if j == 0 && bs[j] == "*" {
|
|
return false
|
|
}
|
|
|
|
// If the elements are both numeric, compare as integers:
|
|
au, aerr := strconv.ParseUint(as[i], 10, 64)
|
|
bu, berr := strconv.ParseUint(bs[j], 10, 64)
|
|
if aerr == nil && berr == nil {
|
|
return au < bu
|
|
} else {
|
|
// otherwise, compare as strings:
|
|
return as[i] < bs[j]
|
|
}
|
|
}
|
|
}
|
|
// The min top elements were equal, so the shorter name is less.
|
|
return ia < ib
|
|
}
|
|
|
|
func zoneRrtypeLess(a, b uint16) bool {
|
|
// Compare two RR types for the purpose of sorting the RRs in a Zone.
|
|
|
|
// If they are equal, we are done. All other code is simplified
|
|
// because we can assume a!=b.
|
|
if a == b {
|
|
return false
|
|
}
|
|
|
|
// List SOAs, then NSs, then all others.
|
|
// i.e. SOA is always less. NS is less than everything but SOA.
|
|
if a == dns.TypeSOA {
|
|
return true
|
|
}
|
|
if b == dns.TypeSOA {
|
|
return false
|
|
}
|
|
if a == dns.TypeNS {
|
|
return true
|
|
}
|
|
if b == dns.TypeNS {
|
|
return false
|
|
}
|
|
return a < b
|
|
}
|