mirror of
https://github.com/daeuniverse/dae.git
synced 2025-01-05 13:08:57 +07:00
189 lines
4.9 KiB
Go
189 lines
4.9 KiB
Go
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
* Copyright (c) 2022-2024, daeuniverse Organization <dae@v2raya.org>
|
|
*/
|
|
|
|
package dns
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"net/url"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/daeuniverse/dae/common/consts"
|
|
"github.com/daeuniverse/dae/common/netutils"
|
|
"github.com/daeuniverse/outbound/protocol/direct"
|
|
)
|
|
|
|
var (
|
|
ErrFormat = fmt.Errorf("format error")
|
|
)
|
|
|
|
type UpstreamScheme string
|
|
|
|
const (
|
|
UpstreamScheme_TCP UpstreamScheme = "tcp"
|
|
UpstreamScheme_UDP UpstreamScheme = "udp"
|
|
UpstreamScheme_TCP_UDP UpstreamScheme = "tcp+udp"
|
|
upstreamScheme_TCP_UDP_Alias UpstreamScheme = "udp+tcp"
|
|
UpstreamScheme_TLS UpstreamScheme = "tls"
|
|
UpstreamScheme_QUIC UpstreamScheme = "quic"
|
|
UpstreamScheme_HTTPS UpstreamScheme = "https"
|
|
upstreamScheme_H3_Alias UpstreamScheme = "http3"
|
|
UpstreamScheme_H3 UpstreamScheme = "h3"
|
|
)
|
|
|
|
func (s UpstreamScheme) ContainsTcp() bool {
|
|
switch s {
|
|
case UpstreamScheme_TCP,
|
|
UpstreamScheme_TCP_UDP:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func ParseRawUpstream(raw *url.URL) (scheme UpstreamScheme, hostname string, port uint16, path string, err error) {
|
|
var __port string
|
|
var __path string
|
|
switch scheme = UpstreamScheme(raw.Scheme); scheme {
|
|
case upstreamScheme_TCP_UDP_Alias:
|
|
scheme = UpstreamScheme_TCP_UDP
|
|
fallthrough
|
|
case UpstreamScheme_TCP, UpstreamScheme_UDP, UpstreamScheme_TCP_UDP:
|
|
__port = raw.Port()
|
|
if __port == "" {
|
|
__port = "53"
|
|
}
|
|
case upstreamScheme_H3_Alias:
|
|
scheme = UpstreamScheme_H3
|
|
fallthrough
|
|
case UpstreamScheme_HTTPS, UpstreamScheme_H3:
|
|
__port = raw.Port()
|
|
if __port == "" {
|
|
__port = "443"
|
|
}
|
|
__path = raw.Path
|
|
if __path == "" {
|
|
__path = "/dns-query"
|
|
}
|
|
case UpstreamScheme_QUIC, UpstreamScheme_TLS:
|
|
__port = raw.Port()
|
|
if __port == "" {
|
|
__port = "853"
|
|
}
|
|
default:
|
|
return "", "", 0, "", fmt.Errorf("unexpected scheme: %v", raw.Scheme)
|
|
}
|
|
_port, err := strconv.ParseUint(__port, 10, 16)
|
|
if err != nil {
|
|
return "", "", 0, "", fmt.Errorf("failed to parse dns_upstream port: %v", err)
|
|
}
|
|
port = uint16(_port)
|
|
hostname = raw.Hostname()
|
|
return scheme, hostname, port, __path, nil
|
|
}
|
|
|
|
type Upstream struct {
|
|
Scheme UpstreamScheme
|
|
Hostname string
|
|
Port uint16
|
|
Path string
|
|
*netutils.Ip46
|
|
}
|
|
|
|
func NewUpstream(ctx context.Context, upstream *url.URL, resolverNetwork string) (up *Upstream, err error) {
|
|
scheme, hostname, port, path, err := ParseRawUpstream(upstream)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("%w: %v", ErrFormat, err)
|
|
}
|
|
|
|
systemDns, err := netutils.SystemDns()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
_ = netutils.TryUpdateSystemDnsElapse(time.Second)
|
|
}
|
|
}()
|
|
|
|
ip46, err := netutils.ResolveIp46(ctx, direct.SymmetricDirect, systemDns, hostname, resolverNetwork, 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 %v has no record", upstream.String())
|
|
}
|
|
|
|
return &Upstream{
|
|
Scheme: scheme,
|
|
Hostname: hostname,
|
|
Port: port,
|
|
Path: path,
|
|
Ip46: ip46,
|
|
}, nil
|
|
}
|
|
|
|
func (u *Upstream) 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 UpstreamScheme_TCP, UpstreamScheme_HTTPS, UpstreamScheme_TLS:
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_TCP}
|
|
case UpstreamScheme_UDP, UpstreamScheme_QUIC, UpstreamScheme_H3:
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_UDP}
|
|
case UpstreamScheme_TCP_UDP:
|
|
// UDP first.
|
|
l4protos = []consts.L4ProtoStr{consts.L4ProtoStr_UDP, consts.L4ProtoStr_TCP}
|
|
}
|
|
return ipversions, l4protos
|
|
}
|
|
|
|
func (u *Upstream) String() string {
|
|
return string(u.Scheme) + "://" + net.JoinHostPort(u.Hostname, strconv.Itoa(int(u.Port))) + u.Path
|
|
}
|
|
|
|
type UpstreamResolver struct {
|
|
Raw *url.URL
|
|
Network string
|
|
// FinishInitCallback may be invoked again if err is not nil
|
|
FinishInitCallback func(raw *url.URL, upstream *Upstream) (err error)
|
|
mu sync.Mutex
|
|
upstream *Upstream
|
|
init bool
|
|
}
|
|
|
|
func (u *UpstreamResolver) GetUpstream() (_ *Upstream, 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.upstream, err = NewUpstream(ctx, u.Raw, u.Network); err != nil {
|
|
return nil, fmt.Errorf("failed to init dns upstream: %w", err)
|
|
}
|
|
}
|
|
return u.upstream, nil
|
|
}
|