defaults.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. package dns
  2. import (
  3. "errors"
  4. "net"
  5. "strconv"
  6. )
  7. const hexDigit = "0123456789abcdef"
  8. // Everything is assumed in ClassINET.
  9. // SetReply creates a reply message from a request message.
  10. func (dns *Msg) SetReply(request *Msg) *Msg {
  11. dns.Id = request.Id
  12. dns.Response = true
  13. dns.Opcode = request.Opcode
  14. if dns.Opcode == OpcodeQuery {
  15. dns.RecursionDesired = request.RecursionDesired // Copy rd bit
  16. dns.CheckingDisabled = request.CheckingDisabled // Copy cd bit
  17. }
  18. dns.Rcode = RcodeSuccess
  19. if len(request.Question) > 0 {
  20. dns.Question = make([]Question, 1)
  21. dns.Question[0] = request.Question[0]
  22. }
  23. return dns
  24. }
  25. // SetQuestion creates a question message, it sets the Question
  26. // section, generates an Id and sets the RecursionDesired (RD)
  27. // bit to true.
  28. func (dns *Msg) SetQuestion(z string, t uint16) *Msg {
  29. dns.Id = Id()
  30. dns.RecursionDesired = true
  31. dns.Question = make([]Question, 1)
  32. dns.Question[0] = Question{z, t, ClassINET}
  33. return dns
  34. }
  35. // SetNotify creates a notify message, it sets the Question
  36. // section, generates an Id and sets the Authoritative (AA)
  37. // bit to true.
  38. func (dns *Msg) SetNotify(z string) *Msg {
  39. dns.Opcode = OpcodeNotify
  40. dns.Authoritative = true
  41. dns.Id = Id()
  42. dns.Question = make([]Question, 1)
  43. dns.Question[0] = Question{z, TypeSOA, ClassINET}
  44. return dns
  45. }
  46. // SetRcode creates an error message suitable for the request.
  47. func (dns *Msg) SetRcode(request *Msg, rcode int) *Msg {
  48. dns.SetReply(request)
  49. dns.Rcode = rcode
  50. return dns
  51. }
  52. // SetRcodeFormatError creates a message with FormError set.
  53. func (dns *Msg) SetRcodeFormatError(request *Msg) *Msg {
  54. dns.Rcode = RcodeFormatError
  55. dns.Opcode = OpcodeQuery
  56. dns.Response = true
  57. dns.Authoritative = false
  58. dns.Id = request.Id
  59. return dns
  60. }
  61. // SetUpdate makes the message a dynamic update message. It
  62. // sets the ZONE section to: z, TypeSOA, ClassINET.
  63. func (dns *Msg) SetUpdate(z string) *Msg {
  64. dns.Id = Id()
  65. dns.Response = false
  66. dns.Opcode = OpcodeUpdate
  67. dns.Compress = false // BIND9 cannot handle compression
  68. dns.Question = make([]Question, 1)
  69. dns.Question[0] = Question{z, TypeSOA, ClassINET}
  70. return dns
  71. }
  72. // SetIxfr creates message for requesting an IXFR.
  73. func (dns *Msg) SetIxfr(z string, serial uint32, ns, mbox string) *Msg {
  74. dns.Id = Id()
  75. dns.Question = make([]Question, 1)
  76. dns.Ns = make([]RR, 1)
  77. s := new(SOA)
  78. s.Hdr = RR_Header{z, TypeSOA, ClassINET, defaultTtl, 0}
  79. s.Serial = serial
  80. s.Ns = ns
  81. s.Mbox = mbox
  82. dns.Question[0] = Question{z, TypeIXFR, ClassINET}
  83. dns.Ns[0] = s
  84. return dns
  85. }
  86. // SetAxfr creates message for requesting an AXFR.
  87. func (dns *Msg) SetAxfr(z string) *Msg {
  88. dns.Id = Id()
  89. dns.Question = make([]Question, 1)
  90. dns.Question[0] = Question{z, TypeAXFR, ClassINET}
  91. return dns
  92. }
  93. // SetTsig appends a TSIG RR to the message.
  94. // This is only a skeleton TSIG RR that is added as the last RR in the
  95. // additional section. The Tsig is calculated when the message is being send.
  96. func (dns *Msg) SetTsig(z, algo string, fudge uint16, timesigned int64) *Msg {
  97. t := new(TSIG)
  98. t.Hdr = RR_Header{z, TypeTSIG, ClassANY, 0, 0}
  99. t.Algorithm = algo
  100. t.Fudge = fudge
  101. t.TimeSigned = uint64(timesigned)
  102. t.OrigId = dns.Id
  103. dns.Extra = append(dns.Extra, t)
  104. return dns
  105. }
  106. // SetEdns0 appends a EDNS0 OPT RR to the message.
  107. // TSIG should always the last RR in a message.
  108. func (dns *Msg) SetEdns0(udpsize uint16, do bool) *Msg {
  109. e := new(OPT)
  110. e.Hdr.Name = "."
  111. e.Hdr.Rrtype = TypeOPT
  112. e.SetUDPSize(udpsize)
  113. if do {
  114. e.SetDo()
  115. }
  116. dns.Extra = append(dns.Extra, e)
  117. return dns
  118. }
  119. // IsTsig checks if the message has a TSIG record as the last record
  120. // in the additional section. It returns the TSIG record found or nil.
  121. func (dns *Msg) IsTsig() *TSIG {
  122. if len(dns.Extra) > 0 {
  123. if dns.Extra[len(dns.Extra)-1].Header().Rrtype == TypeTSIG {
  124. return dns.Extra[len(dns.Extra)-1].(*TSIG)
  125. }
  126. }
  127. return nil
  128. }
  129. // IsEdns0 checks if the message has a EDNS0 (OPT) record, any EDNS0
  130. // record in the additional section will do. It returns the OPT record
  131. // found or nil.
  132. func (dns *Msg) IsEdns0() *OPT {
  133. // EDNS0 is at the end of the additional section, start there.
  134. // We might want to change this to *only* look at the last two
  135. // records. So we see TSIG and/or OPT - this a slightly bigger
  136. // change though.
  137. for i := len(dns.Extra) - 1; i >= 0; i-- {
  138. if dns.Extra[i].Header().Rrtype == TypeOPT {
  139. return dns.Extra[i].(*OPT)
  140. }
  141. }
  142. return nil
  143. }
  144. // IsDomainName checks if s is a valid domain name, it returns the number of
  145. // labels and true, when a domain name is valid. Note that non fully qualified
  146. // domain name is considered valid, in this case the last label is counted in
  147. // the number of labels. When false is returned the number of labels is not
  148. // defined. Also note that this function is extremely liberal; almost any
  149. // string is a valid domain name as the DNS is 8 bit protocol. It checks if each
  150. // label fits in 63 characters, but there is no length check for the entire
  151. // string s. I.e. a domain name longer than 255 characters is considered valid.
  152. func IsDomainName(s string) (labels int, ok bool) {
  153. _, labels, err := packDomainName(s, nil, 0, nil, false)
  154. return labels, err == nil
  155. }
  156. // IsSubDomain checks if child is indeed a child of the parent. If child and parent
  157. // are the same domain true is returned as well.
  158. func IsSubDomain(parent, child string) bool {
  159. // Entire child is contained in parent
  160. return CompareDomainName(parent, child) == CountLabel(parent)
  161. }
  162. // IsMsg sanity checks buf and returns an error if it isn't a valid DNS packet.
  163. // The checking is performed on the binary payload.
  164. func IsMsg(buf []byte) error {
  165. // Header
  166. if len(buf) < 12 {
  167. return errors.New("dns: bad message header")
  168. }
  169. // Header: Opcode
  170. // TODO(miek): more checks here, e.g. check all header bits.
  171. return nil
  172. }
  173. // IsFqdn checks if a domain name is fully qualified.
  174. func IsFqdn(s string) bool {
  175. l := len(s)
  176. if l == 0 {
  177. return false
  178. }
  179. return s[l-1] == '.'
  180. }
  181. // IsRRset checks if a set of RRs is a valid RRset as defined by RFC 2181.
  182. // This means the RRs need to have the same type, name, and class. Returns true
  183. // if the RR set is valid, otherwise false.
  184. func IsRRset(rrset []RR) bool {
  185. if len(rrset) == 0 {
  186. return false
  187. }
  188. if len(rrset) == 1 {
  189. return true
  190. }
  191. rrHeader := rrset[0].Header()
  192. rrType := rrHeader.Rrtype
  193. rrClass := rrHeader.Class
  194. rrName := rrHeader.Name
  195. for _, rr := range rrset[1:] {
  196. curRRHeader := rr.Header()
  197. if curRRHeader.Rrtype != rrType || curRRHeader.Class != rrClass || curRRHeader.Name != rrName {
  198. // Mismatch between the records, so this is not a valid rrset for
  199. //signing/verifying
  200. return false
  201. }
  202. }
  203. return true
  204. }
  205. // Fqdn return the fully qualified domain name from s.
  206. // If s is already fully qualified, it behaves as the identity function.
  207. func Fqdn(s string) string {
  208. if IsFqdn(s) {
  209. return s
  210. }
  211. return s + "."
  212. }
  213. // Copied from the official Go code.
  214. // ReverseAddr returns the in-addr.arpa. or ip6.arpa. hostname of the IP
  215. // address suitable for reverse DNS (PTR) record lookups or an error if it fails
  216. // to parse the IP address.
  217. func ReverseAddr(addr string) (arpa string, err error) {
  218. ip := net.ParseIP(addr)
  219. if ip == nil {
  220. return "", &Error{err: "unrecognized address: " + addr}
  221. }
  222. if ip.To4() != nil {
  223. return strconv.Itoa(int(ip[15])) + "." + strconv.Itoa(int(ip[14])) + "." + strconv.Itoa(int(ip[13])) + "." +
  224. strconv.Itoa(int(ip[12])) + ".in-addr.arpa.", nil
  225. }
  226. // Must be IPv6
  227. buf := make([]byte, 0, len(ip)*4+len("ip6.arpa."))
  228. // Add it, in reverse, to the buffer
  229. for i := len(ip) - 1; i >= 0; i-- {
  230. v := ip[i]
  231. buf = append(buf, hexDigit[v&0xF])
  232. buf = append(buf, '.')
  233. buf = append(buf, hexDigit[v>>4])
  234. buf = append(buf, '.')
  235. }
  236. // Append "ip6.arpa." and return (buf already has the final .)
  237. buf = append(buf, "ip6.arpa."...)
  238. return string(buf), nil
  239. }
  240. // String returns the string representation for the type t.
  241. func (t Type) String() string {
  242. if t1, ok := TypeToString[uint16(t)]; ok {
  243. return t1
  244. }
  245. return "TYPE" + strconv.Itoa(int(t))
  246. }
  247. // String returns the string representation for the class c.
  248. func (c Class) String() string {
  249. if s, ok := ClassToString[uint16(c)]; ok {
  250. // Only emit mnemonics when they are unambiguous, specically ANY is in both.
  251. if _, ok := StringToType[s]; !ok {
  252. return s
  253. }
  254. }
  255. return "CLASS" + strconv.Itoa(int(c))
  256. }
  257. // String returns the string representation for the name n.
  258. func (n Name) String() string {
  259. return sprintName(string(n))
  260. }