From 7b98ae921ef1eb7d1a6fe25f6546b21332399893 Mon Sep 17 00:00:00 2001 From: James Phillips Date: Thu, 3 Nov 2016 12:21:16 -0700 Subject: [PATCH] Adds support for DNS recurse truncation. (#2467) * Return message from recurse even if truncated Signed-off-by: Evan Farrar * Tweaks unit test. --- command/agent/dns.go | 2 +- command/agent/dns_test.go | 54 +++++++++++++++++++++++++++++++++++++++ 2 files changed, 55 insertions(+), 1 deletion(-) diff --git a/command/agent/dns.go b/command/agent/dns.go index 3effaef7fd..417685a52b 100644 --- a/command/agent/dns.go +++ b/command/agent/dns.go @@ -913,7 +913,7 @@ func (d *DNSServer) handleRecurse(resp dns.ResponseWriter, req *dns.Msg) { var err error for _, recursor := range d.recursors { r, rtt, err = c.Exchange(req, recursor) - if err == nil { + if err == nil || err == dns.ErrTruncated { // Compress the response; we don't know if the incoming // response was compressed or not, so by not compressing // we might generate an invalid packet on the way out. diff --git a/command/agent/dns_test.go b/command/agent/dns_test.go index 109ccb3816..64c6886135 100644 --- a/command/agent/dns_test.go +++ b/command/agent/dns_test.go @@ -86,6 +86,25 @@ func makeRecursor(t *testing.T, answer []dns.RR) *dns.Server { return server } +func makeRecursorWithMessage(t *testing.T, answer dns.Msg) *dns.Server { + dnsConf := nextConfig() + dnsAddr := fmt.Sprintf("%s:%d", dnsConf.Addresses.DNS, dnsConf.Ports.DNS) + mux := dns.NewServeMux() + mux.HandleFunc(".", func(resp dns.ResponseWriter, msg *dns.Msg) { + answer.SetReply(msg) + if err := resp.WriteMsg(&answer); err != nil { + t.Fatalf("err: %s", err) + } + }) + server := &dns.Server{ + Addr: dnsAddr, + Net: "udp", + Handler: mux, + } + go server.ListenAndServe() + return server +} + // dnsCNAME returns a DNS CNAME record struct func dnsCNAME(src, dest string) *dns.CNAME { return &dns.CNAME{ @@ -1823,6 +1842,41 @@ func TestDNS_Recurse(t *testing.T) { } } +func TestDNS_Recurse_Truncation(t *testing.T) { + answerMessage := dns.Msg{ + MsgHdr: dns.MsgHdr{Truncated: true}, + Answer: []dns.RR{dnsA("apple.com", "1.2.3.4")}, + } + + recursor := makeRecursorWithMessage(t, answerMessage) + defer recursor.Shutdown() + + dir, srv := makeDNSServerConfig(t, func(c *Config) { + c.DNSRecursor = recursor.Addr + }, nil) + defer os.RemoveAll(dir) + defer srv.agent.Shutdown() + + m := new(dns.Msg) + m.SetQuestion("apple.com.", dns.TypeANY) + + c := new(dns.Client) + addr, _ := srv.agent.config.ClientListener("", srv.agent.config.Ports.DNS) + in, _, err := c.Exchange(m, addr.String()) + if err == nil || err != dns.ErrTruncated { + t.Fatalf("err: %v", err) + } + if in.Truncated != true { + t.Fatalf("err: message should have been truncated %v", in) + } + if len(in.Answer) == 0 { + t.Fatalf("Bad: Truncated message ignored, expected some reply %#v", in) + } + if in.Rcode != dns.RcodeSuccess { + t.Fatalf("Bad: %#v", in) + } +} + func TestDNS_RecursorTimeout(t *testing.T) { serverClientTimeout := 3 * time.Second testClientTimeout := serverClientTimeout + 5*time.Second