Per RFC 1996, a master that mutates a zone SHOULD notify its
secondaries so they can immediately AXFR rather than wait for their
next SOA-refresh poll. Without this, propagation lag from UPDATE to
public DNS is bounded by the secondary's refresh interval (300s for
us) — which is borderline for ACME validation timing.
New Corefile directive:
notify <host[:port]> [<host[:port]>...]
Targets accept bare hostnames (port 53 default), host:port, or
[ipv6]:port. The same list applies to every zone in the rfc2136
block.
Implementation: fire-and-forget UDP per target, each in its own
goroutine, capped by a 2s timeout. The UPDATE response to the client
is never held pending NOTIFY acks (RFC 1996 §4 explicitly decouples
them). Failures log at DEBUG only — a briefly-unreachable secondary
is normal and would otherwise spam logs.
Retires the external scripts/notify-secondaries.py workflow for any
deployment that wires the directive: secondaries now hear about
changes within seconds of the UPDATE landing, no cron or manual
invocation needed.
New tests:
- TestSendNotify_DeliversToTarget — packet arrives, opcode + zone correct
- TestSendNotify_NoTargets_NoCrash — empty list short-circuits
- TestSendNotify_BadTarget_LogsButDoesNotBlock — fire-and-forget timing
- TestNotifyOne_AppendsDefaultPort — host vs host:port normalization
122 lines
4.6 KiB
Go
122 lines
4.6 KiB
Go
// Package rfc2136 is a CoreDNS plugin that accepts dynamic DNS updates
|
|
// per RFC 2136 (UPDATE opcode), authenticated via TSIG, and applies
|
|
// them to on-disk zone files. This is the right shape for stacks where
|
|
// the operator wants to keep zones in flat files (perhaps under git,
|
|
// with HE pulling AXFR), but also wants programmatic updates from
|
|
// clients like Caddy's caddy-dns/rfc2136 module.
|
|
//
|
|
// The plugin does NOT serve any queries — that's the job of the
|
|
// `auto`/`file` plugin running alongside it. This plugin's only
|
|
// responsibility is the UPDATE opcode path: verify TSIG, dissect the
|
|
// UPDATE, write the zone file, bump the SOA serial, optionally
|
|
// auto-commit to git. CoreDNS's auto plugin notices the mtime change
|
|
// and re-serves the zone within its reload interval.
|
|
//
|
|
// See the plan at
|
|
// ~/.claude/plans/dood-does-coredns-offer-enumerated-piglet.md
|
|
// for the architectural rationale.
|
|
package rfc2136
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/coredns/coredns/plugin"
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
// DefaultTTL is applied to dynamically-added records whose UPDATE
|
|
// messages carry TTL=0. 60s matches the short-lived nature of ACME
|
|
// challenge records and keeps stale answers from lingering in
|
|
// resolver caches.
|
|
const DefaultTTL uint32 = 60
|
|
|
|
// RFC2136 is the plugin handler. One instance per Corefile server block.
|
|
type RFC2136 struct {
|
|
// Next is the downstream plugin in the chain — queries always
|
|
// pass through; only UPDATE opcode is intercepted.
|
|
Next plugin.Handler
|
|
|
|
// Zones is the set of canonical (dot-terminated, lowercase) zone
|
|
// names this instance accepts UPDATEs for. UPDATEs for any other
|
|
// zone are rejected with NOTAUTH.
|
|
Zones []string
|
|
|
|
// TSIGKeys is keyed by canonical key name (lowercased, trailing
|
|
// dot). Empty means TSIG is disabled — UPDATEs are refused
|
|
// unconditionally as a safety default.
|
|
TSIGKeys map[string]tsigKey
|
|
|
|
// TTL is applied to dynamically-injected records that don't carry
|
|
// an explicit TTL in the UPDATE message.
|
|
TTL uint32
|
|
|
|
// ZonesDir is the directory where <zone>.zone files live (matching
|
|
// the mount path inside the CoreDNS container). The plugin reads
|
|
// and writes files at <ZonesDir>/<zone>.zone.
|
|
ZonesDir string
|
|
|
|
// AutoCommit governs whether the plugin auto-commits zone-file
|
|
// changes to git after every successful UPDATE.
|
|
AutoCommit bool
|
|
|
|
// zones holds per-zone file handlers, keyed by canonical zone name.
|
|
// Populated in setup; mutexes live inside each zoneFile.
|
|
zones map[string]*zoneFile
|
|
|
|
// rateLimit caps UPDATE traffic per TSIG key (Hamilton M8). nil
|
|
// disables rate limiting (test mode, or insecure deployments).
|
|
// Populated in setup() once TSIG keys are known.
|
|
rateLimit *rateLimiter
|
|
|
|
// NotifyTargets is the list of secondary servers (IP[:port]) to
|
|
// send DNS NOTIFY messages to after every successful UPDATE.
|
|
// Default port 53. Empty list = no NOTIFY (secondaries rely on
|
|
// their own SOA-refresh polling). Configured via the `notify`
|
|
// directive in the Corefile.
|
|
NotifyTargets []string
|
|
}
|
|
|
|
// Name implements plugin.Handler.
|
|
func (p *RFC2136) Name() string { return "rfc2136" }
|
|
|
|
// ServeDNS implements plugin.Handler.
|
|
//
|
|
// Dispatch:
|
|
//
|
|
// UPDATE opcode → verify TSIG, then apply via the UPDATE handler.
|
|
// Anything else → pass through to Next (the auto plugin handles
|
|
// queries against the zone files we maintain).
|
|
func (p *RFC2136) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
|
|
if r.Opcode == dns.OpcodeUpdate {
|
|
if err := p.checkTSIG(w, r); err != nil {
|
|
log.Warningf("UPDATE rejected: %v", err)
|
|
resp := new(dns.Msg)
|
|
resp.SetRcode(r, dns.RcodeRefused)
|
|
// Do NOT sign rejection responses. Signing a refusal with
|
|
// the named key would attest the key exists on this server
|
|
// (Hamilton M9). Unsigned Refused is the right shape — the
|
|
// client sees "no TSIG on reply" and treats that as auth
|
|
// failure, which is correct because auth DID fail.
|
|
_ = w.WriteMsg(resp)
|
|
return dns.RcodeRefused, nil
|
|
}
|
|
// Hamilton M8: per-key rate limit. TSIG just authenticates the
|
|
// sender — it doesn't prove the sender's behavior is sane. A
|
|
// compromised key or a runaway client must not be able to
|
|
// exhaust disk/git/serial-counter resources.
|
|
if p.rateLimit != nil {
|
|
if tsig := r.IsTsig(); tsig != nil && !p.rateLimit.allow(strings.ToLower(tsig.Hdr.Name), time.Now()) {
|
|
log.Warningf("UPDATE rate-limited for key %q", tsig.Hdr.Name)
|
|
resp := new(dns.Msg)
|
|
resp.SetRcode(r, dns.RcodeRefused)
|
|
_ = w.WriteMsg(resp)
|
|
return dns.RcodeRefused, nil
|
|
}
|
|
}
|
|
return p.handleUpdate(w, r, true)
|
|
}
|
|
return plugin.NextOrFailure(p.Name(), p.Next, ctx, w, r)
|
|
}
|