mirror of
https://github.com/xvzc/SpoofDPI.git
synced 2025-01-03 04:50:11 +00:00
116 lines
2.3 KiB
Go
116 lines
2.3 KiB
Go
package client
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/xvzc/SpoofDPI/dns/addrselect"
|
|
)
|
|
|
|
type Resolver interface {
|
|
Resolve(ctx context.Context, host string, qTypes []uint16) ([]net.IPAddr, error)
|
|
String() string
|
|
}
|
|
|
|
type exchangeFunc = func(ctx context.Context, msg *dns.Msg) (*dns.Msg, error)
|
|
|
|
func recordTypeIDToName(id uint16) string {
|
|
switch id {
|
|
case 1:
|
|
return "A"
|
|
case 28:
|
|
return "AAAA"
|
|
}
|
|
return strconv.FormatUint(uint64(id), 10)
|
|
}
|
|
|
|
func parseAddrsFromMsg(msg *dns.Msg) []net.IPAddr {
|
|
var addrs []net.IPAddr
|
|
|
|
for _, record := range msg.Answer {
|
|
switch ipRecord := record.(type) {
|
|
case *dns.A:
|
|
addrs = append(addrs, net.IPAddr{IP: ipRecord.A})
|
|
case *dns.AAAA:
|
|
addrs = append(addrs, net.IPAddr{IP: ipRecord.AAAA})
|
|
}
|
|
}
|
|
return addrs
|
|
}
|
|
|
|
func sortAddrs(addrs []net.IPAddr) {
|
|
addrselect.SortByRFC6724(addrs)
|
|
}
|
|
|
|
func lookup(ctx context.Context, host string, queryTypes []uint16, sendMsg exchangeFunc) <-chan *DNSResult {
|
|
var wg sync.WaitGroup
|
|
resCh := make(chan *DNSResult)
|
|
|
|
lookup := func(qType uint16) {
|
|
defer wg.Done()
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case resCh <- query(ctx, host, qType, sendMsg):
|
|
}
|
|
}
|
|
|
|
for _, queryType := range queryTypes {
|
|
wg.Add(1)
|
|
go lookup(queryType)
|
|
}
|
|
|
|
go func() {
|
|
wg.Wait()
|
|
close(resCh)
|
|
}()
|
|
return resCh
|
|
}
|
|
|
|
func query(ctx context.Context, host string, queryType uint16, exchange exchangeFunc) *DNSResult {
|
|
msg := newMsg(host, queryType)
|
|
resp, err := exchange(ctx, msg)
|
|
if err != nil {
|
|
queryName := recordTypeIDToName(queryType)
|
|
err = fmt.Errorf("resolving %s, query type %s: %w", host, queryName, err)
|
|
return &DNSResult{err: err}
|
|
}
|
|
return &DNSResult{msg: resp}
|
|
}
|
|
|
|
func newMsg(host string, qType uint16) *dns.Msg {
|
|
msg := new(dns.Msg)
|
|
msg.SetQuestion(dns.Fqdn(host), qType)
|
|
return msg
|
|
}
|
|
|
|
func processResults(ctx context.Context, resCh <-chan *DNSResult) ([]net.IPAddr, error) {
|
|
var errs []error
|
|
var addrs []net.IPAddr
|
|
|
|
for result := range resCh {
|
|
if result.err != nil {
|
|
errs = append(errs, result.err)
|
|
continue
|
|
}
|
|
resultAddrs := parseAddrsFromMsg(result.msg)
|
|
addrs = append(addrs, resultAddrs...)
|
|
}
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, errors.New("cancelled")
|
|
default:
|
|
if len(addrs) == 0 {
|
|
return addrs, errors.Join(errs...)
|
|
}
|
|
}
|
|
|
|
sortAddrs(addrs)
|
|
return addrs, nil
|
|
}
|