serve.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. package main
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "log"
  6. "net"
  7. "os"
  8. "strconv"
  9. "strings"
  10. "time"
  11. "github.com/abh/geodns/Godeps/_workspace/src/github.com/miekg/dns"
  12. "github.com/abh/geodns/Godeps/_workspace/src/github.com/rcrowley/go-metrics"
  13. )
  14. func getQuestionName(z *Zone, req *dns.Msg) string {
  15. lx := dns.SplitDomainName(req.Question[0].Name)
  16. ql := lx[0 : len(lx)-z.LabelCount]
  17. return strings.ToLower(strings.Join(ql, "."))
  18. }
  19. func serve(w dns.ResponseWriter, req *dns.Msg, z *Zone) {
  20. qtype := req.Question[0].Qtype
  21. logPrintf("[zone %s] incoming %s %s %d from %s\n", z.Origin, req.Question[0].Name,
  22. dns.TypeToString[qtype], req.MsgHdr.Id, w.RemoteAddr())
  23. // Global meter
  24. metrics.Get("queries").(metrics.Meter).Mark(1)
  25. // Zone meter
  26. z.Metrics.Queries.Mark(1)
  27. logPrintln("Got request", req)
  28. label := getQuestionName(z, req)
  29. z.Metrics.LabelStats.Add(label)
  30. // IP that's talking to us (not EDNS CLIENT SUBNET)
  31. var realIP net.IP
  32. if addr, ok := w.RemoteAddr().(*net.UDPAddr); ok {
  33. realIP = addr.IP
  34. } else if addr, ok := w.RemoteAddr().(*net.TCPAddr); ok {
  35. realIP = addr.IP
  36. }
  37. z.Metrics.ClientStats.Add(realIP.String())
  38. var ip net.IP // EDNS or real IP
  39. var edns *dns.EDNS0_SUBNET
  40. var opt_rr *dns.OPT
  41. for _, extra := range req.Extra {
  42. switch extra.(type) {
  43. case *dns.OPT:
  44. for _, o := range extra.(*dns.OPT).Option {
  45. opt_rr = extra.(*dns.OPT)
  46. switch e := o.(type) {
  47. case *dns.EDNS0_NSID:
  48. // do stuff with e.Nsid
  49. case *dns.EDNS0_SUBNET:
  50. z.Metrics.EdnsQueries.Mark(1)
  51. logPrintln("Got edns", e.Address, e.Family, e.SourceNetmask, e.SourceScope)
  52. if e.Address != nil {
  53. edns = e
  54. ip = e.Address
  55. }
  56. }
  57. }
  58. }
  59. }
  60. if len(ip) == 0 { // no edns subnet
  61. ip = realIP
  62. }
  63. targets, netmask := z.Options.Targeting.GetTargets(ip)
  64. m := new(dns.Msg)
  65. m.SetReply(req)
  66. if e := m.IsEdns0(); e != nil {
  67. m.SetEdns0(4096, e.Do())
  68. }
  69. m.Authoritative = true
  70. // TODO: set scope to 0 if there are no alternate responses
  71. if edns != nil {
  72. if edns.Family != 0 {
  73. if netmask < 16 {
  74. netmask = 16
  75. }
  76. edns.SourceScope = uint8(netmask)
  77. m.Extra = append(m.Extra, opt_rr)
  78. }
  79. }
  80. labels, labelQtype := z.findLabels(label, targets, qTypes{dns.TypeMF, dns.TypeCNAME, qtype})
  81. if labelQtype == 0 {
  82. labelQtype = qtype
  83. }
  84. if labels == nil {
  85. permitDebug := !*flagPrivateDebug || (realIP != nil && realIP.IsLoopback())
  86. firstLabel := (strings.Split(label, "."))[0]
  87. if permitDebug && firstLabel == "_status" {
  88. if qtype == dns.TypeANY || qtype == dns.TypeTXT {
  89. m.Answer = statusRR(label + "." + z.Origin + ".")
  90. } else {
  91. m.Ns = append(m.Ns, z.SoaRR())
  92. }
  93. m.Authoritative = true
  94. w.WriteMsg(m)
  95. return
  96. }
  97. if firstLabel == "_country" {
  98. if qtype == dns.TypeANY || qtype == dns.TypeTXT {
  99. h := dns.RR_Header{Ttl: 1, Class: dns.ClassINET, Rrtype: dns.TypeTXT}
  100. h.Name = label + "." + z.Origin + "."
  101. txt := []string{
  102. w.RemoteAddr().String(),
  103. ip.String(),
  104. }
  105. targets, netmask := z.Options.Targeting.GetTargets(ip)
  106. txt = append(txt, strings.Join(targets, " "))
  107. txt = append(txt, fmt.Sprintf("/%d", netmask), serverID, serverIP)
  108. m.Answer = []dns.RR{&dns.TXT{Hdr: h,
  109. Txt: txt,
  110. }}
  111. } else {
  112. m.Ns = append(m.Ns, z.SoaRR())
  113. }
  114. m.Authoritative = true
  115. w.WriteMsg(m)
  116. return
  117. }
  118. // return NXDOMAIN
  119. m.SetRcode(req, dns.RcodeNameError)
  120. m.Authoritative = true
  121. m.Ns = []dns.RR{z.SoaRR()}
  122. w.WriteMsg(m)
  123. return
  124. }
  125. if servers := labels.Picker(labelQtype, labels.MaxHosts); servers != nil {
  126. var rrs []dns.RR
  127. for _, record := range servers {
  128. rr := dns.Copy(record.RR)
  129. rr.Header().Name = req.Question[0].Name
  130. rrs = append(rrs, rr)
  131. }
  132. m.Answer = rrs
  133. }
  134. if len(m.Answer) == 0 {
  135. m.Ns = append(m.Ns, z.SoaRR())
  136. }
  137. logPrintln(m)
  138. err := w.WriteMsg(m)
  139. if err != nil {
  140. // if Pack'ing fails the Write fails. Return SERVFAIL.
  141. log.Println("Error writing packet", m)
  142. dns.HandleFailed(w, req)
  143. }
  144. return
  145. }
  146. func statusRR(label string) []dns.RR {
  147. h := dns.RR_Header{Ttl: 1, Class: dns.ClassINET, Rrtype: dns.TypeTXT}
  148. h.Name = label
  149. status := map[string]string{"v": VERSION, "id": serverID}
  150. hostname, err := os.Hostname()
  151. if err == nil {
  152. status["h"] = hostname
  153. }
  154. qCounter := metrics.Get("queries").(metrics.Meter)
  155. status["up"] = strconv.Itoa(int(time.Since(timeStarted).Seconds()))
  156. status["qs"] = strconv.FormatInt(qCounter.Count(), 10)
  157. status["qps1"] = fmt.Sprintf("%.4f", qCounter.Rate1())
  158. js, err := json.Marshal(status)
  159. return []dns.RR{&dns.TXT{Hdr: h, Txt: []string{string(js)}}}
  160. }
  161. func setupServerFunc(Zone *Zone) func(dns.ResponseWriter, *dns.Msg) {
  162. return func(w dns.ResponseWriter, r *dns.Msg) {
  163. serve(w, r, Zone)
  164. }
  165. }
  166. func listenAndServe(ip string) {
  167. prots := []string{"udp", "tcp"}
  168. for _, prot := range prots {
  169. go func(p string) {
  170. server := &dns.Server{Addr: ip, Net: p}
  171. log.Printf("Opening on %s %s", ip, p)
  172. if err := server.ListenAndServe(); err != nil {
  173. log.Fatalf("geodns: failed to setup %s %s: %s", ip, p, err)
  174. }
  175. log.Fatalf("geodns: ListenAndServe unexpectedly returned")
  176. }(prot)
  177. }
  178. }