* plugin/edns: remove truncating of question section on bad EDNS version EDNS requests of "Unknown Version" removed the query section altogether. Not sure why since this is not require (see [link](https://kb.isc.org/docs/edns-compatibility-dig-queries) This cause issues with DNS solutions that uses this information (initial queried name, type and class) in order to route the response to the right client (e.g. PDNS). The change here is to keep the inital query section as is. Signed-off-by: Ben Kaplan <ben.kaplan@redis.com> * adding tests for edns0 version check Signed-off-by: Ben Kaplan <ben.kaplan@redis.com> * adding tests for non-edns0 version check Signed-off-by: Ben Kaplan <ben.kaplan@redis.com> Signed-off-by: Ben Kaplan <ben.kaplan@redis.com>
71 lines
1.7 KiB
Go
71 lines
1.7 KiB
Go
// Package edns provides function useful for adding/inspecting OPT records to/in messages.
|
|
package edns
|
|
|
|
import (
|
|
"errors"
|
|
"sync"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
var sup = &supported{m: make(map[uint16]struct{})}
|
|
|
|
type supported struct {
|
|
m map[uint16]struct{}
|
|
sync.RWMutex
|
|
}
|
|
|
|
// SetSupportedOption adds a new supported option the set of EDNS0 options that we support. Plugins typically call
|
|
// this in their setup code to signal support for a new option.
|
|
// By default we support:
|
|
// dns.EDNS0NSID, dns.EDNS0EXPIRE, dns.EDNS0COOKIE, dns.EDNS0TCPKEEPALIVE, dns.EDNS0PADDING. These
|
|
// values are not in this map and checked directly in the server.
|
|
func SetSupportedOption(option uint16) {
|
|
sup.Lock()
|
|
sup.m[option] = struct{}{}
|
|
sup.Unlock()
|
|
}
|
|
|
|
// SupportedOption returns true if the option code is supported as an extra EDNS0 option.
|
|
func SupportedOption(option uint16) bool {
|
|
sup.RLock()
|
|
_, ok := sup.m[option]
|
|
sup.RUnlock()
|
|
return ok
|
|
}
|
|
|
|
// Version checks the EDNS version in the request. If error
|
|
// is nil everything is OK and we can invoke the plugin. If non-nil, the
|
|
// returned Msg is valid to be returned to the client (and should).
|
|
func Version(req *dns.Msg) (*dns.Msg, error) {
|
|
opt := req.IsEdns0()
|
|
if opt == nil {
|
|
return nil, nil
|
|
}
|
|
if opt.Version() == 0 {
|
|
return nil, nil
|
|
}
|
|
m := new(dns.Msg)
|
|
m.SetReply(req)
|
|
|
|
o := new(dns.OPT)
|
|
o.Hdr.Name = "."
|
|
o.Hdr.Rrtype = dns.TypeOPT
|
|
o.SetVersion(0)
|
|
m.Rcode = dns.RcodeBadVers
|
|
o.SetExtendedRcode(dns.RcodeBadVers)
|
|
m.Extra = []dns.RR{o}
|
|
|
|
return m, errors.New("EDNS0 BADVERS")
|
|
}
|
|
|
|
// Size returns a normalized size based on proto.
|
|
func Size(proto string, size uint16) uint16 {
|
|
if proto == "tcp" {
|
|
return dns.MaxMsgSize
|
|
}
|
|
if size < dns.MinMsgSize {
|
|
return dns.MinMsgSize
|
|
}
|
|
return size
|
|
}
|