mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-11 09:59:59 +08:00
de4455942b
* Replace RecordConfig.Name and .NameFQDN with getters and setters. * Replace RecordConfig.Target with getters and setters. * Eliminate the CombinedTarget concept. * Add RecordConfig.PopulateFromString to reduce code in all providers. * encode and decode name.com txt records (#315) * Replace fmt.Errorf with errors.Errorf
227 lines
7.7 KiB
Go
227 lines
7.7 KiB
Go
package diff
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"sort"
|
|
|
|
"github.com/StackExchange/dnscontrol/models"
|
|
"github.com/miekg/dns/dnsutil"
|
|
)
|
|
|
|
// Correlation stores a difference between two domains.
|
|
type Correlation struct {
|
|
d *differ
|
|
Existing *models.RecordConfig
|
|
Desired *models.RecordConfig
|
|
}
|
|
|
|
// Changeset stores many Correlation.
|
|
type Changeset []Correlation
|
|
|
|
// Differ is an interface for computing the difference between two zones.
|
|
type Differ interface {
|
|
// IncrementalDiff performs a diff on a record-by-record basis, and returns a sets for which records need to be created, deleted, or modified.
|
|
IncrementalDiff(existing []*models.RecordConfig) (unchanged, create, toDelete, modify Changeset)
|
|
// ChangedGroups performs a diff more appropriate for providers with a "RecordSet" model, where all records with the same name and type are grouped.
|
|
// Individual record changes are often not useful in such scenarios. Instead we return a map of record keys to a list of change descriptions within that group.
|
|
ChangedGroups(existing []*models.RecordConfig) map[models.RecordKey][]string
|
|
}
|
|
|
|
// New is a constructor for a Differ.
|
|
func New(dc *models.DomainConfig, extraValues ...func(*models.RecordConfig) map[string]string) Differ {
|
|
return &differ{
|
|
dc: dc,
|
|
extraValues: extraValues,
|
|
}
|
|
}
|
|
|
|
type differ struct {
|
|
dc *models.DomainConfig
|
|
extraValues []func(*models.RecordConfig) map[string]string
|
|
}
|
|
|
|
// get normalized content for record. target, ttl, mxprio, and specified metadata
|
|
func (d *differ) content(r *models.RecordConfig) string {
|
|
content := fmt.Sprintf("%v ttl=%d", r.GetTargetCombined(), r.TTL)
|
|
for _, f := range d.extraValues {
|
|
// sort the extra values map keys to perform a deterministic
|
|
// comparison since Golang maps iteration order is not guaranteed
|
|
valueMap := f(r)
|
|
keys := make([]string, 0)
|
|
for k := range valueMap {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
for _, k := range keys {
|
|
v := valueMap[k]
|
|
content += fmt.Sprintf(" %s=%s", k, v)
|
|
}
|
|
}
|
|
return content
|
|
}
|
|
|
|
func (d *differ) IncrementalDiff(existing []*models.RecordConfig) (unchanged, create, toDelete, modify Changeset) {
|
|
unchanged = Changeset{}
|
|
create = Changeset{}
|
|
toDelete = Changeset{}
|
|
modify = Changeset{}
|
|
desired := d.dc.Records
|
|
|
|
// sort existing and desired by name
|
|
type key struct {
|
|
name, rType string
|
|
}
|
|
existingByNameAndType := map[key][]*models.RecordConfig{}
|
|
desiredByNameAndType := map[key][]*models.RecordConfig{}
|
|
for _, e := range existing {
|
|
if d.matchIgnored(e.NameFQDN, d.dc.Name) {
|
|
log.Printf("Ignoring record %s %s due to IGNORE", e.NameFQDN, e.Type)
|
|
} else {
|
|
k := key{e.NameFQDN, e.Type}
|
|
existingByNameAndType[k] = append(existingByNameAndType[k], e)
|
|
}
|
|
}
|
|
for _, dr := range desired {
|
|
if d.matchIgnored(dr.NameFQDN, d.dc.Name) {
|
|
panic(fmt.Sprintf("Trying to update/add IGNOREd record: %s %s", dr.NameFQDN, dr.Type))
|
|
} else {
|
|
k := key{dr.NameFQDN, dr.Type}
|
|
desiredByNameAndType[k] = append(desiredByNameAndType[k], dr)
|
|
}
|
|
}
|
|
// if NO_PURGE is set, just remove anything that is only in existing.
|
|
if d.dc.KeepUnknown {
|
|
for k := range existingByNameAndType {
|
|
if _, ok := desiredByNameAndType[k]; !ok {
|
|
log.Printf("Ignoring record set %s %s due to NO_PURGE", k.rType, k.name)
|
|
delete(existingByNameAndType, k)
|
|
}
|
|
}
|
|
}
|
|
// Look through existing records. This will give us changes and deletions and some additions.
|
|
// Each iteration is only for a single type/name record set
|
|
for key, existingRecords := range existingByNameAndType {
|
|
desiredRecords := desiredByNameAndType[key]
|
|
// first look through records that are the same target on both sides. Those are either modifications or unchanged
|
|
for i := len(existingRecords) - 1; i >= 0; i-- {
|
|
ex := existingRecords[i]
|
|
for j, de := range desiredRecords {
|
|
if de.Target == ex.Target {
|
|
// they're either identical or should be a modification of each other (ttl or metadata changes)
|
|
if d.content(de) == d.content(ex) {
|
|
unchanged = append(unchanged, Correlation{d, ex, de})
|
|
} else {
|
|
modify = append(modify, Correlation{d, ex, de})
|
|
}
|
|
// remove from both slices by index
|
|
existingRecords = existingRecords[:i+copy(existingRecords[i:], existingRecords[i+1:])]
|
|
desiredRecords = desiredRecords[:j+copy(desiredRecords[j:], desiredRecords[j+1:])]
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
desiredLookup := map[string]*models.RecordConfig{}
|
|
existingLookup := map[string]*models.RecordConfig{}
|
|
// build index based on normalized content data
|
|
for _, ex := range existingRecords {
|
|
normalized := d.content(ex)
|
|
if existingLookup[normalized] != nil {
|
|
panic(fmt.Sprintf("DUPLICATE E_RECORD FOUND: %s %s", key, normalized))
|
|
}
|
|
existingLookup[normalized] = ex
|
|
}
|
|
for _, de := range desiredRecords {
|
|
normalized := d.content(de)
|
|
if desiredLookup[normalized] != nil {
|
|
panic(fmt.Sprintf("DUPLICATE D_RECORD FOUND: %s %s", key, normalized))
|
|
}
|
|
desiredLookup[normalized] = de
|
|
}
|
|
// if a record is in both, it is unchanged
|
|
for norm, ex := range existingLookup {
|
|
if de, ok := desiredLookup[norm]; ok {
|
|
unchanged = append(unchanged, Correlation{d, ex, de})
|
|
delete(existingLookup, norm)
|
|
delete(desiredLookup, norm)
|
|
}
|
|
}
|
|
// sort records by normalized text. Keeps behaviour deterministic
|
|
existingStrings, desiredStrings := sortedKeys(existingLookup), sortedKeys(desiredLookup)
|
|
// Modifications. Take 1 from each side.
|
|
for len(desiredStrings) > 0 && len(existingStrings) > 0 {
|
|
modify = append(modify, Correlation{d, existingLookup[existingStrings[0]], desiredLookup[desiredStrings[0]]})
|
|
existingStrings = existingStrings[1:]
|
|
desiredStrings = desiredStrings[1:]
|
|
}
|
|
// If desired still has things they are additions
|
|
for _, norm := range desiredStrings {
|
|
rec := desiredLookup[norm]
|
|
create = append(create, Correlation{d, nil, rec})
|
|
}
|
|
// if found , but not desired, delete it
|
|
for _, norm := range existingStrings {
|
|
rec := existingLookup[norm]
|
|
toDelete = append(toDelete, Correlation{d, rec, nil})
|
|
}
|
|
// remove this set from the desired list to indicate we have processed it.
|
|
delete(desiredByNameAndType, key)
|
|
}
|
|
|
|
// any name/type sets not already processed are pure additions
|
|
for name := range existingByNameAndType {
|
|
delete(desiredByNameAndType, name)
|
|
}
|
|
for _, desiredList := range desiredByNameAndType {
|
|
for _, rec := range desiredList {
|
|
create = append(create, Correlation{d, nil, rec})
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (d *differ) ChangedGroups(existing []*models.RecordConfig) map[models.RecordKey][]string {
|
|
changedKeys := map[models.RecordKey][]string{}
|
|
_, create, delete, modify := d.IncrementalDiff(existing)
|
|
for _, c := range create {
|
|
changedKeys[c.Desired.Key()] = append(changedKeys[c.Desired.Key()], c.String())
|
|
}
|
|
for _, d := range delete {
|
|
changedKeys[d.Existing.Key()] = append(changedKeys[d.Existing.Key()], d.String())
|
|
}
|
|
for _, m := range modify {
|
|
changedKeys[m.Desired.Key()] = append(changedKeys[m.Desired.Key()], m.String())
|
|
}
|
|
return changedKeys
|
|
}
|
|
|
|
func (c Correlation) String() string {
|
|
if c.Existing == nil {
|
|
return fmt.Sprintf("CREATE %s %s %s", c.Desired.Type, c.Desired.NameFQDN, c.d.content(c.Desired))
|
|
}
|
|
if c.Desired == nil {
|
|
return fmt.Sprintf("DELETE %s %s %s", c.Existing.Type, c.Existing.NameFQDN, c.d.content(c.Existing))
|
|
}
|
|
return fmt.Sprintf("MODIFY %s %s: (%s) -> (%s)", c.Existing.Type, c.Existing.NameFQDN, c.d.content(c.Existing), c.d.content(c.Desired))
|
|
}
|
|
|
|
func sortedKeys(m map[string]*models.RecordConfig) []string {
|
|
s := []string{}
|
|
for v := range m {
|
|
s = append(s, v)
|
|
}
|
|
sort.Strings(s)
|
|
return s
|
|
}
|
|
|
|
func (d *differ) matchIgnored(nameFQDN, domain string) bool {
|
|
// ignored labels are not fqdn
|
|
name := dnsutil.TrimDomainName(nameFQDN, domain)
|
|
for _, tst := range d.dc.IgnoredLabels {
|
|
if name == tst || nameFQDN == tst {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|