feat: Refactor output format

This commit is contained in:
Karan Sharma 2020-12-24 21:55:20 +05:30
parent 539e89e1fe
commit 4e5b074987
11 changed files with 447 additions and 384 deletions

View file

@ -1,77 +1,92 @@
package resolvers
import (
"time"
"github.com/miekg/dns"
"github.com/sirupsen/logrus"
)
// ClassicResolver represents the config options for setting up a Resolver.
type ClassicResolver struct {
client *dns.Client
server string
client *dns.Client
server string
resolverOptions Options
}
// ClassicResolverOpts holds options for setting up a Classic resolver.
type ClassicResolverOpts struct {
IPv4Only bool
IPv6Only bool
Timeout time.Duration
UseTLS bool
UseTCP bool
}
// NewClassicResolver accepts a list of nameservers and configures a DNS resolver.
func NewClassicResolver(server string, opts ClassicResolverOpts) (Resolver, error) {
func NewClassicResolver(server string, classicOpts ClassicResolverOpts, resolverOpts Options) (Resolver, error) {
net := "udp"
client := &dns.Client{
Timeout: opts.Timeout,
Timeout: resolverOpts.Timeout,
Net: "udp",
}
if opts.UseTCP {
if classicOpts.UseTCP {
net = "tcp"
}
if opts.IPv4Only {
if classicOpts.IPv4Only {
net = net + "4"
}
if opts.IPv6Only {
if classicOpts.IPv6Only {
net = net + "6"
}
if opts.UseTLS {
if classicOpts.UseTLS {
net = net + "-tls"
}
client.Net = net
return &ClassicResolver{
client: client,
server: server,
client: client,
server: server,
resolverOptions: resolverOpts,
}, 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 (r *ClassicResolver) Lookup(questions []dns.Question) ([]Response, error) {
// Lookup takes a dns.Question and sends them to DNS Server.
// It parses the Response from the server in a custom output format.
func (r *ClassicResolver) Lookup(question dns.Question) (Response, error) {
var (
messages = prepareMessages(questions)
responses []Response
rsp Response
messages = prepareMessages(question, r.resolverOptions.Ndots, r.resolverOptions.SearchList)
)
for _, msg := range messages {
r.resolverOptions.Logger.WithFields(logrus.Fields{
"domain": msg.Question[0].Name,
"ndots": r.resolverOptions.Ndots,
"nameserver": r.server,
}).Debug("Attempting to resolve")
in, rtt, err := r.client.Exchange(&msg, r.server)
if err != nil {
return nil, err
return rsp, err
}
rsp := Response{
Message: *in,
RTT: rtt,
Nameserver: r.server,
// pack questions in output.
for _, q := range msg.Question {
ques := Question{
Name: q.Name,
Class: dns.ClassToString[q.Qclass],
Type: dns.TypeToString[q.Qtype],
}
rsp.Questions = append(rsp.Questions, ques)
}
// get the authorities and answers.
output := parseMessage(in, rtt, r.server)
rsp.Authorities = output.Authorities
rsp.Answers = output.Answers
if len(output.Answers) > 0 {
// stop iterating the searchlist.
break
}
responses = append(responses, rsp)
}
return responses, nil
return rsp, nil
}