serve_test.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. package main
  2. import (
  3. "github.com/abh/dns"
  4. . "launchpad.net/gocheck"
  5. "net"
  6. "strings"
  7. "time"
  8. )
  9. const (
  10. PORT = ":8853"
  11. )
  12. type ServeSuite struct {
  13. }
  14. var _ = Suite(&ServeSuite{})
  15. func (s *ServeSuite) SetUpSuite(c *C) {
  16. // log.Println("Setting up serve test suite")
  17. Zones := make(Zones)
  18. setupPgeodnsZone(Zones)
  19. zonesReadDir("dns", Zones)
  20. go listenAndServe(PORT)
  21. time.Sleep(200 * time.Millisecond)
  22. }
  23. func (s *ServeSuite) TestServing(c *C) {
  24. r := exchange(c, "_status.pgeodns.", dns.TypeTXT)
  25. txt := r.Answer[0].(*dns.TXT).Txt[0]
  26. if !strings.HasPrefix(txt, "{") {
  27. c.Log("Unexpected result for _status.pgeodns", txt)
  28. c.Fail()
  29. }
  30. // Allow _country and _status queries as long as the first label is that
  31. r = exchange(c, "_country.foo.pgeodns.", dns.TypeTXT)
  32. txt = r.Answer[0].(*dns.TXT).Txt[0]
  33. // Got appropriate response for _country txt query
  34. if !strings.HasPrefix(txt, "127.0.0.1:") {
  35. c.Log("Unexpected result for _country.foo.pgeodns", txt)
  36. c.Fail()
  37. }
  38. // Make sure A requests for _status doesn't NXDOMAIN
  39. r = exchange(c, "_status.pgeodns.", dns.TypeA)
  40. c.Check(r.Answer, HasLen, 0)
  41. // Got one SOA record
  42. c.Check(r.Ns, HasLen, 1)
  43. // NOERROR for A request
  44. c.Check(r.Rcode, Equals, dns.RcodeSuccess)
  45. r = exchange(c, "bar.test.example.com.", dns.TypeA)
  46. ip := r.Answer[0].(*dns.A).A
  47. c.Check(ip.String(), Equals, "192.168.1.2")
  48. r = exchange(c, "test.example.com.", dns.TypeSOA)
  49. soa := r.Answer[0].(*dns.SOA)
  50. serial := soa.Serial
  51. c.Check(int(serial), Equals, 3)
  52. // no AAAA records for 'bar', so check we get a soa record back
  53. r = exchange(c, "test.example.com.", dns.TypeAAAA)
  54. soa2 := r.Ns[0].(*dns.SOA)
  55. c.Check(soa, DeepEquals, soa2)
  56. // CNAMEs
  57. r = exchange(c, "www.test.example.com.", dns.TypeA)
  58. c.Check(r.Answer[0].(*dns.CNAME).Target, Equals, "geo.bitnames.com.")
  59. // MX
  60. r = exchange(c, "test.example.com.", dns.TypeMX)
  61. c.Check(r.Answer[0].(*dns.MX).Mx, Equals, "mx.example.net.")
  62. c.Check(r.Answer[1].(*dns.MX).Mx, Equals, "mx2.example.net.")
  63. c.Check(r.Answer[1].(*dns.MX).Preference, Equals, uint16(20))
  64. }
  65. func (s *ServeSuite) TestServingAliases(c *C) {
  66. // Alias, no geo matches
  67. r := exchange(c, "bar-alias.test.example.com.", dns.TypeA)
  68. ip := r.Answer[0].(*dns.A).A
  69. c.Check(ip.String(), Equals, "192.168.1.2")
  70. // Alias to a cname record
  71. r = exchange(c, "www-alias.test.example.com.", dns.TypeA)
  72. c.Check(r.Answer[0].(*dns.CNAME).Target, Equals, "geo.bitnames.com.")
  73. // Alias returning a cname, with geo overrides
  74. r = exchangeSubnet(c, "www-alias.test.example.com.", dns.TypeA, "194.239.134.1")
  75. c.Check(r.Answer, HasLen, 1)
  76. if len(r.Answer) > 0 {
  77. c.Check(r.Answer[0].(*dns.CNAME).Target, Equals, "geo-europe.bitnames.com.")
  78. }
  79. // Alias to Ns records
  80. r = exchange(c, "sub-alias.test.example.org.", dns.TypeNS)
  81. c.Check(r.Answer[0].(*dns.NS).Ns, Equals, "ns1.example.com.")
  82. }
  83. func (s *ServeSuite) TestServingEDNS(c *C) {
  84. // MX test
  85. r := exchangeSubnet(c, "test.example.com.", dns.TypeMX, "194.239.134.1")
  86. c.Check(r.Answer, HasLen, 1)
  87. if len(r.Answer) > 0 {
  88. c.Check(r.Answer[0].(*dns.MX).Mx, Equals, "mx-eu.example.net.")
  89. }
  90. c.Log("Testing www.test.example.com from .dk, should match www.europe (a cname)")
  91. r = exchangeSubnet(c, "www.test.example.com.", dns.TypeA, "194.239.134.0")
  92. c.Check(r.Answer, HasLen, 1)
  93. if len(r.Answer) > 0 {
  94. // EDNS-SUBNET test (request A, respond CNAME)
  95. c.Check(r.Answer[0].(*dns.CNAME).Target, Equals, "geo-europe.bitnames.com.")
  96. }
  97. }
  98. func exchangeSubnet(c *C, name string, dnstype uint16, ip string) *dns.Msg {
  99. msg := new(dns.Msg)
  100. msg.SetQuestion(name, dnstype)
  101. o := new(dns.OPT)
  102. o.Hdr.Name = "."
  103. o.Hdr.Rrtype = dns.TypeOPT
  104. e := new(dns.EDNS0_SUBNET)
  105. e.Code = dns.EDNS0SUBNET
  106. e.SourceScope = 0
  107. e.Address = net.ParseIP(ip)
  108. e.Family = 1 // IP4
  109. e.SourceNetmask = net.IPv4len * 8
  110. o.Option = append(o.Option, e)
  111. msg.Extra = append(msg.Extra, o)
  112. c.Log("msg", msg)
  113. return dorequest(c, msg)
  114. }
  115. func exchange(c *C, name string, dnstype uint16) *dns.Msg {
  116. msg := new(dns.Msg)
  117. msg.SetQuestion(name, dnstype)
  118. return dorequest(c, msg)
  119. }
  120. func dorequest(c *C, msg *dns.Msg) *dns.Msg {
  121. cli := new(dns.Client)
  122. r, _, err := cli.Exchange(msg, "127.0.0.1"+PORT)
  123. if err != nil {
  124. c.Log("err", err)
  125. c.Fail()
  126. }
  127. return r
  128. }