mirror of
https://github.com/coredns/coredns.git
synced 2025-10-28 00:34:24 -04:00
plugin/template: Persist truncated state to client if CNAME lookup response is truncated (#4713)
* persist truncated state to client if cname lookup response is truncated Signed-off-by: Chris O'Haver <cohaver@infoblox.com>
This commit is contained in:
96
plugin/template/cname_test.go
Normal file
96
plugin/template/cname_test.go
Normal file
@@ -0,0 +1,96 @@
|
||||
package template
|
||||
|
||||
import (
|
||||
"context"
|
||||
"regexp"
|
||||
"testing"
|
||||
gotmpl "text/template"
|
||||
|
||||
"github.com/coredns/coredns/plugin/pkg/dnstest"
|
||||
"github.com/coredns/coredns/plugin/test"
|
||||
"github.com/coredns/coredns/request"
|
||||
|
||||
"github.com/miekg/dns"
|
||||
)
|
||||
|
||||
func TestTruncatedCNAME(t *testing.T) {
|
||||
up := &Upstub{
|
||||
Qclass: dns.ClassINET,
|
||||
Truncated: true,
|
||||
Case: test.Case{
|
||||
Qname: "cname.test.",
|
||||
Qtype: dns.TypeA,
|
||||
Rcode: dns.RcodeSuccess,
|
||||
Answer: []dns.RR{
|
||||
test.CNAME("cname.test. 600 IN CNAME test.up"),
|
||||
test.A("test.up. 600 IN A 1.2.3.4"),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
handler := Handler{
|
||||
Zones: []string{"."},
|
||||
Templates: []template{{
|
||||
regex: []*regexp.Regexp{regexp.MustCompile("^cname\\.test\\.$")},
|
||||
answer: []*gotmpl.Template{gotmpl.Must(gotmpl.New("answer").Parse(up.Answer[0].String()))},
|
||||
qclass: dns.ClassINET,
|
||||
qtype: dns.TypeA,
|
||||
zones: []string{"test."},
|
||||
upstream: up,
|
||||
}},
|
||||
}
|
||||
|
||||
r := &dns.Msg{Question: []dns.Question{{Name: up.Qname, Qclass: up.Qclass, Qtype: up.Qtype}}}
|
||||
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
||||
|
||||
_, err := handler.ServeDNS(context.TODO(), w, r)
|
||||
|
||||
if err != nil {
|
||||
t.Fatalf("Unexpecetd error %q", err)
|
||||
}
|
||||
if w.Msg == nil {
|
||||
t.Fatalf("Unexpecetd empty response.")
|
||||
}
|
||||
if !w.Msg.Truncated {
|
||||
t.Error("Expected reply to be marked truncated.")
|
||||
}
|
||||
err = test.SortAndCheck(w.Msg, up.Case)
|
||||
if err != nil {
|
||||
t.Error(err)
|
||||
}
|
||||
}
|
||||
|
||||
// Upstub implements an Upstreamer that returns a set response for test purposes
|
||||
type Upstub struct {
|
||||
test.Case
|
||||
Truncated bool
|
||||
Qclass uint16
|
||||
}
|
||||
|
||||
// Lookup returns a set response
|
||||
func (t *Upstub) Lookup(ctx context.Context, state request.Request, name string, typ uint16) (*dns.Msg, error) {
|
||||
var answer []dns.RR
|
||||
// if query type is not CNAME, remove any CNAME with same name as qname from the answer
|
||||
if t.Qtype != dns.TypeCNAME {
|
||||
for _, a := range t.Answer {
|
||||
if c, ok := a.(*dns.CNAME); ok && c.Header().Name == t.Qname {
|
||||
continue
|
||||
}
|
||||
answer = append(answer, a)
|
||||
}
|
||||
} else {
|
||||
answer = t.Answer
|
||||
}
|
||||
|
||||
return &dns.Msg{
|
||||
MsgHdr: dns.MsgHdr{
|
||||
Response: true,
|
||||
Truncated: t.Truncated,
|
||||
Rcode: t.Rcode,
|
||||
},
|
||||
Question: []dns.Question{{Name: t.Qname, Qtype: t.Qtype, Qclass: t.Qclass}},
|
||||
Answer: answer,
|
||||
Extra: t.Extra,
|
||||
Ns: t.Ns,
|
||||
}, nil
|
||||
}
|
||||
Reference in New Issue
Block a user