This methods returns an OPT record which can be used to create a new message with the same bufsize and Do bit as the original one.
155 lines
3.9 KiB
Go
155 lines
3.9 KiB
Go
package middleware
|
|
|
|
import (
|
|
"net"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/miekg/dns"
|
|
)
|
|
|
|
// This file contains the state nd functions available for use in the templates.
|
|
|
|
// State contains some connection state and is useful in middleware.
|
|
type State struct {
|
|
Root http.FileSystem // TODO(miek): needed?
|
|
Req *dns.Msg
|
|
W dns.ResponseWriter
|
|
}
|
|
|
|
// Now returns the current timestamp in the specified format.
|
|
func (s State) Now(format string) string { return time.Now().Format(format) }
|
|
|
|
// NowDate returns the current date/time that can be used in other time functions.
|
|
func (s State) NowDate() time.Time { return time.Now() }
|
|
|
|
// Header gets the heaser of the request in State.
|
|
func (s State) Header() *dns.RR_Header {
|
|
// TODO(miek)
|
|
return nil
|
|
}
|
|
|
|
// IP gets the (remote) IP address of the client making the request.
|
|
func (s State) IP() string {
|
|
ip, _, err := net.SplitHostPort(s.W.RemoteAddr().String())
|
|
if err != nil {
|
|
return s.W.RemoteAddr().String()
|
|
}
|
|
return ip
|
|
}
|
|
|
|
// Post gets the (remote) Port of the client making the request.
|
|
func (s State) Port() (string, error) {
|
|
_, port, err := net.SplitHostPort(s.W.RemoteAddr().String())
|
|
if err != nil {
|
|
return "0", err
|
|
}
|
|
return port, nil
|
|
}
|
|
|
|
// Proto gets the protocol used as the transport. This
|
|
// will be udp or tcp.
|
|
func (s State) Proto() string {
|
|
if _, ok := s.W.RemoteAddr().(*net.UDPAddr); ok {
|
|
return "udp"
|
|
}
|
|
if _, ok := s.W.RemoteAddr().(*net.TCPAddr); ok {
|
|
return "tcp"
|
|
}
|
|
return "udp"
|
|
}
|
|
|
|
// Family returns the family of the transport.
|
|
// 1 for IPv4 and 2 for IPv6.
|
|
func (s State) Family() int {
|
|
var a net.IP
|
|
ip := s.W.RemoteAddr()
|
|
if i, ok := ip.(*net.UDPAddr); ok {
|
|
a = i.IP
|
|
}
|
|
if i, ok := ip.(*net.TCPAddr); ok {
|
|
a = i.IP
|
|
}
|
|
|
|
if a.To4() != nil {
|
|
return 1
|
|
}
|
|
return 2
|
|
}
|
|
|
|
// Do returns if the request has the DO (DNSSEC OK) bit set.
|
|
func (s State) Do() bool {
|
|
if o := s.Req.IsEdns0(); o != nil {
|
|
return o.Do()
|
|
}
|
|
return false
|
|
}
|
|
|
|
// UDPSize returns if UDP buffer size advertised in the requests OPT record.
|
|
// Or when the request was over TCP, we return the maximum allowed size of 64K.
|
|
func (s State) Size() int {
|
|
if s.Proto() == "tcp" {
|
|
return dns.MaxMsgSize
|
|
}
|
|
if o := s.Req.IsEdns0(); o != nil {
|
|
s := o.UDPSize()
|
|
if s < dns.MinMsgSize {
|
|
s = dns.MinMsgSize
|
|
}
|
|
return int(s)
|
|
}
|
|
return dns.MinMsgSize
|
|
}
|
|
|
|
// SizeAndDo returns a ready made OPT record that the reflects the intent
|
|
// from the state. This can be added to upstream requests that will then
|
|
// hopefully return a message that is understandable by the original client.
|
|
func (s State) SizeAndDo() *dns.OPT {
|
|
size := s.Size()
|
|
Do := s.Do()
|
|
|
|
o := new(dns.OPT)
|
|
o.Hdr.Name = "."
|
|
o.Hdr.Rrtype = dns.TypeOPT
|
|
o.SetUDPSize(uint16(size))
|
|
if Do {
|
|
o.SetDo()
|
|
}
|
|
return o
|
|
}
|
|
|
|
// Type returns the type of the question as a string.
|
|
func (s State) Type() string { return dns.Type(s.Req.Question[0].Qtype).String() }
|
|
|
|
// QType returns the type of the question as a uint16.
|
|
func (s State) QType() uint16 { return s.Req.Question[0].Qtype }
|
|
|
|
// Name returns the name of the question in the request. Note
|
|
// this name will always have a closing dot and will be lower cased.
|
|
func (s State) Name() string { return strings.ToLower(dns.Name(s.Req.Question[0].Name).String()) }
|
|
|
|
// QName returns the name of the question in the request.
|
|
func (s State) QName() string { return dns.Name(s.Req.Question[0].Name).String() }
|
|
|
|
// Class returns the class of the question in the request.
|
|
func (s State) Class() string { return dns.Class(s.Req.Question[0].Qclass).String() }
|
|
|
|
// QClass returns the class of the question in the request.
|
|
func (s State) QClass() uint16 { return s.Req.Question[0].Qclass }
|
|
|
|
// ErrorMessage returns an error message suitable for sending
|
|
// back to the client.
|
|
func (s State) ErrorMessage(rcode int) *dns.Msg {
|
|
m := new(dns.Msg)
|
|
m.SetRcode(s.Req, rcode)
|
|
return m
|
|
}
|
|
|
|
// AnswerMessage returns an error message suitable for sending
|
|
// back to the client.
|
|
func (s State) AnswerMessage() *dns.Msg {
|
|
m := new(dns.Msg)
|
|
m.SetReply(s.Req)
|
|
return m
|
|
}
|