2024-07-21 07:57:47 +00:00
|
|
|
package dns
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
2024-08-06 08:48:18 +00:00
|
|
|
"net"
|
2024-07-21 07:57:47 +00:00
|
|
|
"regexp"
|
2024-07-31 11:47:19 +00:00
|
|
|
"strconv"
|
2024-08-10 07:15:22 +00:00
|
|
|
"time"
|
2024-07-21 07:57:47 +00:00
|
|
|
|
|
|
|
"github.com/miekg/dns"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"github.com/xvzc/SpoofDPI/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
type DnsResolver struct {
|
|
|
|
host string
|
|
|
|
port string
|
|
|
|
enableDoh bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewResolver(config *util.Config) *DnsResolver {
|
|
|
|
return &DnsResolver{
|
|
|
|
host: *config.DnsAddr,
|
2024-07-21 08:12:16 +00:00
|
|
|
port: strconv.Itoa(*config.DnsPort),
|
2024-07-21 07:57:47 +00:00
|
|
|
enableDoh: *config.EnableDoh,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-06 08:48:18 +00:00
|
|
|
func (d *DnsResolver) Lookup(domain string, useSystemDns bool) (string, error) {
|
2024-07-21 07:57:47 +00:00
|
|
|
ipRegex := "^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$"
|
|
|
|
|
|
|
|
if r, _ := regexp.MatchString(ipRegex, domain); r {
|
|
|
|
return domain, nil
|
|
|
|
}
|
|
|
|
|
2024-08-06 08:48:18 +00:00
|
|
|
if useSystemDns {
|
2024-08-10 06:34:37 +00:00
|
|
|
log.Debug("[DNS] ", domain, " resolving with system dns")
|
2024-08-06 08:48:18 +00:00
|
|
|
return systemLookup(domain)
|
|
|
|
}
|
|
|
|
|
2024-07-21 07:57:47 +00:00
|
|
|
if d.enableDoh {
|
2024-08-10 06:34:37 +00:00
|
|
|
log.Debug("[DNS] ", domain, " resolving with dns over https")
|
2024-07-21 07:57:47 +00:00
|
|
|
return dohLookup(domain)
|
|
|
|
}
|
|
|
|
|
2024-08-10 06:34:37 +00:00
|
|
|
log.Debug("[DNS] ", domain, " resolving with custom dns")
|
|
|
|
return customLookup(d.host, d.port, domain)
|
2024-08-06 08:48:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func customLookup(host string, port string, domain string) (string, error) {
|
|
|
|
|
|
|
|
dnsServer := host + ":" + port
|
2024-07-21 07:57:47 +00:00
|
|
|
|
|
|
|
msg := new(dns.Msg)
|
|
|
|
msg.SetQuestion(dns.Fqdn(domain), dns.TypeA)
|
|
|
|
|
|
|
|
c := new(dns.Client)
|
|
|
|
|
|
|
|
response, _, err := c.Exchange(msg, dnsServer)
|
|
|
|
if err != nil {
|
2024-08-12 22:37:20 +00:00
|
|
|
return "", errors.New("could not resolve the domain(custom)")
|
2024-07-21 07:57:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, answer := range response.Answer {
|
|
|
|
if record, ok := answer.(*dns.A); ok {
|
|
|
|
return record.A.String(), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-06 08:48:18 +00:00
|
|
|
return "", errors.New("no record found(custom)")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func systemLookup(domain string) (string, error) {
|
|
|
|
systemResolver := net.Resolver{PreferGo: true}
|
|
|
|
ips, err := systemResolver.LookupIPAddr(context.Background(), domain)
|
|
|
|
if err != nil {
|
2024-08-12 22:37:20 +00:00
|
|
|
return "", errors.New("could not resolve the domain(system)")
|
2024-08-06 08:48:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, ip := range ips {
|
|
|
|
return ip.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", errors.New("no record found(system)")
|
2024-07-21 07:57:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func dohLookup(domain string) (string, error) {
|
2024-08-10 07:15:22 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
|
|
|
|
defer cancel()
|
2024-08-12 15:33:31 +00:00
|
|
|
|
2024-08-12 22:37:20 +00:00
|
|
|
client := GetDOHClient(*util.GetConfig().DnsAddr)
|
|
|
|
|
|
|
|
msg := new(dns.Msg)
|
|
|
|
msg.SetQuestion(dns.Fqdn(domain), dns.TypeA)
|
|
|
|
|
|
|
|
response, err := client.Exchange(ctx, domain, msg)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.New("could not resolve the domain(doh)")
|
2024-07-21 07:57:47 +00:00
|
|
|
}
|
|
|
|
|
2024-08-12 22:37:20 +00:00
|
|
|
for _, answer := range response.Answer {
|
|
|
|
if record, ok := answer.(*dns.A); ok {
|
|
|
|
return record.A.String(), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "", errors.New("no record found(system)")
|
2024-07-21 07:57:47 +00:00
|
|
|
}
|