Cache elements of State

Cache the size and the do bit whenever someone asked for it. We can

probably add more:



PASS

BenchmarkStateDo-4  	100000000	        11.9 ns/op

BenchmarkStateSize-4	 5000000	       265 ns/op

ok  	github.com/miekg/coredns/middleware	2.828s



PASS

BenchmarkStateDo-4  	1000000000	         2.86 ns/op

BenchmarkStateSize-4	500000000	         3.10 ns/op

ok  	github.com/miekg/coredns/middleware	5.032s



This PR also includes some testing cleanups as well.
This commit is contained in:
Miek Gieben
2016-04-04 08:19:06 +01:00
parent b3b6fe48f6
commit 84dfdab584
19 changed files with 170 additions and 73 deletions

View File

@@ -4,6 +4,7 @@ import (
"testing" "testing"
"github.com/miekg/coredns/middleware" "github.com/miekg/coredns/middleware"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns" "github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
@@ -57,7 +58,7 @@ func TestChaos(t *testing.T) {
req.Question[0].Qclass = dns.ClassCHAOS req.Question[0].Qclass = dns.ClassCHAOS
em.Next = test.next em.Next = test.next
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
code, err := em.ServeDNS(ctx, rec, req) code, err := em.ServeDNS(ctx, rec, req)
if err != test.expectedErr { if err != test.expectedErr {

View File

@@ -9,6 +9,7 @@ import (
"testing" "testing"
"github.com/miekg/coredns/middleware" "github.com/miekg/coredns/middleware"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns" "github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
@@ -46,7 +47,7 @@ func TestErrors(t *testing.T) {
for i, test := range tests { for i, test := range tests {
em.Next = test.next em.Next = test.next
buf.Reset() buf.Reset()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
code, err := em.ServeDNS(ctx, rec, req) code, err := em.ServeDNS(ctx, rec, req)
if err != test.expectedErr { if err != test.expectedErr {
@@ -77,7 +78,7 @@ func TestVisibleErrorWithPanic(t *testing.T) {
req := new(dns.Msg) req := new(dns.Msg)
req.SetQuestion("example.org.", dns.TypeA) req.SetQuestion("example.org.", dns.TypeA)
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
code, err := eh.ServeDNS(ctx, rec, req) code, err := eh.ServeDNS(ctx, rec, req)
if code != 0 { if code != 0 {

View File

@@ -23,7 +23,7 @@ func TestCnameLookup(t *testing.T) {
for _, tc := range dnsTestCasesCname { for _, tc := range dnsTestCasesCname {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := etc.ServeDNS(ctx, rec, m) _, err := etc.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -25,7 +25,7 @@ func TestGroupLookup(t *testing.T) {
for _, tc := range dnsTestCasesGroup { for _, tc := range dnsTestCasesGroup {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := etc.ServeDNS(ctx, rec, m) _, err := etc.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -28,7 +28,7 @@ func TestMultiLookup(t *testing.T) {
for _, tc := range dnsTestCasesMulti { for _, tc := range dnsTestCasesMulti {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := etcMulti.ServeDNS(ctx, rec, m) _, err := etcMulti.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -27,7 +27,7 @@ func TestOtherLookup(t *testing.T) {
for _, tc := range dnsTestCasesOther { for _, tc := range dnsTestCasesOther {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := etc.ServeDNS(ctx, rec, m) _, err := etc.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -16,9 +16,9 @@ import (
"github.com/miekg/coredns/middleware/etcd/singleflight" "github.com/miekg/coredns/middleware/etcd/singleflight"
"github.com/miekg/coredns/middleware/proxy" "github.com/miekg/coredns/middleware/proxy"
coretest "github.com/miekg/coredns/middleware/testing" coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns"
etcdc "github.com/coreos/etcd/client" etcdc "github.com/coreos/etcd/client"
"github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
) )
@@ -67,7 +67,7 @@ func TestLookup(t *testing.T) {
for _, tc := range dnsTestCases { for _, tc := range dnsTestCases {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := etc.ServeDNS(ctx, rec, m) _, err := etc.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -109,7 +109,7 @@ func TestLookupDNSSEC(t *testing.T) {
for _, tc := range dnssecTestCases { for _, tc := range dnssecTestCases {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := fm.ServeDNS(ctx, rec, m) _, err := fm.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)
@@ -147,7 +147,7 @@ func BenchmarkLookupDNSSEC(b *testing.B) {
fm := File{Next: coretest.ErrorHandler(), Zones: Zones{Z: map[string]*Zone{testzone: zone}, Names: []string{testzone}}} fm := File{Next: coretest.ErrorHandler(), Zones: Zones{Z: map[string]*Zone{testzone: zone}, Names: []string{testzone}}}
ctx := context.TODO() ctx := context.TODO()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
tc := coretest.Case{ tc := coretest.Case{
Qname: "b.miek.nl.", Qtype: dns.TypeA, Do: true, Qname: "b.miek.nl.", Qtype: dns.TypeA, Do: true,

View File

@@ -42,7 +42,7 @@ func TestLookupENT(t *testing.T) {
for _, tc := range entTestCases { for _, tc := range entTestCases {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := fm.ServeDNS(ctx, rec, m) _, err := fm.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -77,7 +77,7 @@ func TestLookup(t *testing.T) {
for _, tc := range dnsTestCases { for _, tc := range dnsTestCases {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := fm.ServeDNS(ctx, rec, m) _, err := fm.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)
@@ -112,7 +112,7 @@ func TestLookupNil(t *testing.T) {
ctx := context.TODO() ctx := context.TODO()
m := dnsTestCases[0].Msg() m := dnsTestCases[0].Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
fm.ServeDNS(ctx, rec, m) fm.ServeDNS(ctx, rec, m)
} }
@@ -124,7 +124,7 @@ func BenchmarkLookup(b *testing.B) {
fm := File{Next: coretest.ErrorHandler(), Zones: Zones{Z: map[string]*Zone{testzone: zone}, Names: []string{testzone}}} fm := File{Next: coretest.ErrorHandler(), Zones: Zones{Z: map[string]*Zone{testzone: zone}, Names: []string{testzone}}}
ctx := context.TODO() ctx := context.TODO()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
tc := coretest.Case{ tc := coretest.Case{
Qname: "www.miek.nl.", Qtype: dns.TypeA, Qname: "www.miek.nl.", Qtype: dns.TypeA,

View File

@@ -56,7 +56,7 @@ func TestLookupWildcard(t *testing.T) {
for _, tc := range wildcardTestCases { for _, tc := range wildcardTestCases {
m := tc.Msg() m := tc.Msg()
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
_, err := fm.ServeDNS(ctx, rec, m) _, err := fm.ServeDNS(ctx, rec, m)
if err != nil { if err != nil {
t.Errorf("expected no error, got %v\n", err) t.Errorf("expected no error, got %v\n", err)

View File

@@ -4,6 +4,7 @@ import (
"testing" "testing"
"github.com/miekg/coredns/middleware" "github.com/miekg/coredns/middleware"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns" "github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
@@ -55,7 +56,7 @@ func TestLoadBalance(t *testing.T) {
}, },
} }
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
for i, test := range tests { for i, test := range tests {
req := new(dns.Msg) req := new(dns.Msg)

View File

@@ -7,6 +7,8 @@ import (
"testing" "testing"
"github.com/miekg/coredns/middleware" "github.com/miekg/coredns/middleware"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns" "github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
) )
@@ -35,7 +37,7 @@ func TestLoggedStatus(t *testing.T) {
r := new(dns.Msg) r := new(dns.Msg)
r.SetQuestion("example.org.", dns.TypeA) r.SetQuestion("example.org.", dns.TypeA)
rec := middleware.NewResponseRecorder(&middleware.TestResponseWriter{}) rec := middleware.NewResponseRecorder(&coretest.ResponseWriter{})
rcode, _ := logger.ServeDNS(ctx, rec, r) rcode, _ := logger.ServeDNS(ctx, rec, r)
if rcode != 0 { if rcode != 0 {

View File

@@ -7,6 +7,8 @@ import (
"testing" "testing"
"github.com/miekg/coredns/middleware" "github.com/miekg/coredns/middleware"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns" "github.com/miekg/dns"
) )
@@ -30,5 +32,5 @@ func TestLookupProxy(t *testing.T) {
} }
func fakeState() middleware.State { func fakeState() middleware.State {
return middleware.State{W: &middleware.TestResponseWriter{}, Req: new(dns.Msg)} return middleware.State{W: &coretest.ResponseWriter{}, Req: new(dns.Msg)}
} }

View File

@@ -1,7 +1,6 @@
package middleware package middleware
import ( import (
"net"
"time" "time"
"github.com/miekg/dns" "github.com/miekg/dns"
@@ -79,24 +78,3 @@ func (r *ResponseRecorder) Hijack() {
r.ResponseWriter.Hijack() r.ResponseWriter.Hijack()
return return
} }
type TestResponseWriter struct{}
func (t *TestResponseWriter) LocalAddr() net.Addr {
ip := net.ParseIP("127.0.0.1")
port := 53
return &net.UDPAddr{IP: ip, Port: port, Zone: ""}
}
func (t *TestResponseWriter) RemoteAddr() net.Addr {
ip := net.ParseIP("10.240.0.1")
port := 40212
return &net.UDPAddr{IP: ip, Port: port, Zone: ""}
}
func (t *TestResponseWriter) WriteMsg(m *dns.Msg) error { return nil }
func (t *TestResponseWriter) Write(buf []byte) (int, error) { return len(buf), nil }
func (t *TestResponseWriter) Close() error { return nil }
func (t *TestResponseWriter) TsigStatus() error { return nil }
func (t *TestResponseWriter) TsigTimersOnly(bool) { return }
func (t *TestResponseWriter) Hijack() { return }

View File

@@ -2,36 +2,38 @@ package middleware
import ( import (
"net" "net"
"net/http"
"strings" "strings"
"time" "time"
"github.com/miekg/dns" "github.com/miekg/dns"
) )
// This file contains the state nd functions available for use in the templates. // This file contains the state functions available for use in the middlewares.
// State contains some connection state and is useful in middleware. // State contains some connection state and is useful in middleware.
type State struct { type State struct {
Root http.FileSystem // TODO(miek): needed? Req *dns.Msg
Req *dns.Msg W dns.ResponseWriter
W dns.ResponseWriter
// Cache size after first call to Size or Do
size int
do int // 0: not, 1: true: 2: false
} }
// Now returns the current timestamp in the specified format. // Now returns the current timestamp in the specified format.
func (s State) Now(format string) string { return time.Now().Format(format) } func (s *State) Now(format string) string { return time.Now().Format(format) }
// NowDate returns the current date/time that can be used in other time functions. // NowDate returns the current date/time that can be used in other time functions.
func (s State) NowDate() time.Time { return time.Now() } func (s *State) NowDate() time.Time { return time.Now() }
// Header gets the heaser of the request in State. // Header gets the heaser of the request in State.
func (s State) Header() *dns.RR_Header { func (s *State) Header() *dns.RR_Header {
// TODO(miek) // TODO(miek)
return nil return nil
} }
// IP gets the (remote) IP address of the client making the request. // IP gets the (remote) IP address of the client making the request.
func (s State) IP() string { func (s *State) IP() string {
ip, _, err := net.SplitHostPort(s.W.RemoteAddr().String()) ip, _, err := net.SplitHostPort(s.W.RemoteAddr().String())
if err != nil { if err != nil {
return s.W.RemoteAddr().String() return s.W.RemoteAddr().String()
@@ -40,7 +42,7 @@ func (s State) IP() string {
} }
// Post gets the (remote) Port of the client making the request. // Post gets the (remote) Port of the client making the request.
func (s State) Port() (string, error) { func (s *State) Port() (string, error) {
_, port, err := net.SplitHostPort(s.W.RemoteAddr().String()) _, port, err := net.SplitHostPort(s.W.RemoteAddr().String())
if err != nil { if err != nil {
return "0", err return "0", err
@@ -50,7 +52,7 @@ func (s State) Port() (string, error) {
// Proto gets the protocol used as the transport. This // Proto gets the protocol used as the transport. This
// will be udp or tcp. // will be udp or tcp.
func (s State) Proto() string { func (s *State) Proto() string {
if _, ok := s.W.RemoteAddr().(*net.UDPAddr); ok { if _, ok := s.W.RemoteAddr().(*net.UDPAddr); ok {
return "udp" return "udp"
} }
@@ -62,7 +64,7 @@ func (s State) Proto() string {
// Family returns the family of the transport. // Family returns the family of the transport.
// 1 for IPv4 and 2 for IPv6. // 1 for IPv4 and 2 for IPv6.
func (s State) Family() int { func (s *State) Family() int {
var a net.IP var a net.IP
ip := s.W.RemoteAddr() ip := s.W.RemoteAddr()
if i, ok := ip.(*net.UDPAddr); ok { if i, ok := ip.(*net.UDPAddr); ok {
@@ -79,33 +81,56 @@ func (s State) Family() int {
} }
// Do returns if the request has the DO (DNSSEC OK) bit set. // Do returns if the request has the DO (DNSSEC OK) bit set.
func (s State) Do() bool { func (s *State) Do() bool {
if s.do != 0 {
return s.do == doTrue
}
if o := s.Req.IsEdns0(); o != nil { if o := s.Req.IsEdns0(); o != nil {
if o.Do() {
s.do = doTrue
} else {
s.do = doFalse
}
return o.Do() return o.Do()
} }
s.do = doFalse
return false return false
} }
// UDPSize returns if UDP buffer size advertised in the requests OPT record. // UDPSize returns if UDP buffer size advertised in the requests OPT record.
// Or when the request was over TCP, we return the maximum allowed size of 64K. // Or when the request was over TCP, we return the maximum allowed size of 64K.
func (s State) Size() int { func (s *State) Size() int {
if s.size != 0 {
return s.size
}
if s.Proto() == "tcp" { if s.Proto() == "tcp" {
s.size = dns.MaxMsgSize
return dns.MaxMsgSize return dns.MaxMsgSize
} }
if o := s.Req.IsEdns0(); o != nil { if o := s.Req.IsEdns0(); o != nil {
s := o.UDPSize() if o.Do() == true {
if s < dns.MinMsgSize { s.do = doTrue
s = dns.MinMsgSize } else {
s.do = doFalse
} }
return int(s)
size := o.UDPSize()
if size < dns.MinMsgSize {
size = dns.MinMsgSize
}
s.size = int(size)
return int(size)
} }
s.size = dns.MinMsgSize
return dns.MinMsgSize return dns.MinMsgSize
} }
// SizeAndDo returns a ready made OPT record that the reflects the intent from // SizeAndDo returns a ready made OPT record that the reflects the intent from
// state. This can be added to upstream requests that will then hopefully // state. This can be added to upstream requests that will then hopefully
// return a message that is fits the buffer in the client. // return a message that is fits the buffer in the client.
func (s State) SizeAndDo() *dns.OPT { func (s *State) SizeAndDo() *dns.OPT {
size := s.Size() size := s.Size()
Do := s.Do() Do := s.Do()
@@ -134,7 +159,7 @@ const (
// the TC bit will be set regardless of protocol, even TCP message will get the bit, the client // the TC bit will be set regardless of protocol, even TCP message will get the bit, the client
// should then retry with pigeons. // should then retry with pigeons.
// TODO(referral). // TODO(referral).
func (s State) Scrub(reply *dns.Msg) (*dns.Msg, Result) { func (s *State) Scrub(reply *dns.Msg) (*dns.Msg, Result) {
size := s.Size() size := s.Size()
l := reply.Len() l := reply.Len()
if size >= l { if size >= l {
@@ -150,32 +175,36 @@ func (s State) Scrub(reply *dns.Msg) (*dns.Msg, Result) {
// Still?!! does not fit. // Still?!! does not fit.
reply.Truncated = true reply.Truncated = true
return reply, ScrubDone return reply, ScrubDone
} }
// Type returns the type of the question as a string. // Type returns the type of the question as a string.
func (s State) Type() string { return dns.Type(s.Req.Question[0].Qtype).String() } func (s *State) Type() string { return dns.Type(s.Req.Question[0].Qtype).String() }
// QType returns the type of the question as a uint16. // QType returns the type of the question as a uint16.
func (s State) QType() uint16 { return s.Req.Question[0].Qtype } func (s *State) QType() uint16 { return s.Req.Question[0].Qtype }
// Name returns the name of the question in the request. Note // Name returns the name of the question in the request. Note
// this name will always have a closing dot and will be lower cased. // this name will always have a closing dot and will be lower cased.
func (s State) Name() string { return strings.ToLower(dns.Name(s.Req.Question[0].Name).String()) } func (s *State) Name() string { return strings.ToLower(dns.Name(s.Req.Question[0].Name).String()) }
// QName returns the name of the question in the request. // QName returns the name of the question in the request.
func (s State) QName() string { return dns.Name(s.Req.Question[0].Name).String() } func (s *State) QName() string { return dns.Name(s.Req.Question[0].Name).String() }
// Class returns the class of the question in the request. // Class returns the class of the question in the request.
func (s State) Class() string { return dns.Class(s.Req.Question[0].Qclass).String() } func (s *State) Class() string { return dns.Class(s.Req.Question[0].Qclass).String() }
// QClass returns the class of the question in the request. // QClass returns the class of the question in the request.
func (s State) QClass() uint16 { return s.Req.Question[0].Qclass } func (s *State) QClass() uint16 { return s.Req.Question[0].Qclass }
// ErrorMessage returns an error message suitable for sending // ErrorMessage returns an error message suitable for sending
// back to the client. // back to the client.
func (s State) ErrorMessage(rcode int) *dns.Msg { func (s *State) ErrorMessage(rcode int) *dns.Msg {
m := new(dns.Msg) m := new(dns.Msg)
m.SetRcode(s.Req, rcode) m.SetRcode(s.Req, rcode)
return m return m
} }
const (
doTrue = 1
doFalse = 2
)

View File

@@ -1,5 +1,46 @@
package middleware package middleware
import (
"testing"
coretest "github.com/miekg/coredns/middleware/testing"
"github.com/miekg/dns"
)
func TestStateDo(t *testing.T) {
st := testState()
st.Do()
if st.do == 0 {
t.Fatalf("expected st.do to be set")
}
}
func BenchmarkStateDo(b *testing.B) {
st := testState()
for i := 0; i < b.N; i++ {
st.Do()
}
}
func BenchmarkStateSize(b *testing.B) {
st := testState()
for i := 0; i < b.N; i++ {
st.Size()
}
}
func testState() State {
m := new(dns.Msg)
m.SetQuestion("example.com.", dns.TypeA)
m.SetEdns0(4097, true)
return State{W: &coretest.ResponseWriter{}, Req: m}
}
/* /*
func TestHeader(t *testing.T) { func TestHeader(t *testing.T) {
state := getContextOrFail(t) state := getContextOrFail(t)

View File

@@ -3,8 +3,6 @@ package testing
import ( import (
"testing" "testing"
"github.com/miekg/coredns/middleware"
"github.com/miekg/dns" "github.com/miekg/dns"
"golang.org/x/net/context" "golang.org/x/net/context"
) )
@@ -199,11 +197,27 @@ func Section(t *testing.T, tc Case, sect Sect, rr []dns.RR) bool {
return true return true
} }
func ErrorHandler() middleware.Handler { func ErrorHandler() Handler {
return middleware.HandlerFunc(func(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) { return HandlerFunc(func(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
m := new(dns.Msg) m := new(dns.Msg)
m.SetRcode(r, dns.RcodeServerFailure) m.SetRcode(r, dns.RcodeServerFailure)
w.WriteMsg(m) w.WriteMsg(m)
return dns.RcodeServerFailure, nil return dns.RcodeServerFailure, nil
}) })
} }
// Copied here to prevent an import cycle.
type (
// HandlerFunc is a convenience type like dns.HandlerFunc, except
// ServeDNS returns an rcode and an error.
HandlerFunc func(context.Context, dns.ResponseWriter, *dns.Msg) (int, error)
Handler interface {
ServeDNS(context.Context, dns.ResponseWriter, *dns.Msg) (int, error)
}
)
// ServeDNS implements the Handler interface.
func (f HandlerFunc) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
return f(ctx, w, r)
}

View File

@@ -0,0 +1,28 @@
package testing
import (
"net"
"github.com/miekg/dns"
)
type ResponseWriter struct{}
func (t *ResponseWriter) LocalAddr() net.Addr {
ip := net.ParseIP("127.0.0.1")
port := 53
return &net.UDPAddr{IP: ip, Port: port, Zone: ""}
}
func (t *ResponseWriter) RemoteAddr() net.Addr {
ip := net.ParseIP("10.240.0.1")
port := 40212
return &net.UDPAddr{IP: ip, Port: port, Zone: ""}
}
func (t *ResponseWriter) WriteMsg(m *dns.Msg) error { return nil }
func (t *ResponseWriter) Write(buf []byte) (int, error) { return len(buf), nil }
func (t *ResponseWriter) Close() error { return nil }
func (t *ResponseWriter) TsigStatus() error { return nil }
func (t *ResponseWriter) TsigTimersOnly(bool) { return }
func (t *ResponseWriter) Hijack() { return }