123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- package dns
- import (
- "errors"
- "net"
- "strconv"
- )
- const hexDigit = "0123456789abcdef"
- // Everything is assumed in ClassINET.
- // SetReply creates a reply message from a request message.
- func (dns *Msg) SetReply(request *Msg) *Msg {
- dns.Id = request.Id
- dns.RecursionDesired = request.RecursionDesired // Copy rd bit
- dns.Response = true
- dns.Opcode = OpcodeQuery
- dns.Rcode = RcodeSuccess
- if len(request.Question) > 0 {
- dns.Question = make([]Question, 1)
- dns.Question[0] = request.Question[0]
- }
- return dns
- }
- // SetQuestion creates a question message, it sets the Question
- // section, generates an Id and sets the RecursionDesired (RD)
- // bit to true.
- func (dns *Msg) SetQuestion(z string, t uint16) *Msg {
- dns.Id = Id()
- dns.RecursionDesired = true
- dns.Question = make([]Question, 1)
- dns.Question[0] = Question{z, t, ClassINET}
- return dns
- }
- // SetNotify creates a notify message, it sets the Question
- // section, generates an Id and sets the Authoritative (AA)
- // bit to true.
- func (dns *Msg) SetNotify(z string) *Msg {
- dns.Opcode = OpcodeNotify
- dns.Authoritative = true
- dns.Id = Id()
- dns.Question = make([]Question, 1)
- dns.Question[0] = Question{z, TypeSOA, ClassINET}
- return dns
- }
- // SetRcode creates an error message suitable for the request.
- func (dns *Msg) SetRcode(request *Msg, rcode int) *Msg {
- dns.SetReply(request)
- dns.Rcode = rcode
- return dns
- }
- // SetRcodeFormatError creates a message with FormError set.
- func (dns *Msg) SetRcodeFormatError(request *Msg) *Msg {
- dns.Rcode = RcodeFormatError
- dns.Opcode = OpcodeQuery
- dns.Response = true
- dns.Authoritative = false
- dns.Id = request.Id
- return dns
- }
- // SetUpdate makes the message a dynamic update message. It
- // sets the ZONE section to: z, TypeSOA, ClassINET.
- func (dns *Msg) SetUpdate(z string) *Msg {
- dns.Id = Id()
- dns.Response = false
- dns.Opcode = OpcodeUpdate
- dns.Compress = false // BIND9 cannot handle compression
- dns.Question = make([]Question, 1)
- dns.Question[0] = Question{z, TypeSOA, ClassINET}
- return dns
- }
- // SetIxfr creates message for requesting an IXFR.
- func (dns *Msg) SetIxfr(z string, serial uint32, ns, mbox string) *Msg {
- dns.Id = Id()
- dns.Question = make([]Question, 1)
- dns.Ns = make([]RR, 1)
- s := new(SOA)
- s.Hdr = RR_Header{z, TypeSOA, ClassINET, defaultTtl, 0}
- s.Serial = serial
- s.Ns = ns
- s.Mbox = mbox
- dns.Question[0] = Question{z, TypeIXFR, ClassINET}
- dns.Ns[0] = s
- return dns
- }
- // SetAxfr creates message for requesting an AXFR.
- func (dns *Msg) SetAxfr(z string) *Msg {
- dns.Id = Id()
- dns.Question = make([]Question, 1)
- dns.Question[0] = Question{z, TypeAXFR, ClassINET}
- return dns
- }
- // SetTsig appends a TSIG RR to the message.
- // This is only a skeleton TSIG RR that is added as the last RR in the
- // additional section. The Tsig is calculated when the message is being send.
- func (dns *Msg) SetTsig(z, algo string, fudge, timesigned int64) *Msg {
- t := new(TSIG)
- t.Hdr = RR_Header{z, TypeTSIG, ClassANY, 0, 0}
- t.Algorithm = algo
- t.Fudge = 300
- t.TimeSigned = uint64(timesigned)
- t.OrigId = dns.Id
- dns.Extra = append(dns.Extra, t)
- return dns
- }
- // SetEdns0 appends a EDNS0 OPT RR to the message.
- // TSIG should always the last RR in a message.
- func (dns *Msg) SetEdns0(udpsize uint16, do bool) *Msg {
- e := new(OPT)
- e.Hdr.Name = "."
- e.Hdr.Rrtype = TypeOPT
- e.SetUDPSize(udpsize)
- if do {
- e.SetDo()
- }
- dns.Extra = append(dns.Extra, e)
- return dns
- }
- // IsTsig checks if the message has a TSIG record as the last record
- // in the additional section. It returns the TSIG record found or nil.
- func (dns *Msg) IsTsig() *TSIG {
- if len(dns.Extra) > 0 {
- if dns.Extra[len(dns.Extra)-1].Header().Rrtype == TypeTSIG {
- return dns.Extra[len(dns.Extra)-1].(*TSIG)
- }
- }
- return nil
- }
- // IsEdns0 checks if the message has a EDNS0 (OPT) record, any EDNS0
- // record in the additional section will do. It returns the OPT record
- // found or nil.
- func (dns *Msg) IsEdns0() *OPT {
- for _, r := range dns.Extra {
- if r.Header().Rrtype == TypeOPT {
- return r.(*OPT)
- }
- }
- return nil
- }
- // IsDomainName checks if s is a valid domain name, it returns the number of
- // labels and true, when a domain name is valid. Note that non fully qualified
- // domain name is considered valid, in this case the last label is counted in
- // the number of labels. When false is returned the number of labels is not
- // defined. Also note that this function is extremely liberal; almost any
- // string is a valid domain name as the DNS is 8 bit protocol. It checks if each
- // label fits in 63 characters, but there is no length check for the entire
- // string s. I.e. a domain name longer than 255 characters is considered valid.
- func IsDomainName(s string) (labels int, ok bool) {
- _, labels, err := packDomainName(s, nil, 0, nil, false)
- return labels, err == nil
- }
- // IsSubDomain checks if child is indeed a child of the parent. Both child and
- // parent are *not* downcased before doing the comparison.
- func IsSubDomain(parent, child string) bool {
- // Entire child is contained in parent
- return CompareDomainName(parent, child) == CountLabel(parent)
- }
- // IsMsg sanity checks buf and returns an error if it isn't a valid DNS packet.
- // The checking is performed on the binary payload.
- func IsMsg(buf []byte) error {
- // Header
- if len(buf) < 12 {
- return errors.New("dns: bad message header")
- }
- // Header: Opcode
- // TODO(miek): more checks here, e.g. check all header bits.
- return nil
- }
- // IsFqdn checks if a domain name is fully qualified.
- func IsFqdn(s string) bool {
- l := len(s)
- if l == 0 {
- return false
- }
- return s[l-1] == '.'
- }
- // IsRRset checks if a set of RRs is a valid RRset as defined by RFC 2181.
- // This means the RRs need to have the same type, name, and class. Returns true
- // if the RR set is valid, otherwise false.
- func IsRRset(rrset []RR) bool {
- if len(rrset) == 0 {
- return false
- }
- if len(rrset) == 1 {
- return true
- }
- rrHeader := rrset[0].Header()
- rrType := rrHeader.Rrtype
- rrClass := rrHeader.Class
- rrName := rrHeader.Name
- for _, rr := range rrset[1:] {
- curRRHeader := rr.Header()
- if curRRHeader.Rrtype != rrType || curRRHeader.Class != rrClass || curRRHeader.Name != rrName {
- // Mismatch between the records, so this is not a valid rrset for
- //signing/verifying
- return false
- }
- }
- return true
- }
- // Fqdn return the fully qualified domain name from s.
- // If s is already fully qualified, it behaves as the identity function.
- func Fqdn(s string) string {
- if IsFqdn(s) {
- return s
- }
- return s + "."
- }
- // Copied from the official Go code.
- // ReverseAddr returns the in-addr.arpa. or ip6.arpa. hostname of the IP
- // address suitable for reverse DNS (PTR) record lookups or an error if it fails
- // to parse the IP address.
- func ReverseAddr(addr string) (arpa string, err error) {
- ip := net.ParseIP(addr)
- if ip == nil {
- return "", &Error{err: "unrecognized address: " + addr}
- }
- if ip.To4() != nil {
- return strconv.Itoa(int(ip[15])) + "." + strconv.Itoa(int(ip[14])) + "." + strconv.Itoa(int(ip[13])) + "." +
- strconv.Itoa(int(ip[12])) + ".in-addr.arpa.", nil
- }
- // Must be IPv6
- buf := make([]byte, 0, len(ip)*4+len("ip6.arpa."))
- // Add it, in reverse, to the buffer
- for i := len(ip) - 1; i >= 0; i-- {
- v := ip[i]
- buf = append(buf, hexDigit[v&0xF])
- buf = append(buf, '.')
- buf = append(buf, hexDigit[v>>4])
- buf = append(buf, '.')
- }
- // Append "ip6.arpa." and return (buf already has the final .)
- buf = append(buf, "ip6.arpa."...)
- return string(buf), nil
- }
- // String returns the string representation for the type t.
- func (t Type) String() string {
- if t1, ok := TypeToString[uint16(t)]; ok {
- return t1
- }
- return "TYPE" + strconv.Itoa(int(t))
- }
- // String returns the string representation for the class c.
- func (c Class) String() string {
- if c1, ok := ClassToString[uint16(c)]; ok {
- return c1
- }
- return "CLASS" + strconv.Itoa(int(c))
- }
- // String returns the string representation for the name n.
- func (n Name) String() string {
- return sprintName(string(n))
- }
|