mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-17 04:51:11 +08:00
200 lines
4.6 KiB
Go
200 lines
4.6 KiB
Go
package prettyzone
|
|
|
|
// Generate zonefiles.
|
|
// This generates a zonefile that prioritizes beauty over efficiency.
|
|
|
|
import (
|
|
"bytes"
|
|
"log"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/StackExchange/dnscontrol/v3/models"
|
|
)
|
|
|
|
// ZoneGenData is the configuration description for the zone generator.
|
|
type ZoneGenData struct {
|
|
Origin string
|
|
DefaultTTL uint32
|
|
Records models.Records
|
|
Comments []string
|
|
}
|
|
|
|
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]
|
|
|
|
// Sort by name.
|
|
|
|
// If we are at the apex, use "@" in the sorting.
|
|
compA, compB := a.NameFQDN, b.NameFQDN
|
|
if compA != compB {
|
|
if a.Name == "@" {
|
|
compA = "@"
|
|
}
|
|
if b.Name == "@" {
|
|
compB = "@"
|
|
}
|
|
return LabelLess(compA, compB)
|
|
}
|
|
|
|
// sub-sort by type
|
|
if a.Type != b.Type {
|
|
return zoneRrtypeLess(a.Type, b.Type)
|
|
}
|
|
|
|
// sub-sort within type:
|
|
switch a.Type { // #rtype_variations
|
|
case "A":
|
|
ta2, tb2 := a.GetTargetIP(), b.GetTargetIP()
|
|
ipa, ipb := ta2.To4(), tb2.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 "AAAA":
|
|
ta2, tb2 := a.GetTargetIP(), b.GetTargetIP()
|
|
ipa, ipb := ta2.To16(), tb2.To16()
|
|
if ipa == nil || ipb == nil {
|
|
log.Fatalf("should not happen: IPs are not 16 bytes: %#v %#v", ta2, tb2)
|
|
}
|
|
return bytes.Compare(ipa, ipb) == -1
|
|
case "MX":
|
|
// sort by priority. If they are equal, sort by Mx.
|
|
if a.MxPreference == b.MxPreference {
|
|
return a.GetTargetField() < b.GetTargetField()
|
|
}
|
|
return a.MxPreference < b.MxPreference
|
|
case "SRV":
|
|
//ta2, tb2 := a.(*dns.SRV), b.(*dns.SRV)
|
|
pa, pb := a.SrvPort, b.SrvPort
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
pa, pb = a.SrvPriority, b.SrvPriority
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
pa, pb = a.SrvWeight, b.SrvWeight
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
case "PTR":
|
|
//ta2, tb2 := a.(*dns.PTR), b.(*dns.PTR)
|
|
pa, pb := a.GetTargetField(), b.GetTargetField()
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
case "CAA":
|
|
//ta2, tb2 := a.(*dns.CAA), b.(*dns.CAA)
|
|
// sort by tag
|
|
pa, pb := a.CaaTag, b.CaaTag
|
|
if pa != pb {
|
|
return pa < pb
|
|
}
|
|
// then flag
|
|
fa, fb := a.CaaFlag, b.CaaFlag
|
|
if fa != fb {
|
|
// flag set goes before ones without flag set
|
|
return fa > fb
|
|
}
|
|
default:
|
|
// pass through. String comparison is sufficient.
|
|
}
|
|
return a.String() < b.String()
|
|
}
|
|
|
|
// LabelLess provides a "Less" function for two labels as needed for sorting. It
|
|
// sorts labels in prefix order, to make output pretty.
|
|
func LabelLess(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
|
|
}
|
|
// 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 string) bool {
|
|
// Compare two RR types for the purpose of sorting the RRs in a Zone.
|
|
|
|
if a == b {
|
|
return false
|
|
}
|
|
|
|
// List SOAs, NSs, etc. then all others alphabetically.
|
|
|
|
for _, t := range []string{"SOA", "NS", "CNAME",
|
|
"A", "AAAA", "MX", "SRV", "TXT",
|
|
} {
|
|
if a == t {
|
|
return true
|
|
}
|
|
if b == t {
|
|
return false
|
|
}
|
|
}
|
|
return a < b
|
|
}
|