mirror of
https://github.com/daeuniverse/dae.git
synced 2025-01-22 02:07:50 +07:00
283 lines
6.3 KiB
Go
283 lines
6.3 KiB
Go
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
|
|
*/
|
|
|
|
package netutils
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"github.com/mzz2017/softwind/netproxy"
|
|
"github.com/mzz2017/softwind/pkg/fastrand"
|
|
"github.com/mzz2017/softwind/pool"
|
|
"github.com/sirupsen/logrus"
|
|
"golang.org/x/net/dns/dnsmessage"
|
|
"io"
|
|
"math"
|
|
"net/netip"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
var (
|
|
systemDnsMu sync.Mutex
|
|
systemDns netip.AddrPort
|
|
systemDnsNextUpdateAfter time.Time
|
|
|
|
BadDnsAnsError = fmt.Errorf("bad dns answer")
|
|
|
|
BootstrapDns = netip.MustParseAddrPort("208.67.222.222:5353")
|
|
)
|
|
|
|
func TryUpdateSystemDns() (err error) {
|
|
systemDnsMu.Lock()
|
|
err = tryUpdateSystemDns()
|
|
systemDnsMu.Unlock()
|
|
return err
|
|
}
|
|
|
|
// TryUpdateSystemDnsElapse will update system DNS if duration has elapsed since the last TryUpdateSystemDns1s call.
|
|
func TryUpdateSystemDnsElapse(k time.Duration) (err error) {
|
|
systemDnsMu.Lock()
|
|
defer systemDnsMu.Unlock()
|
|
return tryUpdateSystemDnsElapse(k)
|
|
}
|
|
func tryUpdateSystemDnsElapse(k time.Duration) (err error) {
|
|
if time.Now().Before(systemDnsNextUpdateAfter) {
|
|
return fmt.Errorf("update too quickly")
|
|
}
|
|
err = tryUpdateSystemDns()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
systemDnsNextUpdateAfter = time.Now().Add(k)
|
|
return nil
|
|
}
|
|
|
|
func tryUpdateSystemDns() (err error) {
|
|
dnsConf := dnsReadConfig("/etc/resolv.conf")
|
|
if len(dnsConf.servers) == 0 {
|
|
err = fmt.Errorf("no valid dns server in /etc/resolv.conf")
|
|
return err
|
|
}
|
|
systemDns = netip.AddrPort{}
|
|
for _, s := range dnsConf.servers {
|
|
ipPort := netip.MustParseAddrPort(s)
|
|
if !ipPort.Addr().IsLoopback() {
|
|
systemDns = ipPort
|
|
break
|
|
}
|
|
}
|
|
if !systemDns.IsValid() {
|
|
systemDns = BootstrapDns
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func SystemDns() (dns netip.AddrPort, err error) {
|
|
systemDnsMu.Lock()
|
|
defer systemDnsMu.Unlock()
|
|
if !systemDns.IsValid() {
|
|
if err = tryUpdateSystemDns(); err != nil {
|
|
return netip.AddrPort{}, err
|
|
}
|
|
}
|
|
// To avoid environment changing.
|
|
_ = tryUpdateSystemDnsElapse(5 * time.Second)
|
|
return systemDns, nil
|
|
}
|
|
|
|
func ResolveNetip(ctx context.Context, d netproxy.Dialer, dns netip.AddrPort, host string, typ dnsmessage.Type, tcp bool) (addrs []netip.Addr, err error) {
|
|
resources, err := resolve(ctx, d, dns, host, typ, tcp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, ans := range resources {
|
|
if ans.Header.Type != typ {
|
|
continue
|
|
}
|
|
switch typ {
|
|
case dnsmessage.TypeA:
|
|
a, ok := ans.Body.(*dnsmessage.AResource)
|
|
if !ok {
|
|
return nil, BadDnsAnsError
|
|
}
|
|
addrs = append(addrs, netip.AddrFrom4(a.A))
|
|
case dnsmessage.TypeAAAA:
|
|
a, ok := ans.Body.(*dnsmessage.AAAAResource)
|
|
if !ok {
|
|
return nil, BadDnsAnsError
|
|
}
|
|
addrs = append(addrs, netip.AddrFrom16(a.AAAA))
|
|
}
|
|
}
|
|
return addrs, nil
|
|
}
|
|
|
|
func ResolveNS(ctx context.Context, d netproxy.Dialer, dns netip.AddrPort, host string, tcp bool) (records []string, err error) {
|
|
typ := dnsmessage.TypeNS
|
|
resources, err := resolve(ctx, d, dns, host, typ, tcp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
logrus.Println(host, len(resources))
|
|
for _, ans := range resources {
|
|
if ans.Header.Type != typ {
|
|
logrus.Println(host, ans.Header.Type)
|
|
continue
|
|
}
|
|
ns, ok := ans.Body.(*dnsmessage.NSResource)
|
|
if !ok {
|
|
return nil, BadDnsAnsError
|
|
}
|
|
records = append(records, ns.NS.String())
|
|
}
|
|
return records, nil
|
|
}
|
|
|
|
func resolve(ctx context.Context, d netproxy.Dialer, dns netip.AddrPort, host string, typ dnsmessage.Type, tcp bool) (ans []dnsmessage.Resource, err error) {
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
switch typ {
|
|
case dnsmessage.TypeA, dnsmessage.TypeAAAA:
|
|
if addr, err := netip.ParseAddr(host); err == nil {
|
|
if (addr.Is4() || addr.Is4In6()) && typ == dnsmessage.TypeA {
|
|
return []dnsmessage.Resource{
|
|
{
|
|
Header: dnsmessage.ResourceHeader{
|
|
Type: typ,
|
|
},
|
|
Body: &dnsmessage.AResource{A: addr.As4()},
|
|
},
|
|
}, nil
|
|
} else if addr.Is6() && typ == dnsmessage.TypeAAAA {
|
|
return []dnsmessage.Resource{
|
|
{
|
|
Header: dnsmessage.ResourceHeader{
|
|
Type: typ,
|
|
},
|
|
Body: &dnsmessage.AAAAResource{AAAA: addr.As16()},
|
|
},
|
|
}, nil
|
|
}
|
|
// MUST No record.
|
|
return nil, nil
|
|
}
|
|
default:
|
|
}
|
|
// Build DNS req.
|
|
builder := dnsmessage.NewBuilder(nil, dnsmessage.Header{
|
|
ID: uint16(fastrand.Intn(math.MaxUint16 + 1)),
|
|
Response: false,
|
|
OpCode: 0,
|
|
Truncated: false,
|
|
RecursionDesired: true,
|
|
Authoritative: false,
|
|
})
|
|
if err = builder.StartQuestions(); err != nil {
|
|
return nil, err
|
|
}
|
|
fqdn := host
|
|
if !strings.HasSuffix(fqdn, ".") {
|
|
fqdn += "."
|
|
}
|
|
if err = builder.Question(dnsmessage.Question{
|
|
Name: dnsmessage.MustNewName(fqdn),
|
|
Type: typ,
|
|
Class: dnsmessage.ClassINET,
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
b, err := builder.Finish()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if tcp {
|
|
// Put DNS request length
|
|
buf := pool.Get(2 + len(b))
|
|
defer pool.Put(buf)
|
|
binary.BigEndian.PutUint16(buf, uint16(len(b)))
|
|
copy(buf[2:], b)
|
|
b = buf
|
|
}
|
|
|
|
// Dial and write.
|
|
cd := &netproxy.ContextDialer{Dialer: d}
|
|
var c netproxy.Conn
|
|
if tcp {
|
|
c, err = cd.DialTcpContext(ctx, dns.String())
|
|
} else {
|
|
c, err = cd.DialUdpContext(ctx, dns.String())
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer c.Close()
|
|
_, err = c.Write(b)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ch := make(chan error, 2)
|
|
if !tcp {
|
|
go func() {
|
|
// Resend every 3 seconds for UDP.
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
time.Sleep(3 * time.Second)
|
|
}
|
|
_, err := c.Write(b)
|
|
if err != nil {
|
|
ch <- err
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
go func() {
|
|
buf := pool.Get(512)
|
|
defer pool.Put(buf)
|
|
if tcp {
|
|
// Read DNS response length
|
|
_, err := io.ReadFull(c, buf[:2])
|
|
if err != nil {
|
|
ch <- err
|
|
return
|
|
}
|
|
n := binary.BigEndian.Uint16(buf)
|
|
if n > 512 {
|
|
ch <- fmt.Errorf("too big dns resp")
|
|
return
|
|
}
|
|
buf = buf[:n]
|
|
}
|
|
n, err := c.Read(buf)
|
|
if err != nil {
|
|
ch <- err
|
|
return
|
|
}
|
|
// Resolve DNS response and extract A/AAAA record.
|
|
var msg dnsmessage.Message
|
|
if err = msg.Unpack(buf[:n]); err != nil {
|
|
ch <- err
|
|
return
|
|
}
|
|
ans = msg.Answers
|
|
ch <- nil
|
|
}()
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, fmt.Errorf("timeout")
|
|
case err = <-ch:
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return ans, nil
|
|
}
|
|
}
|