mirror of
https://github.com/fosrl/olm.git
synced 2026-02-07 21:46:40 +00:00
497 lines
14 KiB
Go
497 lines
14 KiB
Go
package dns
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
// RecordType represents the type of DNS record
|
|
type RecordType uint16
|
|
|
|
const (
|
|
RecordTypeA RecordType = RecordType(dns.TypeA)
|
|
RecordTypeAAAA RecordType = RecordType(dns.TypeAAAA)
|
|
RecordTypePTR RecordType = RecordType(dns.TypePTR)
|
|
)
|
|
|
|
// DNSRecordStore manages local DNS records for A, AAAA, and PTR queries
|
|
type DNSRecordStore struct {
|
|
mu sync.RWMutex
|
|
aRecords map[string][]net.IP // domain -> list of IPv4 addresses
|
|
aaaaRecords map[string][]net.IP // domain -> list of IPv6 addresses
|
|
aWildcards map[string][]net.IP // wildcard pattern -> list of IPv4 addresses
|
|
aaaaWildcards map[string][]net.IP // wildcard pattern -> list of IPv6 addresses
|
|
ptrRecords map[string]string // IP address string -> domain name
|
|
}
|
|
|
|
// NewDNSRecordStore creates a new DNS record store
|
|
func NewDNSRecordStore() *DNSRecordStore {
|
|
return &DNSRecordStore{
|
|
aRecords: make(map[string][]net.IP),
|
|
aaaaRecords: make(map[string][]net.IP),
|
|
aWildcards: make(map[string][]net.IP),
|
|
aaaaWildcards: make(map[string][]net.IP),
|
|
ptrRecords: make(map[string]string),
|
|
}
|
|
}
|
|
|
|
// AddRecord adds a DNS record mapping (A or AAAA)
|
|
// domain should be in FQDN format (e.g., "example.com.")
|
|
// domain can contain wildcards: * (0+ chars) and ? (exactly 1 char)
|
|
// ip should be a valid IPv4 or IPv6 address
|
|
// Automatically adds a corresponding PTR record for non-wildcard domains
|
|
func (s *DNSRecordStore) AddRecord(domain string, ip net.IP) error {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
// Ensure domain ends with a dot (FQDN format)
|
|
if len(domain) == 0 || domain[len(domain)-1] != '.' {
|
|
domain = domain + "."
|
|
}
|
|
|
|
// Normalize domain to lowercase FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
// Check if domain contains wildcards
|
|
isWildcard := strings.ContainsAny(domain, "*?")
|
|
|
|
if ip.To4() != nil {
|
|
// IPv4 address
|
|
if isWildcard {
|
|
s.aWildcards[domain] = append(s.aWildcards[domain], ip)
|
|
} else {
|
|
s.aRecords[domain] = append(s.aRecords[domain], ip)
|
|
// Automatically add PTR record for non-wildcard domains
|
|
s.ptrRecords[ip.String()] = domain
|
|
}
|
|
} else if ip.To16() != nil {
|
|
// IPv6 address
|
|
if isWildcard {
|
|
s.aaaaWildcards[domain] = append(s.aaaaWildcards[domain], ip)
|
|
} else {
|
|
s.aaaaRecords[domain] = append(s.aaaaRecords[domain], ip)
|
|
// Automatically add PTR record for non-wildcard domains
|
|
s.ptrRecords[ip.String()] = domain
|
|
}
|
|
} else {
|
|
return &net.ParseError{Type: "IP address", Text: ip.String()}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// AddPTRRecord adds a PTR record mapping an IP address to a domain name
|
|
// ip should be a valid IPv4 or IPv6 address
|
|
// domain should be in FQDN format (e.g., "example.com.")
|
|
func (s *DNSRecordStore) AddPTRRecord(ip net.IP, domain string) error {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
// Ensure domain ends with a dot (FQDN format)
|
|
if len(domain) == 0 || domain[len(domain)-1] != '.' {
|
|
domain = domain + "."
|
|
}
|
|
|
|
// Normalize domain to lowercase FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
// Store PTR record using IP string as key
|
|
s.ptrRecords[ip.String()] = domain
|
|
|
|
return nil
|
|
}
|
|
|
|
// RemoveRecord removes a specific DNS record mapping
|
|
// If ip is nil, removes all records for the domain (including wildcards)
|
|
// Automatically removes corresponding PTR records for non-wildcard domains
|
|
func (s *DNSRecordStore) RemoveRecord(domain string, ip net.IP) {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
// Ensure domain ends with a dot (FQDN format)
|
|
if len(domain) == 0 || domain[len(domain)-1] != '.' {
|
|
domain = domain + "."
|
|
}
|
|
|
|
// Normalize domain to lowercase FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
// Check if domain contains wildcards
|
|
isWildcard := strings.ContainsAny(domain, "*?")
|
|
|
|
if ip == nil {
|
|
// Remove all records for this domain
|
|
if isWildcard {
|
|
delete(s.aWildcards, domain)
|
|
delete(s.aaaaWildcards, domain)
|
|
} else {
|
|
// For non-wildcard domains, remove PTR records for all IPs
|
|
if ips, ok := s.aRecords[domain]; ok {
|
|
for _, ipAddr := range ips {
|
|
// Only remove PTR if it points to this domain
|
|
if ptrDomain, exists := s.ptrRecords[ipAddr.String()]; exists && ptrDomain == domain {
|
|
delete(s.ptrRecords, ipAddr.String())
|
|
}
|
|
}
|
|
}
|
|
if ips, ok := s.aaaaRecords[domain]; ok {
|
|
for _, ipAddr := range ips {
|
|
// Only remove PTR if it points to this domain
|
|
if ptrDomain, exists := s.ptrRecords[ipAddr.String()]; exists && ptrDomain == domain {
|
|
delete(s.ptrRecords, ipAddr.String())
|
|
}
|
|
}
|
|
}
|
|
delete(s.aRecords, domain)
|
|
delete(s.aaaaRecords, domain)
|
|
}
|
|
return
|
|
}
|
|
|
|
if ip.To4() != nil {
|
|
// Remove specific IPv4 address
|
|
if isWildcard {
|
|
if ips, ok := s.aWildcards[domain]; ok {
|
|
s.aWildcards[domain] = removeIP(ips, ip)
|
|
if len(s.aWildcards[domain]) == 0 {
|
|
delete(s.aWildcards, domain)
|
|
}
|
|
}
|
|
} else {
|
|
if ips, ok := s.aRecords[domain]; ok {
|
|
s.aRecords[domain] = removeIP(ips, ip)
|
|
if len(s.aRecords[domain]) == 0 {
|
|
delete(s.aRecords, domain)
|
|
}
|
|
// Automatically remove PTR record if it points to this domain
|
|
if ptrDomain, exists := s.ptrRecords[ip.String()]; exists && ptrDomain == domain {
|
|
delete(s.ptrRecords, ip.String())
|
|
}
|
|
}
|
|
}
|
|
} else if ip.To16() != nil {
|
|
// Remove specific IPv6 address
|
|
if isWildcard {
|
|
if ips, ok := s.aaaaWildcards[domain]; ok {
|
|
s.aaaaWildcards[domain] = removeIP(ips, ip)
|
|
if len(s.aaaaWildcards[domain]) == 0 {
|
|
delete(s.aaaaWildcards, domain)
|
|
}
|
|
}
|
|
} else {
|
|
if ips, ok := s.aaaaRecords[domain]; ok {
|
|
s.aaaaRecords[domain] = removeIP(ips, ip)
|
|
if len(s.aaaaRecords[domain]) == 0 {
|
|
delete(s.aaaaRecords, domain)
|
|
}
|
|
// Automatically remove PTR record if it points to this domain
|
|
if ptrDomain, exists := s.ptrRecords[ip.String()]; exists && ptrDomain == domain {
|
|
delete(s.ptrRecords, ip.String())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// RemovePTRRecord removes a PTR record for an IP address
|
|
func (s *DNSRecordStore) RemovePTRRecord(ip net.IP) {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
delete(s.ptrRecords, ip.String())
|
|
}
|
|
|
|
// GetRecords returns all IP addresses for a domain and record type
|
|
// First checks for exact matches, then checks wildcard patterns
|
|
func (s *DNSRecordStore) GetRecords(domain string, recordType RecordType) []net.IP {
|
|
s.mu.RLock()
|
|
defer s.mu.RUnlock()
|
|
|
|
// Normalize domain to lowercase FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
var records []net.IP
|
|
switch recordType {
|
|
case RecordTypeA:
|
|
// Check exact match first
|
|
if ips, ok := s.aRecords[domain]; ok {
|
|
// Return a copy to prevent external modifications
|
|
records = make([]net.IP, len(ips))
|
|
copy(records, ips)
|
|
return records
|
|
}
|
|
// Check wildcard patterns
|
|
for pattern, ips := range s.aWildcards {
|
|
if matchWildcard(pattern, domain) {
|
|
records = append(records, ips...)
|
|
}
|
|
}
|
|
if len(records) > 0 {
|
|
// Return a copy
|
|
result := make([]net.IP, len(records))
|
|
copy(result, records)
|
|
return result
|
|
}
|
|
|
|
case RecordTypeAAAA:
|
|
// Check exact match first
|
|
if ips, ok := s.aaaaRecords[domain]; ok {
|
|
// Return a copy to prevent external modifications
|
|
records = make([]net.IP, len(ips))
|
|
copy(records, ips)
|
|
return records
|
|
}
|
|
// Check wildcard patterns
|
|
for pattern, ips := range s.aaaaWildcards {
|
|
if matchWildcard(pattern, domain) {
|
|
records = append(records, ips...)
|
|
}
|
|
}
|
|
if len(records) > 0 {
|
|
// Return a copy
|
|
result := make([]net.IP, len(records))
|
|
copy(result, records)
|
|
return result
|
|
}
|
|
}
|
|
|
|
return records
|
|
}
|
|
|
|
// GetPTRRecord returns the domain name for a PTR record query
|
|
// domain should be in reverse DNS format (e.g., "1.0.0.127.in-addr.arpa.")
|
|
func (s *DNSRecordStore) GetPTRRecord(domain string) (string, bool) {
|
|
s.mu.RLock()
|
|
defer s.mu.RUnlock()
|
|
|
|
// Convert reverse DNS format to IP address
|
|
ip := reverseDNSToIP(domain)
|
|
if ip == nil {
|
|
return "", false
|
|
}
|
|
|
|
// Look up the PTR record
|
|
if ptrDomain, ok := s.ptrRecords[ip.String()]; ok {
|
|
return ptrDomain, true
|
|
}
|
|
|
|
return "", false
|
|
}
|
|
|
|
// HasRecord checks if a domain has any records of the specified type
|
|
// Checks both exact matches and wildcard patterns
|
|
func (s *DNSRecordStore) HasRecord(domain string, recordType RecordType) bool {
|
|
s.mu.RLock()
|
|
defer s.mu.RUnlock()
|
|
|
|
// Normalize domain to lowercase FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
switch recordType {
|
|
case RecordTypeA:
|
|
// Check exact match
|
|
if _, ok := s.aRecords[domain]; ok {
|
|
return true
|
|
}
|
|
// Check wildcard patterns
|
|
for pattern := range s.aWildcards {
|
|
if matchWildcard(pattern, domain) {
|
|
return true
|
|
}
|
|
}
|
|
case RecordTypeAAAA:
|
|
// Check exact match
|
|
if _, ok := s.aaaaRecords[domain]; ok {
|
|
return true
|
|
}
|
|
// Check wildcard patterns
|
|
for pattern := range s.aaaaWildcards {
|
|
if matchWildcard(pattern, domain) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// HasPTRRecord checks if a PTR record exists for the given reverse DNS domain
|
|
func (s *DNSRecordStore) HasPTRRecord(domain string) bool {
|
|
s.mu.RLock()
|
|
defer s.mu.RUnlock()
|
|
|
|
// Convert reverse DNS format to IP address
|
|
ip := reverseDNSToIP(domain)
|
|
if ip == nil {
|
|
return false
|
|
}
|
|
|
|
_, ok := s.ptrRecords[ip.String()]
|
|
return ok
|
|
}
|
|
|
|
// Clear removes all records from the store
|
|
func (s *DNSRecordStore) Clear() {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
s.aRecords = make(map[string][]net.IP)
|
|
s.aaaaRecords = make(map[string][]net.IP)
|
|
s.aWildcards = make(map[string][]net.IP)
|
|
s.aaaaWildcards = make(map[string][]net.IP)
|
|
s.ptrRecords = make(map[string]string)
|
|
}
|
|
|
|
// removeIP is a helper function to remove a specific IP from a slice
|
|
func removeIP(ips []net.IP, toRemove net.IP) []net.IP {
|
|
result := make([]net.IP, 0, len(ips))
|
|
for _, ip := range ips {
|
|
if !ip.Equal(toRemove) {
|
|
result = append(result, ip)
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
// matchWildcard checks if a domain matches a wildcard pattern
|
|
// Pattern supports * (0+ chars) and ? (exactly 1 char)
|
|
// Special case: *.domain.com does not match domain.com itself
|
|
func matchWildcard(pattern, domain string) bool {
|
|
return matchWildcardInternal(pattern, domain, 0, 0)
|
|
}
|
|
|
|
// matchWildcardInternal performs the actual wildcard matching recursively
|
|
func matchWildcardInternal(pattern, domain string, pi, di int) bool {
|
|
plen := len(pattern)
|
|
dlen := len(domain)
|
|
|
|
// Base cases
|
|
if pi == plen && di == dlen {
|
|
return true
|
|
}
|
|
if pi == plen {
|
|
return false
|
|
}
|
|
|
|
// Handle wildcard characters
|
|
if pattern[pi] == '*' {
|
|
// Special case: if pattern starts with "*." and we're at the beginning,
|
|
// ensure we don't match the domain without a prefix
|
|
// e.g., *.autoco.internal should not match autoco.internal
|
|
if pi == 0 && pi+1 < plen && pattern[pi+1] == '.' {
|
|
// The * must match at least one character
|
|
if di == dlen {
|
|
return false
|
|
}
|
|
// Try matching 1 or more characters before the dot
|
|
for i := di + 1; i <= dlen; i++ {
|
|
if matchWildcardInternal(pattern, domain, pi+1, i) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Normal * matching (0 or more characters)
|
|
// Try matching 0 characters (skip the *)
|
|
if matchWildcardInternal(pattern, domain, pi+1, di) {
|
|
return true
|
|
}
|
|
// Try matching 1+ characters
|
|
if di < dlen {
|
|
return matchWildcardInternal(pattern, domain, pi, di+1)
|
|
}
|
|
return false
|
|
}
|
|
|
|
if pattern[pi] == '?' {
|
|
// ? matches exactly one character
|
|
if di >= dlen {
|
|
return false
|
|
}
|
|
return matchWildcardInternal(pattern, domain, pi+1, di+1)
|
|
}
|
|
|
|
// Regular character - must match exactly
|
|
if di >= dlen || pattern[pi] != domain[di] {
|
|
return false
|
|
}
|
|
|
|
return matchWildcardInternal(pattern, domain, pi+1, di+1)
|
|
}
|
|
|
|
// reverseDNSToIP converts a reverse DNS query name to an IP address
|
|
// Supports both IPv4 (in-addr.arpa) and IPv6 (ip6.arpa) formats
|
|
func reverseDNSToIP(domain string) net.IP {
|
|
// Normalize to lowercase and ensure FQDN
|
|
domain = strings.ToLower(dns.Fqdn(domain))
|
|
|
|
// Check for IPv4 reverse DNS (in-addr.arpa)
|
|
if strings.HasSuffix(domain, ".in-addr.arpa.") {
|
|
// Remove the suffix
|
|
ipPart := strings.TrimSuffix(domain, ".in-addr.arpa.")
|
|
// Split by dots and reverse
|
|
parts := strings.Split(ipPart, ".")
|
|
if len(parts) != 4 {
|
|
return nil
|
|
}
|
|
// Reverse the octets
|
|
reversed := make([]string, 4)
|
|
for i := 0; i < 4; i++ {
|
|
reversed[i] = parts[3-i]
|
|
}
|
|
// Parse as IP
|
|
return net.ParseIP(strings.Join(reversed, "."))
|
|
}
|
|
|
|
// Check for IPv6 reverse DNS (ip6.arpa)
|
|
if strings.HasSuffix(domain, ".ip6.arpa.") {
|
|
// Remove the suffix
|
|
ipPart := strings.TrimSuffix(domain, ".ip6.arpa.")
|
|
// Split by dots and reverse
|
|
parts := strings.Split(ipPart, ".")
|
|
if len(parts) != 32 {
|
|
return nil
|
|
}
|
|
// Reverse the nibbles and group into 16-bit hex values
|
|
reversed := make([]string, 32)
|
|
for i := 0; i < 32; i++ {
|
|
reversed[i] = parts[31-i]
|
|
}
|
|
// Join into IPv6 format (groups of 4 nibbles separated by colons)
|
|
var ipv6Parts []string
|
|
for i := 0; i < 32; i += 4 {
|
|
ipv6Parts = append(ipv6Parts, reversed[i]+reversed[i+1]+reversed[i+2]+reversed[i+3])
|
|
}
|
|
// Parse as IP
|
|
return net.ParseIP(strings.Join(ipv6Parts, ":"))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// IPToReverseDNS converts an IP address to reverse DNS format
|
|
// Returns the domain name for PTR queries (e.g., "1.0.0.127.in-addr.arpa.")
|
|
func IPToReverseDNS(ip net.IP) string {
|
|
if ip4 := ip.To4(); ip4 != nil {
|
|
// IPv4: reverse octets and append .in-addr.arpa.
|
|
return dns.Fqdn(fmt.Sprintf("%d.%d.%d.%d.in-addr.arpa",
|
|
ip4[3], ip4[2], ip4[1], ip4[0]))
|
|
}
|
|
|
|
if ip6 := ip.To16(); ip6 != nil && ip.To4() == nil {
|
|
// IPv6: expand to 32 nibbles, reverse, and append .ip6.arpa.
|
|
var nibbles []string
|
|
for i := 15; i >= 0; i-- {
|
|
nibbles = append(nibbles, fmt.Sprintf("%x", ip6[i]&0x0f))
|
|
nibbles = append(nibbles, fmt.Sprintf("%x", ip6[i]>>4))
|
|
}
|
|
return dns.Fqdn(strings.Join(nibbles, ".") + ".ip6.arpa")
|
|
}
|
|
|
|
return ""
|
|
} |