123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 |
- //go:build !windows
- package libnetwork
- import (
- "net"
- "testing"
- "github.com/docker/docker/internal/testutils/netnsutils"
- "github.com/miekg/dns"
- )
- // test only works on linux
- func TestDNSIPQuery(t *testing.T) {
- defer netnsutils.SetupTestOSContext(t)()
- c, err := New()
- if err != nil {
- t.Fatal(err)
- }
- defer c.Stop()
- n, err := c.NewNetwork("bridge", "dtnet1", "", nil)
- if err != nil {
- t.Fatal(err)
- }
- defer func() {
- if err := n.Delete(); err != nil {
- t.Fatal(err)
- }
- }()
- ep, err := n.CreateEndpoint("testep")
- if err != nil {
- t.Fatal(err)
- }
- sb, err := c.NewSandbox("c1")
- if err != nil {
- t.Fatal(err)
- }
- defer func() {
- if err := sb.Delete(); err != nil {
- t.Fatal(err)
- }
- }()
- // we need the endpoint only to populate ep_list for the sandbox as part of resolve_name
- // it is not set as a target for name resolution and does not serve any other purpose
- err = ep.Join(sb)
- if err != nil {
- t.Fatal(err)
- }
- // add service records which are used to resolve names. These are the real targets for the DNS querries
- n.addSvcRecords("ep1", "name1", "svc1", net.ParseIP("192.168.0.1"), net.IP{}, true, "test")
- w := new(tstwriter)
- // the unit tests right now will focus on non-proxyed DNS requests
- r := NewResolver(resolverIPSandbox, false, sb)
- // test name1's IP is resolved correctly with the default A type query
- // Also make sure DNS lookups are case insensitive
- names := []string{"name1", "NaMe1"}
- for _, name := range names {
- q := new(dns.Msg)
- q.SetQuestion(name, dns.TypeA)
- r.serveDNS(w, q)
- resp := w.GetResponse()
- checkNonNullResponse(t, resp)
- t.Log("Response: ", resp.String())
- checkDNSResponseCode(t, resp, dns.RcodeSuccess)
- checkDNSAnswersCount(t, resp, 1)
- checkDNSRRType(t, resp.Answer[0].Header().Rrtype, dns.TypeA)
- if answer, ok := resp.Answer[0].(*dns.A); ok {
- if !answer.A.Equal(net.ParseIP("192.168.0.1")) {
- t.Fatalf("IP response in Answer %v does not match 192.168.0.1", answer.A)
- }
- } else {
- t.Fatal("Answer of type A not found")
- }
- w.ClearResponse()
- }
- // test MX query with name1 results in Success response with 0 answer records
- q := new(dns.Msg)
- q.SetQuestion("name1", dns.TypeMX)
- r.serveDNS(w, q)
- resp := w.GetResponse()
- checkNonNullResponse(t, resp)
- t.Log("Response: ", resp.String())
- checkDNSResponseCode(t, resp, dns.RcodeSuccess)
- checkDNSAnswersCount(t, resp, 0)
- w.ClearResponse()
- // test MX query with non existent name results in ServFail response with 0 answer records
- // since this is a unit test env, we disable proxying DNS above which results in ServFail rather than NXDOMAIN
- q = new(dns.Msg)
- q.SetQuestion("nonexistent", dns.TypeMX)
- r.serveDNS(w, q)
- resp = w.GetResponse()
- checkNonNullResponse(t, resp)
- t.Log("Response: ", resp.String())
- checkDNSResponseCode(t, resp, dns.RcodeServerFailure)
- w.ClearResponse()
- }
- // test only works on linux
- func TestDNSProxyServFail(t *testing.T) {
- osctx := netnsutils.SetupTestOSContextEx(t)
- defer osctx.Cleanup(t)
- c, err := New()
- if err != nil {
- t.Fatal(err)
- }
- defer c.Stop()
- n, err := c.NewNetwork("bridge", "dtnet2", "", nil)
- if err != nil {
- t.Fatal(err)
- }
- defer func() {
- if err := n.Delete(); err != nil {
- t.Fatal(err)
- }
- }()
- sb, err := c.NewSandbox("c1")
- if err != nil {
- t.Fatal(err)
- }
- defer func() {
- if err := sb.Delete(); err != nil {
- t.Fatal(err)
- }
- }()
- var nRequests int
- // initialize a local DNS server and configure it to fail the first query
- dns.HandleFunc(".", newDNSHandlerServFailOnce(&nRequests))
- // use TCP for predictable results. Connection tests (to figure out DNS server initialization) don't work with UDP
- server := &dns.Server{Addr: "127.0.0.1:53", Net: "tcp"}
- srvErrCh := make(chan error, 1)
- osctx.Go(t, func() {
- srvErrCh <- server.ListenAndServe()
- })
- defer func() {
- server.Shutdown() //nolint:errcheck
- if err := <-srvErrCh; err != nil {
- t.Error(err)
- }
- }()
- waitForLocalDNSServer(t)
- t.Log("DNS Server can be reached")
- w := new(tstwriter)
- r := NewResolver(resolverIPSandbox, true, sb)
- q := new(dns.Msg)
- q.SetQuestion("name1.", dns.TypeA)
- var localDNSEntries []extDNSEntry
- extTestDNSEntry := extDNSEntry{IPStr: "127.0.0.1", HostLoopback: true}
- // configure two external DNS entries and point both to local DNS server thread
- localDNSEntries = append(localDNSEntries, extTestDNSEntry)
- localDNSEntries = append(localDNSEntries, extTestDNSEntry)
- // this should generate two requests: the first will fail leading to a retry
- r.SetExtServers(localDNSEntries)
- r.serveDNS(w, q)
- if nRequests != 2 {
- t.Fatalf("Expected 2 DNS querries. Found: %d", nRequests)
- }
- t.Logf("Expected number of DNS requests generated")
- }
|