feat: Multiple resolvers continued
This commit is contained in:
parent
b46b64c1ae
commit
53e7f2e59c
11 changed files with 253 additions and 214 deletions
|
@ -2,7 +2,6 @@ package resolvers
|
|||
|
||||
import (
|
||||
"bytes"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"net/http"
|
||||
|
@ -14,34 +13,30 @@ import (
|
|||
|
||||
// DOHResolver represents the config options for setting up a DOH based resolver.
|
||||
type DOHResolver struct {
|
||||
client *http.Client
|
||||
servers []string
|
||||
client *http.Client
|
||||
server string
|
||||
}
|
||||
|
||||
type DOHResolverOpts struct {
|
||||
Timeout time.Duration
|
||||
}
|
||||
|
||||
// NewDOHResolver accepts a list of nameservers and configures a DOH based resolver.
|
||||
func NewDOHResolver(servers []string, opts DOHResolverOpts) (Resolver, error) {
|
||||
if len(servers) == 0 {
|
||||
return nil, errors.New(`no DOH server specified`)
|
||||
// NewDOHResolver accepts a nameserver address and configures a DOH based resolver.
|
||||
func NewDOHResolver(server string, opts DOHResolverOpts) (Resolver, error) {
|
||||
// do basic validation
|
||||
u, err := url.ParseRequestURI(server)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("%s is not a valid HTTPS nameserver", server)
|
||||
}
|
||||
for _, s := range servers {
|
||||
u, err := url.ParseRequestURI(s)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("%s is not a valid HTTPS nameserver", s)
|
||||
}
|
||||
if u.Scheme != "https" {
|
||||
return nil, fmt.Errorf("missing https in %s", s)
|
||||
}
|
||||
if u.Scheme != "https" {
|
||||
return nil, fmt.Errorf("missing https in %s", server)
|
||||
}
|
||||
httpClient := &http.Client{
|
||||
Timeout: opts.Timeout,
|
||||
}
|
||||
return &DOHResolver{
|
||||
client: httpClient,
|
||||
servers: servers,
|
||||
client: httpClient,
|
||||
server: server,
|
||||
}, nil
|
||||
}
|
||||
|
||||
|
@ -57,34 +52,32 @@ func (d *DOHResolver) Lookup(questions []dns.Question) ([]Response, error) {
|
|||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
for _, srv := range d.servers {
|
||||
now := time.Now()
|
||||
// Make an HTTP POST request to the DNS server with the DNS message as wire format bytes in the body.
|
||||
resp, err := d.client.Post(srv, "application/dns-message", bytes.NewBuffer(b))
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
return nil, fmt.Errorf("error from nameserver %s", resp.Status)
|
||||
}
|
||||
rtt := time.Since(now)
|
||||
// extract the binary response in DNS Message.
|
||||
body, err := ioutil.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
err = msg.Unpack(body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
rsp := Response{
|
||||
Message: msg,
|
||||
RTT: rtt,
|
||||
Nameserver: srv,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
now := time.Now()
|
||||
// Make an HTTP POST request to the DNS server with the DNS message as wire format bytes in the body.
|
||||
resp, err := d.client.Post(d.server, "application/dns-message", bytes.NewBuffer(b))
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
return nil, fmt.Errorf("error from nameserver %s", resp.Status)
|
||||
}
|
||||
rtt := time.Since(now)
|
||||
// extract the binary response in DNS Message.
|
||||
body, err := ioutil.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
err = msg.Unpack(body)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
rsp := Response{
|
||||
Message: msg,
|
||||
RTT: rtt,
|
||||
Nameserver: d.server,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
}
|
||||
return responses, nil
|
||||
}
|
||||
|
|
|
@ -1,21 +1,15 @@
|
|||
package resolvers
|
||||
|
||||
import (
|
||||
"net"
|
||||
"time"
|
||||
|
||||
"github.com/miekg/dns"
|
||||
)
|
||||
|
||||
const (
|
||||
// DefaultTLSPort specifies the default port for a DNS server connecting over TCP over TLS
|
||||
DefaultTLSPort = "853"
|
||||
)
|
||||
|
||||
// TCPResolver represents the config options for setting up a Resolver.
|
||||
type TCPResolver struct {
|
||||
client *dns.Client
|
||||
servers []string
|
||||
client *dns.Client
|
||||
server string
|
||||
}
|
||||
|
||||
// TCPResolverOpts represents the config options for setting up a TCPResolver.
|
||||
|
@ -26,31 +20,10 @@ type TCPResolverOpts struct {
|
|||
}
|
||||
|
||||
// NewTCPResolver accepts a list of nameservers and configures a DNS resolver.
|
||||
func NewTCPResolver(servers []string, opts TCPResolverOpts) (Resolver, error) {
|
||||
func NewTCPResolver(server string, opts TCPResolverOpts) (Resolver, error) {
|
||||
client := &dns.Client{
|
||||
Timeout: opts.Timeout,
|
||||
}
|
||||
var nameservers []string
|
||||
|
||||
// load list of nameservers to the config
|
||||
if len(servers) == 0 {
|
||||
ns, err := getDefaultServers()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
nameservers = ns
|
||||
} else {
|
||||
// load the list of servers that user specified.
|
||||
for _, srv := range servers {
|
||||
if i := net.ParseIP(srv); i != nil {
|
||||
// if no port specified in nameserver, append defaults.
|
||||
nameservers = append(nameservers, net.JoinHostPort(srv, DefaultTLSPort))
|
||||
} else {
|
||||
// use the port user specified.
|
||||
nameservers = append(nameservers, srv)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
client.Net = "tcp"
|
||||
if opts.IPv4Only {
|
||||
|
@ -60,8 +33,8 @@ func NewTCPResolver(servers []string, opts TCPResolverOpts) (Resolver, error) {
|
|||
client.Net = "tcp6"
|
||||
}
|
||||
return &TCPResolver{
|
||||
client: client,
|
||||
servers: nameservers,
|
||||
client: client,
|
||||
server: server,
|
||||
}, nil
|
||||
}
|
||||
|
||||
|
@ -75,19 +48,17 @@ func (r *TCPResolver) Lookup(questions []dns.Question) ([]Response, error) {
|
|||
)
|
||||
|
||||
for _, msg := range messages {
|
||||
for _, srv := range r.servers {
|
||||
in, rtt, err := r.client.Exchange(&msg, srv)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
msg.Answer = in.Answer
|
||||
rsp := Response{
|
||||
Message: msg,
|
||||
RTT: rtt,
|
||||
Nameserver: srv,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
in, rtt, err := r.client.Exchange(&msg, r.server)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
msg.Answer = in.Answer
|
||||
rsp := Response{
|
||||
Message: msg,
|
||||
RTT: rtt,
|
||||
Nameserver: r.server,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
}
|
||||
return responses, nil
|
||||
}
|
||||
|
|
|
@ -1,23 +1,15 @@
|
|||
package resolvers
|
||||
|
||||
import (
|
||||
"net"
|
||||
"time"
|
||||
|
||||
"github.com/miekg/dns"
|
||||
)
|
||||
|
||||
const (
|
||||
// DefaultUDPPort specifies the default port for a DNS server connecting over UDP
|
||||
DefaultUDPPort = "53"
|
||||
//DefaultResolvConfPath specifies path to default resolv config file on UNIX.
|
||||
DefaultResolvConfPath = "/etc/resolv.conf"
|
||||
)
|
||||
|
||||
// UDPResolver represents the config options for setting up a Resolver.
|
||||
type UDPResolver struct {
|
||||
client *dns.Client
|
||||
servers []string
|
||||
client *dns.Client
|
||||
server string
|
||||
}
|
||||
|
||||
// UDPResolverOpts holds options for setting up a Classic resolver.
|
||||
|
@ -28,31 +20,10 @@ type UDPResolverOpts struct {
|
|||
}
|
||||
|
||||
// NewUDPResolver accepts a list of nameservers and configures a DNS resolver.
|
||||
func NewUDPResolver(servers []string, opts UDPResolverOpts) (Resolver, error) {
|
||||
func NewUDPResolver(server string, opts UDPResolverOpts) (Resolver, error) {
|
||||
client := &dns.Client{
|
||||
Timeout: opts.Timeout,
|
||||
}
|
||||
var nameservers []string
|
||||
|
||||
// load list of nameservers to the config
|
||||
if len(servers) == 0 {
|
||||
ns, err := getDefaultServers()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
nameservers = ns
|
||||
} else {
|
||||
// load the list of servers that user specified.
|
||||
for _, srv := range servers {
|
||||
if i := net.ParseIP(srv); i != nil {
|
||||
// if no port specified in nameserver, append defaults.
|
||||
nameservers = append(nameservers, net.JoinHostPort(srv, DefaultUDPPort))
|
||||
} else {
|
||||
// use the port user specified.
|
||||
nameservers = append(nameservers, srv)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
client.Net = "udp"
|
||||
if opts.IPv4Only {
|
||||
|
@ -62,34 +33,31 @@ func NewUDPResolver(servers []string, opts UDPResolverOpts) (Resolver, error) {
|
|||
client.Net = "udp6"
|
||||
}
|
||||
return &UDPResolver{
|
||||
client: client,
|
||||
servers: nameservers,
|
||||
client: client,
|
||||
server: server,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// Lookup prepare a list of DNS messages to be sent to the server.
|
||||
// It's possible to send multiple question in one message
|
||||
// but some nameservers are not able to
|
||||
func (c *UDPResolver) Lookup(questions []dns.Question) ([]Response, error) {
|
||||
func (r *UDPResolver) Lookup(questions []dns.Question) ([]Response, error) {
|
||||
var (
|
||||
messages = prepareMessages(questions)
|
||||
responses []Response
|
||||
)
|
||||
|
||||
for _, msg := range messages {
|
||||
for _, srv := range c.servers {
|
||||
in, rtt, err := c.client.Exchange(&msg, srv)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
msg.Answer = in.Answer
|
||||
rsp := Response{
|
||||
Message: msg,
|
||||
RTT: rtt,
|
||||
Nameserver: srv,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
in, rtt, err := r.client.Exchange(&msg, r.server)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
rsp := Response{
|
||||
Message: *in,
|
||||
RTT: rtt,
|
||||
Nameserver: r.server,
|
||||
}
|
||||
responses = append(responses, rsp)
|
||||
}
|
||||
return responses, nil
|
||||
}
|
||||
|
|
|
@ -1,11 +1,6 @@
|
|||
package resolvers
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"net"
|
||||
"runtime"
|
||||
|
||||
"github.com/miekg/dns"
|
||||
)
|
||||
|
||||
|
@ -23,26 +18,3 @@ func prepareMessages(questions []dns.Question) []dns.Msg {
|
|||
}
|
||||
return messages
|
||||
}
|
||||
|
||||
func getDefaultServers() ([]string, error) {
|
||||
if runtime.GOOS == "windows" {
|
||||
// TODO: Add a method for reading system default nameserver in windows.
|
||||
return nil, errors.New(`unable to read default nameservers in this machine`)
|
||||
}
|
||||
// if no nameserver is provided, take it from `resolv.conf`
|
||||
cfg, err := dns.ClientConfigFromFile(DefaultResolvConfPath)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
servers := make([]string, 0, len(cfg.Servers))
|
||||
for _, s := range cfg.Servers {
|
||||
ip := net.ParseIP(s)
|
||||
// handle IPv6
|
||||
if ip != nil && ip.To4() != nil {
|
||||
servers = append(servers, fmt.Sprintf("%s:%s", s, cfg.Port))
|
||||
} else {
|
||||
servers = append(servers, fmt.Sprintf("[%s]:%s", s, cfg.Port))
|
||||
}
|
||||
}
|
||||
return servers, nil
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue