1
0
mirror of https://github.com/StackExchange/dnscontrol.git synced 2024-05-11 05:55:12 +00:00

NEW FEATURE: Order changes based on the record dependencies (#2419)

Co-authored-by: Tom Limoncelli <tlimoncelli@stackoverflow.com>
Co-authored-by: Tom Limoncelli <tal@whatexit.org>
This commit is contained in:
Vincent Hagen
2023-08-29 20:00:09 +02:00
committed by GitHub
parent e26afaf7e0
commit e32bdc053f
22 changed files with 1265 additions and 205 deletions

144
pkg/dnstree/dnstree.go Normal file
View File

@@ -0,0 +1,144 @@
package dnstree
import (
"strings"
)
// Create creates a tree like structure to add arbitrary data to DNS names.
// The DomainTree splits the domain name based on the dot (.), reverses the resulting list and add all strings the tree in order.
// It has support for wildcard domain names the tree nodes (`Set`), but not during retrieval (Get and Has).
// Get always returns the most specific node; it doesn't immediately return the node upon finding a wildcard node.
func Create[T any]() *DomainTree[T] {
return &DomainTree[T]{
IsLeaf: false,
IsWildcard: false,
Name: "",
Children: map[string]*domainNode[T]{},
}
}
type DomainTree[T any] domainNode[T]
type domainNode[T any] struct {
IsLeaf bool
IsWildcard bool
Name string
Children map[string]*domainNode[T]
data T
}
func createNode[T any](name string) *domainNode[T] {
return &domainNode[T]{
IsLeaf: false,
Name: name,
Children: map[string]*domainNode[T]{},
}
}
// Set adds given data to the given fqdn.
// The FQDN can contain a wildcard on the start.
// example fqdn: *.example.com
func (tree *DomainTree[T]) Set(fqdn string, data T) {
domainParts := splitFQDN(fqdn)
isWildcard := domainParts[0] == "*"
if isWildcard {
domainParts = domainParts[1:]
}
ptr := (*domainNode[T])(tree)
for iX := len(domainParts) - 1; iX > 0; iX -= 1 {
ptr = ptr.addIntermediate(domainParts[iX])
}
ptr.addLeaf(domainParts[0], isWildcard, data)
}
// Retrieves the attached data from a given FQDN.
// The tree will return the data entry for the most specific FQDN entry.
// If no entry is found Get will return the default value for the specific type.
//
// tree.Set("*.example.com", 1)
// tree.Set("a.example.com", 2)
// tree.Get("a.example.com") // 2
// tree.Get("a.a.example.com") // 1
// tree.Get("other.com") // 0
func (tree *DomainTree[T]) Get(fqdn string) T {
domainParts := splitFQDN(fqdn)
var mostSpecificNode *domainNode[T]
ptr := (*domainNode[T])(tree)
for iX := len(domainParts) - 1; iX >= 0; iX -= 1 {
node, ok := ptr.Children[domainParts[iX]]
if !ok {
if mostSpecificNode != nil {
return mostSpecificNode.data
}
return *new(T)
}
if node.IsWildcard {
mostSpecificNode = node
}
ptr = node
}
if ptr.IsLeaf || ptr.IsWildcard {
return ptr.data
}
if mostSpecificNode != nil {
return mostSpecificNode.data
}
return *new(T)
}
// Has returns if the tree contains data for given FQDN.
func (tree *DomainTree[T]) Has(fqdn string) bool {
domainParts := splitFQDN(fqdn)
var mostSpecificNode *domainNode[T]
ptr := (*domainNode[T])(tree)
for iX := len(domainParts) - 1; iX >= 0; iX -= 1 {
node, ok := ptr.Children[domainParts[iX]]
if !ok {
return mostSpecificNode != nil
}
if node.IsWildcard {
mostSpecificNode = node
}
ptr = node
}
return ptr.IsLeaf || ptr.IsWildcard || mostSpecificNode != nil
}
func splitFQDN(fqdn string) []string {
normalizedFQDN := strings.TrimSuffix(fqdn, ".")
return strings.Split(normalizedFQDN, ".")
}
func (tree *domainNode[T]) addIntermediate(name string) *domainNode[T] {
if _, ok := tree.Children[name]; !ok {
tree.Children[name] = createNode[T](name)
}
return tree.Children[name]
}
func (tree *domainNode[T]) addLeaf(name string, isWildcard bool, data T) *domainNode[T] {
node := tree.addIntermediate(name)
node.data = data
node.IsLeaf = true
node.IsWildcard = node.IsWildcard || isWildcard
return node
}

View File

@@ -0,0 +1,64 @@
package dnstree_test
import (
"testing"
"github.com/StackExchange/dnscontrol/v4/pkg/dnstree"
)
func Test_domaintree(t *testing.T) {
t.Run("Single FQDN",
executeTreeTest(
[]string{
"other.example.com",
},
[]string{"other.example.com"},
[]string{"com", "x.example.com", "x.www.example.com", "example.com"},
),
)
t.Run("Wildcard",
executeTreeTest(
[]string{
"*.example.com",
},
[]string{"example.com", "other.example.com"},
[]string{"com", "example.nl", "*.com"},
),
)
t.Run("Combined domains",
executeTreeTest(
[]string{
"*.other.example.com",
"specific.example.com",
"specific.example.nl",
},
[]string{"any.other.example.com", "specific.example.com", "specific.example.nl"},
[]string{"com", "nl", "", "example.nl", "other.nl"},
),
)
}
func executeTreeTest(inputs []string, founds []string, missings []string) func(*testing.T) {
return func(t *testing.T) {
t.Helper()
tree := dnstree.Create[interface{}]()
for _, input := range inputs {
tree.Set(input, struct{}{})
}
for _, found := range founds {
if tree.Has(found) == false {
t.Errorf("Expected %s to be found in tree, but is missing", found)
}
}
for _, missing := range missings {
if tree.Has(missing) == true {
t.Errorf("Expected %s to be missing in tree, but is found", missing)
}
}
}
}