279 lines
7.5 KiB
Go
279 lines
7.5 KiB
Go
package resolver
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"math/rand"
|
|
"net"
|
|
"net/netip"
|
|
"time"
|
|
|
|
"github.com/Dreamacro/clash/common/nnip"
|
|
"github.com/Dreamacro/clash/component/trie"
|
|
)
|
|
|
|
var (
|
|
// DefaultResolver aim to resolve ip
|
|
DefaultResolver Resolver
|
|
|
|
// ProxyServerHostResolver resolve ip to proxies server host
|
|
ProxyServerHostResolver Resolver
|
|
|
|
// DisableIPv6 means don't resolve ipv6 host
|
|
// default value is true
|
|
DisableIPv6 = true
|
|
|
|
// DefaultHosts aim to resolve hosts
|
|
DefaultHosts = trie.New[netip.Addr]()
|
|
|
|
// DefaultDNSTimeout defined the default dns request timeout
|
|
DefaultDNSTimeout = time.Second * 5
|
|
)
|
|
|
|
var (
|
|
ErrIPNotFound = errors.New("couldn't find ip")
|
|
ErrIPVersion = errors.New("ip version error")
|
|
ErrIPv6Disabled = errors.New("ipv6 disabled")
|
|
)
|
|
|
|
type Resolver interface {
|
|
LookupIP(ctx context.Context, host string) (ips []netip.Addr, err error)
|
|
LookupIPv4(ctx context.Context, host string) (ips []netip.Addr, err error)
|
|
LookupIPv6(ctx context.Context, host string) (ips []netip.Addr, err error)
|
|
ResolveIP(ctx context.Context, host string) (ip netip.Addr, err error)
|
|
ResolveIPv4(ctx context.Context, host string) (ip netip.Addr, err error)
|
|
ResolveIPv6(ctx context.Context, host string) (ip netip.Addr, err error)
|
|
}
|
|
|
|
// ResolveIPv4 with a host, return ipv4
|
|
func ResolveIPv4(ctx context.Context, host string) (netip.Addr, error) {
|
|
return ResolveIPv4WithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
func ResolveIPv4WithResolver(ctx context.Context, host string, r Resolver) (netip.Addr, error) {
|
|
if ips, err := LookupIPv4WithResolver(ctx, host, r); err == nil {
|
|
return ips[rand.Intn(len(ips))], nil
|
|
} else {
|
|
return netip.Addr{}, err
|
|
}
|
|
}
|
|
|
|
// ResolveIPv6 with a host, return ipv6
|
|
func ResolveIPv6(ctx context.Context, host string) (netip.Addr, error) {
|
|
return ResolveIPv6WithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
func ResolveIPv6WithResolver(ctx context.Context, host string, r Resolver) (netip.Addr, error) {
|
|
if ips, err := LookupIPv6WithResolver(ctx, host, r); err == nil {
|
|
return ips[rand.Intn(len(ips))], nil
|
|
} else {
|
|
return netip.Addr{}, err
|
|
}
|
|
}
|
|
|
|
// ResolveIPWithResolver same as ResolveIP, but with a resolver
|
|
func ResolveIPWithResolver(ctx context.Context, host string, r Resolver) (netip.Addr, error) {
|
|
if ip, err := ResolveIPv4WithResolver(ctx, host, r); err == nil {
|
|
return ip, nil
|
|
} else {
|
|
return ResolveIPv6WithResolver(ctx, host, r)
|
|
}
|
|
}
|
|
|
|
// ResolveIP with a host, return ip
|
|
func ResolveIP(ctx context.Context, host string) (netip.Addr, error) {
|
|
return ResolveIPWithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
// ResolveIPv4ProxyServerHost proxies server host only
|
|
func ResolveIPv4ProxyServerHost(ctx context.Context, host string) (netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
if ip, err := ResolveIPv4WithResolver(ctx, host, ProxyServerHostResolver); err != nil {
|
|
return ResolveIPv4(ctx, host)
|
|
} else {
|
|
return ip, nil
|
|
}
|
|
}
|
|
return ResolveIPv4(ctx, host)
|
|
}
|
|
|
|
// ResolveIPv6ProxyServerHost proxies server host only
|
|
func ResolveIPv6ProxyServerHost(ctx context.Context, host string) (netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
if ip, err := ResolveIPv6WithResolver(ctx, host, ProxyServerHostResolver); err != nil {
|
|
return ResolveIPv6(ctx, host)
|
|
} else {
|
|
return ip, nil
|
|
}
|
|
}
|
|
return ResolveIPv6(ctx, host)
|
|
}
|
|
|
|
// ResolveProxyServerHost proxies server host only
|
|
func ResolveProxyServerHost(ctx context.Context, host string) (netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
if ip, err := ResolveIPWithResolver(ctx, host, ProxyServerHostResolver); err != nil {
|
|
return ResolveIP(ctx, host)
|
|
} else {
|
|
return ip, err
|
|
}
|
|
}
|
|
return ResolveIP(ctx, host)
|
|
}
|
|
|
|
func LookupIPv6WithResolver(ctx context.Context, host string, r Resolver) ([]netip.Addr, error) {
|
|
if DisableIPv6 {
|
|
return []netip.Addr{}, ErrIPv6Disabled
|
|
}
|
|
|
|
if node := DefaultHosts.Search(host); node != nil {
|
|
if ip := node.Data(); ip.Is6() {
|
|
return []netip.Addr{ip}, nil
|
|
}
|
|
}
|
|
|
|
ip, err := netip.ParseAddr(host)
|
|
if err == nil {
|
|
if ip.Is6() {
|
|
return []netip.Addr{ip}, nil
|
|
}
|
|
return []netip.Addr{}, ErrIPVersion
|
|
}
|
|
|
|
if r != nil {
|
|
return r.LookupIPv6(ctx, host)
|
|
}
|
|
|
|
if DefaultResolver == nil {
|
|
ipAddrs, err := net.DefaultResolver.LookupIP(ctx, "ip6", host)
|
|
if err != nil {
|
|
return []netip.Addr{}, err
|
|
} else if len(ipAddrs) == 0 {
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
|
|
addrs := make([]netip.Addr, 0, len(ipAddrs))
|
|
for _, ipAddr := range ipAddrs {
|
|
addrs = append(addrs, nnip.IpToAddr(ipAddr))
|
|
}
|
|
|
|
rand.Shuffle(len(addrs), func(i, j int) {
|
|
addrs[i], addrs[j] = addrs[j], addrs[i]
|
|
})
|
|
return addrs, nil
|
|
}
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
|
|
func LookupIPv4WithResolver(ctx context.Context, host string, r Resolver) ([]netip.Addr, error) {
|
|
if node := DefaultHosts.Search(host); node != nil {
|
|
if ip := node.Data(); ip.Is4() {
|
|
return []netip.Addr{node.Data()}, nil
|
|
}
|
|
}
|
|
|
|
ip, err := netip.ParseAddr(host)
|
|
if err == nil {
|
|
if ip.Is4() || ip.Is4In6() {
|
|
return []netip.Addr{ip}, nil
|
|
}
|
|
return []netip.Addr{}, ErrIPVersion
|
|
}
|
|
|
|
if r != nil {
|
|
return r.LookupIPv4(ctx, host)
|
|
}
|
|
|
|
if DefaultResolver == nil {
|
|
ipAddrs, err := net.DefaultResolver.LookupIP(ctx, "ip4", host)
|
|
if err != nil {
|
|
return []netip.Addr{}, err
|
|
} else if len(ipAddrs) == 0 {
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
|
|
addrs := make([]netip.Addr, 0, len(ipAddrs))
|
|
for _, ipAddr := range ipAddrs {
|
|
addrs = append(addrs, nnip.IpToAddr(ipAddr))
|
|
}
|
|
|
|
rand.Shuffle(len(addrs), func(i, j int) {
|
|
addrs[i], addrs[j] = addrs[j], addrs[i]
|
|
})
|
|
return addrs, nil
|
|
}
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
|
|
func LookupIPWithResolver(ctx context.Context, host string, r Resolver) ([]netip.Addr, error) {
|
|
if node := DefaultHosts.Search(host); node != nil {
|
|
return []netip.Addr{node.Data()}, nil
|
|
}
|
|
|
|
ip, err := netip.ParseAddr(host)
|
|
if err == nil {
|
|
return []netip.Addr{ip}, nil
|
|
}
|
|
|
|
if r != nil {
|
|
if DisableIPv6 {
|
|
return r.LookupIPv4(ctx, host)
|
|
}
|
|
|
|
return r.LookupIP(ctx, host)
|
|
} else if DisableIPv6 {
|
|
return LookupIPv4(ctx, host)
|
|
}
|
|
|
|
if DefaultResolver == nil {
|
|
ipAddrs, err := net.DefaultResolver.LookupIP(ctx, "ip", host)
|
|
if err != nil {
|
|
return []netip.Addr{}, err
|
|
} else if len(ipAddrs) == 0 {
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
addrs := make([]netip.Addr, 0, len(ipAddrs))
|
|
for _, ipAddr := range ipAddrs {
|
|
addrs = append(addrs, nnip.IpToAddr(ipAddr))
|
|
}
|
|
|
|
rand.Shuffle(len(addrs), func(i, j int) {
|
|
addrs[i], addrs[j] = addrs[j], addrs[i]
|
|
})
|
|
return addrs, nil
|
|
}
|
|
return []netip.Addr{}, ErrIPNotFound
|
|
}
|
|
|
|
func LookupIP(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
return LookupIPWithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
func LookupIPv4(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
return LookupIPv4WithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
func LookupIPv6(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
return LookupIPv6WithResolver(ctx, host, DefaultResolver)
|
|
}
|
|
|
|
func LookupIPv6ProxyServerHost(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
return LookupIPv6WithResolver(ctx, host, ProxyServerHostResolver)
|
|
}
|
|
return LookupIPv6(ctx, host)
|
|
}
|
|
|
|
func LookupIPv4ProxyServerHost(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
return LookupIPv4WithResolver(ctx, host, ProxyServerHostResolver)
|
|
}
|
|
return LookupIPv4(ctx, host)
|
|
}
|
|
|
|
func LookupIPProxyServerHost(ctx context.Context, host string) ([]netip.Addr, error) {
|
|
if ProxyServerHostResolver != nil {
|
|
return LookupIPWithResolver(ctx, host, ProxyServerHostResolver)
|
|
}
|
|
return LookupIP(ctx, host)
|
|
}
|