plugin/dnssec: sign each RRset with the zone that owns its name, not the query zone (#8138)

Signed-off-by: Björn Kinscher <code@bjoern-kinscher.de>
Co-authored-by: Björn Kinscher <code@bjoern-kinscher.de>
This commit is contained in:
Isolus
2026-06-06 03:36:28 +02:00
committed by GitHub
parent 3718f0cc81
commit b49fe2d469
2 changed files with 111 additions and 3 deletions

View File

@@ -0,0 +1,95 @@
package dnssec
import (
"testing"
"time"
"github.com/coredns/coredns/plugin/test"
"github.com/coredns/coredns/request"
"github.com/miekg/dns"
)
// numSigs counts the RRSIG records in a message section.
func numSigs(rrs []dns.RR) int {
n := 0
for _, r := range rrs {
if r.Header().Rrtype == dns.TypeRRSIG {
n++
}
}
return n
}
// Checks that a CNAME whose target lies outside every zone we serve is signed for the
// CNAME itself, while the out-of-zone target is left untouched.
func TestSigningCnameOutOfZone(t *testing.T) {
d, rm1, rm2 := newDnssec(t, []string{"miek.nl."})
defer rm1()
defer rm2()
m := &dns.Msg{
Answer: []dns.RR{
test.CNAME("www.miek.nl.\t1800\tIN\tCNAME\ttarget.example.com."),
test.A("target.example.com.\t1800\tIN\tA\t127.0.0.1"),
},
}
state := request.Request{Req: m, Zone: "miek.nl."}
m = d.Sign(state, time.Now().UTC(), server)
if got := numSigs(m.Answer); got != 1 {
t.Fatalf("Answer should have exactly 1 RRSIG (the in-zone CNAME), got %d", got)
}
for _, r := range m.Answer {
sig, ok := r.(*dns.RRSIG)
if !ok {
continue
}
if sig.TypeCovered != dns.TypeCNAME {
t.Errorf("RRSIG should cover CNAME, got %s", dns.TypeToString[sig.TypeCovered])
}
if sig.SignerName != "miek.nl." {
t.Errorf("RRSIG signer should be miek.nl., got %s", sig.SignerName)
}
}
}
// checks that when a CNAME target lives in another zone we are also authoritative for,
// both RRsets are signed, each with the apex of the zone that actually contains its
// owner name as the signer.
func TestSigningCnameCrossZone(t *testing.T) {
d, rm1, rm2 := newDnssec(t, []string{"miek.nl.", "example.org."})
defer rm1()
defer rm2()
m := &dns.Msg{
Answer: []dns.RR{
test.CNAME("www.miek.nl.\t1800\tIN\tCNAME\tdb.example.org."),
test.A("db.example.org.\t1800\tIN\tA\t127.0.0.1"),
},
}
state := request.Request{Req: m, Zone: "miek.nl."}
m = d.Sign(state, time.Now().UTC(), server)
if got := numSigs(m.Answer); got != 2 {
t.Fatalf("Answer should have 2 RRSIGs (CNAME + cross-zone A), got %d", got)
}
for _, r := range m.Answer {
sig, ok := r.(*dns.RRSIG)
if !ok {
continue
}
switch sig.TypeCovered {
case dns.TypeCNAME:
if sig.SignerName != "miek.nl." {
t.Errorf("CNAME RRSIG signer should be miek.nl., got %s", sig.SignerName)
}
case dns.TypeA:
if sig.SignerName != "example.org." {
t.Errorf("cross-zone A RRSIG signer should be example.org., got %s", sig.SignerName)
}
default:
t.Errorf("unexpected RRSIG covering %s", dns.TypeToString[sig.TypeCovered])
}
}
}

View File

@@ -93,21 +93,34 @@ func (d Dnssec) Sign(state request.Request, now time.Time, server string) *dns.M
return req
}
zones := plugin.Zones(d.zones) // only sign if CNAME is not outside of our zones
for _, r := range rrSets(req.Answer) {
signer := zones.Matches(r[0].Header().Name)
if signer == "" {
continue
}
ttl := r[0].Header().Ttl
if sigs, err := d.sign(r, state.Zone, ttl, incep, expir, server); err == nil {
if sigs, err := d.sign(r, signer, ttl, incep, expir, server); err == nil {
req.Answer = append(req.Answer, sigs...)
}
}
for _, r := range rrSets(req.Ns) {
signer := zones.Matches(r[0].Header().Name)
if signer == "" {
continue
}
ttl := r[0].Header().Ttl
if sigs, err := d.sign(r, state.Zone, ttl, incep, expir, server); err == nil {
if sigs, err := d.sign(r, signer, ttl, incep, expir, server); err == nil {
req.Ns = append(req.Ns, sigs...)
}
}
for _, r := range rrSets(req.Extra) {
signer := zones.Matches(r[0].Header().Name)
if signer == "" {
continue
}
ttl := r[0].Header().Ttl
if sigs, err := d.sign(r, state.Zone, ttl, incep, expir, server); err == nil {
if sigs, err := d.sign(r, signer, ttl, incep, expir, server); err == nil {
req.Extra = append(req.Extra, sigs...)
}
}