mirror of
https://github.com/daeuniverse/dae.git
synced 2025-01-19 00:38:18 +07:00
154 lines
3.8 KiB
Go
154 lines
3.8 KiB
Go
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
* Copyright (c) since 2023, mzz2017 <mzz@tuta.io>
|
|
*/
|
|
|
|
package control
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"github.com/v2rayA/dae/common"
|
|
"github.com/v2rayA/dae/common/consts"
|
|
"github.com/v2rayA/dae/common/netutils"
|
|
"github.com/v2rayA/dae/component/outbound/dialer"
|
|
"net/url"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type DnsUpstreamScheme string
|
|
|
|
const (
|
|
DnsUpstreamScheme_TCP DnsUpstreamScheme = "tcp"
|
|
DnsUpstreamScheme_UDP DnsUpstreamScheme = "udp"
|
|
DnsUpstreamScheme_TCP_UDP DnsUpstreamScheme = "tcp+udp"
|
|
)
|
|
|
|
func (s DnsUpstreamScheme) ContainsTcp() bool {
|
|
switch s {
|
|
case DnsUpstreamScheme_TCP,
|
|
DnsUpstreamScheme_TCP_UDP:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
type DnsUpstream struct {
|
|
Scheme DnsUpstreamScheme
|
|
Hostname string
|
|
Port uint16
|
|
*netutils.Ip46
|
|
}
|
|
|
|
func ParseDnsUpstream(dnsUpstream *url.URL) (scheme DnsUpstreamScheme, hostname string, port uint16, err error) {
|
|
var __port string
|
|
switch scheme = DnsUpstreamScheme(dnsUpstream.Scheme); scheme {
|
|
case DnsUpstreamScheme_TCP, DnsUpstreamScheme_UDP, DnsUpstreamScheme_TCP_UDP:
|
|
__port = dnsUpstream.Port()
|
|
if __port == "" {
|
|
__port = "53"
|
|
}
|
|
default:
|
|
return "", "", 0, fmt.Errorf("unexpected dns_upstream format")
|
|
}
|
|
_port, err := strconv.ParseUint(dnsUpstream.Port(), 10, 16)
|
|
port = uint16(_port)
|
|
if err != nil {
|
|
return "", "", 0, fmt.Errorf("parse dns_upstream port: %v", err)
|
|
}
|
|
hostname = dnsUpstream.Hostname()
|
|
return scheme, hostname, port, nil
|
|
}
|
|
|
|
func ResolveDnsUpstream(ctx context.Context, dnsUpstream *url.URL) (up *DnsUpstream, err error) {
|
|
scheme, hostname, port, err := ParseDnsUpstream(dnsUpstream)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
systemDns, err := netutils.SystemDns()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
_ = netutils.TryUpdateSystemDns1s()
|
|
}
|
|
}()
|
|
|
|
ip46, err := netutils.ParseIp46(ctx, dialer.SymmetricDirect, systemDns, hostname, false)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to resolve dns_upstream: %w", err)
|
|
}
|
|
if !ip46.Ip4.IsValid() && !ip46.Ip6.IsValid() {
|
|
return nil, fmt.Errorf("dns_upstream has no record")
|
|
}
|
|
|
|
return &DnsUpstream{
|
|
Scheme: scheme,
|
|
Hostname: hostname,
|
|
Port: port,
|
|
Ip46: ip46,
|
|
}, nil
|
|
}
|
|
|
|
func (u *DnsUpstream) SupportedNetworks() (ipversions []consts.IpVersionStr, l4protos []consts.L4ProtoStr) {
|
|
if u.Ip4.IsValid() && u.Ip6.IsValid() {
|
|
ipversions = []consts.IpVersionStr{consts.IpVersionStr_4, consts.IpVersionStr_6}
|
|
} else {
|
|
if u.Ip4.IsValid() {
|
|
ipversions = []consts.IpVersionStr{consts.IpVersionStr_4}
|
|
} else {
|
|
ipversions = []consts.IpVersionStr{consts.IpVersionStr_6}
|
|
}
|
|
}
|
|
switch u.Scheme {
|
|
case DnsUpstreamScheme_TCP:
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_TCP}
|
|
case DnsUpstreamScheme_UDP:
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_UDP}
|
|
case DnsUpstreamScheme_TCP_UDP:
|
|
// UDP first.
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_UDP, consts.L4ProtoStr_TCP}
|
|
}
|
|
return ipversions, l4protos
|
|
}
|
|
|
|
type DnsUpstreamRaw struct {
|
|
Raw common.UrlOrEmpty
|
|
// FinishInitCallback may be invoked again if err is not nil
|
|
FinishInitCallback func(raw common.UrlOrEmpty, upstream *DnsUpstream) (err error)
|
|
mu sync.Mutex
|
|
upstream *DnsUpstream
|
|
init bool
|
|
}
|
|
|
|
func (u *DnsUpstreamRaw) GetUpstream() (_ *DnsUpstream, err error) {
|
|
u.mu.Lock()
|
|
defer u.mu.Unlock()
|
|
if !u.init {
|
|
defer func() {
|
|
if err == nil {
|
|
if err = u.FinishInitCallback(u.Raw, u.upstream); err != nil {
|
|
u.upstream = nil
|
|
return
|
|
}
|
|
u.init = true
|
|
}
|
|
}()
|
|
ctx, cancel := context.WithTimeout(context.TODO(), 10*time.Second)
|
|
defer cancel()
|
|
if !u.Raw.Empty {
|
|
if u.upstream, err = ResolveDnsUpstream(ctx, u.Raw.Url); err != nil {
|
|
return nil, fmt.Errorf("failed to init dns upstream: %v", err)
|
|
}
|
|
} else {
|
|
// Empty string. As-is.
|
|
}
|
|
}
|
|
return u.upstream, nil
|
|
}
|