coredns-rfc2136/store.go
Ryan Malloy 1cca9a5aa7 Phase 1.3: in-memory store + ServeDNS query dispatch
ServeDNS now answers authoritatively for the configured zone(s):
- Apex SOA → synthetic SOA (serial = store generation counter)
- Apex NS  → synthetic NS pointing at p.Nameserver
- In-store lookups for any qtype
- NODATA vs NXDOMAIN correctly distinguished (SOA in authority section)
- UPDATE opcode → REFUSED (Phase 1.4 implements properly)
- Queries outside our zones pass through to Next

Added:
- store.go: recordStore with sync.RWMutex + atomic generation counter.
  Operations: Add (de-dupes), RemoveRRset, RemoveRR, RemoveName, Lookup
  (returns a copy so callers can't corrupt internal state), NameExists.
  All keyed on canonical lowercase + trailing-dot names.
- plugin.go: ServeDNS dispatch, findZone (longest-suffix match),
  syntheticSOA, syntheticNS. New Nameserver field.
- setup.go: nameserver directive. Default Nameserver = first zone apex.
  Store initialised at parse time.
- store_test.go: 12 unit tests covering add/dedupe/remove/lookup/
  generation/case-insensitivity/copy-safety.
- plugin_test.go: 10 dispatch tests covering pass-through, apex
  synthetics, in-store lookups, NXDOMAIN/NODATA semantics, UPDATE
  refusal, findZone longest-suffix-wins and case behavior.
- setup_test.go: 3 new cases for the nameserver directive + store init.

Total: 38 tests passing.

Module: git.supported.systems/rsp2k/coredns-rfc2136
2026-05-21 10:37:48 -06:00

176 lines
4.5 KiB
Go

package rfc2136
import (
"strings"
"sync"
"sync/atomic"
"github.com/miekg/dns"
)
// recordStore is the in-memory store of dynamic records. It is keyed
// by canonical (lowercased, trailing-dot) owner name, then by RR type,
// to a slice of RRs forming the RRset.
//
// Concurrency: a single RWMutex covers all mutations. Read paths
// (queries) are far more frequent than writes (UPDATEs), so the
// RW split is the right choice. The lock is held only across the
// O(1) map operations; nothing slow happens under the lock.
//
// Serial: every successful mutation bumps the atomic generation
// counter. The synthetic SOA uses this counter so resolvers see a
// fresh serial after every UPDATE — even though we don't run AXFR
// against secondaries, future use-cases (DNS-NOTIFY, debugging,
// cache invalidation) benefit from a monotonic serial.
type recordStore struct {
mu sync.RWMutex
// rrs[name][rrtype] = RRset (slice). The map-of-maps shape means
// per-type lookups are O(1), and "name exists at all" is O(1) on
// the outer map.
rrs map[string]map[uint16][]dns.RR
gen atomic.Uint64
}
func newStore() *recordStore {
return &recordStore{rrs: make(map[string]map[uint16][]dns.RR)}
}
// generation returns the current monotonic counter. Used for SOA serial.
func (s *recordStore) generation() uint64 {
return s.gen.Load()
}
// canon normalises a name to the store's internal form: lowercase +
// trailing dot. miekg/dns sometimes returns names without the dot;
// always passing through this keeps everything consistent.
func canon(name string) string {
return strings.ToLower(dns.Fqdn(name))
}
// Add inserts a single RR into the store. Duplicate RRs (same owner,
// type, AND rdata) are silently de-duplicated — matches RFC 2136 §3.4.2.2
// behavior for `add` semantics.
func (s *recordStore) Add(rr dns.RR) {
s.mu.Lock()
defer s.mu.Unlock()
name := canon(rr.Header().Name)
rtype := rr.Header().Rrtype
byType, ok := s.rrs[name]
if !ok {
byType = make(map[uint16][]dns.RR)
s.rrs[name] = byType
}
// De-duplicate by string comparison of the rdata-bearing form.
rrStr := rr.String()
for _, existing := range byType[rtype] {
if existing.String() == rrStr {
return
}
}
byType[rtype] = append(byType[rtype], rr)
s.gen.Add(1)
}
// RemoveRRset deletes ALL records of the given (name, type) — i.e.
// "delete the RRset" semantics from RFC 2136 §3.4.2.3. If the name
// has no records left after this, the name entry is reaped so
// NameExists returns false.
func (s *recordStore) RemoveRRset(name string, rtype uint16) {
s.mu.Lock()
defer s.mu.Unlock()
name = canon(name)
byType, ok := s.rrs[name]
if !ok {
return
}
if _, hadType := byType[rtype]; !hadType {
return
}
delete(byType, rtype)
if len(byType) == 0 {
delete(s.rrs, name)
}
s.gen.Add(1)
}
// RemoveRR deletes one specific RR (matching owner, type, and rdata).
// This implements RFC 2136 §3.4.2.4 "delete an RR from an RRset".
func (s *recordStore) RemoveRR(rr dns.RR) {
s.mu.Lock()
defer s.mu.Unlock()
name := canon(rr.Header().Name)
rtype := rr.Header().Rrtype
byType, ok := s.rrs[name]
if !ok {
return
}
rrs := byType[rtype]
target := rr.String()
for i, existing := range rrs {
if existing.String() == target {
byType[rtype] = append(rrs[:i], rrs[i+1:]...)
if len(byType[rtype]) == 0 {
delete(byType, rtype)
}
if len(byType) == 0 {
delete(s.rrs, name)
}
s.gen.Add(1)
return
}
}
}
// RemoveName deletes all records for an owner name (§3.4.2.3 "delete
// all RRsets from a name").
func (s *recordStore) RemoveName(name string) {
s.mu.Lock()
defer s.mu.Unlock()
name = canon(name)
if _, ok := s.rrs[name]; !ok {
return
}
delete(s.rrs, name)
s.gen.Add(1)
}
// Lookup returns the RRset for (name, rtype). Returns nil for both
// "name doesn't exist" and "name exists with other types but not this
// one" — use NameExists to distinguish NODATA from NXDOMAIN.
//
// The returned slice is a copy so callers can freely mutate it without
// affecting store state.
func (s *recordStore) Lookup(name string, rtype uint16) []dns.RR {
s.mu.RLock()
defer s.mu.RUnlock()
byType, ok := s.rrs[canon(name)]
if !ok {
return nil
}
rrs := byType[rtype]
if len(rrs) == 0 {
return nil
}
out := make([]dns.RR, len(rrs))
copy(out, rrs)
return out
}
// NameExists reports whether ANY records exist for the given name.
// Used to distinguish NODATA (name exists, no records of asked type)
// from NXDOMAIN (name doesn't exist at all).
func (s *recordStore) NameExists(name string) bool {
s.mu.RLock()
defer s.mu.RUnlock()
_, ok := s.rrs[canon(name)]
return ok
}