2018-10-11 16:34:31 -04:00
|
|
|
package dns
|
|
|
|
|
|
|
|
import (
|
2022-01-02 10:16:23 -05:00
|
|
|
"github.com/v2fly/v2ray-core/v5/common/errors"
|
|
|
|
"github.com/v2fly/v2ray-core/v5/common/net"
|
|
|
|
"github.com/v2fly/v2ray-core/v5/common/serial"
|
|
|
|
"github.com/v2fly/v2ray-core/v5/features"
|
2018-10-11 16:34:31 -04:00
|
|
|
)
|
|
|
|
|
2021-02-22 21:17:20 -05:00
|
|
|
// IPOption is an object for IP query options.
|
|
|
|
type IPOption struct {
|
|
|
|
IPv4Enable bool
|
|
|
|
IPv6Enable bool
|
|
|
|
FakeEnable bool
|
|
|
|
}
|
|
|
|
|
2022-11-29 19:43:39 -05:00
|
|
|
func (opt IPOption) With(other IPOption) IPOption {
|
|
|
|
return IPOption{
|
|
|
|
IPv4Enable: opt.IPv4Enable && other.IPv4Enable,
|
|
|
|
IPv6Enable: opt.IPv6Enable && other.IPv6Enable,
|
|
|
|
FakeEnable: opt.FakeEnable && other.FakeEnable,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (opt IPOption) IsValid() bool {
|
|
|
|
return opt.IPv4Enable || opt.IPv6Enable
|
|
|
|
}
|
|
|
|
|
2018-10-11 16:34:31 -04:00
|
|
|
// Client is a V2Ray feature for querying DNS information.
|
2018-12-03 16:44:42 -05:00
|
|
|
//
|
|
|
|
// v2ray:api:stable
|
2018-10-11 16:34:31 -04:00
|
|
|
type Client interface {
|
|
|
|
features.Feature
|
2018-10-12 17:57:56 -04:00
|
|
|
|
2018-11-19 14:42:02 -05:00
|
|
|
// LookupIP returns IP address for the given domain. IPs may contain IPv4 and/or IPv6 addresses.
|
2021-03-19 03:55:18 -04:00
|
|
|
LookupIP(domain string) ([]net.IP, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IPv4Lookup is an optional feature for querying IPv4 addresses only.
|
|
|
|
//
|
|
|
|
// v2ray:api:beta
|
|
|
|
type IPv4Lookup interface {
|
|
|
|
LookupIPv4(domain string) ([]net.IP, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IPv6Lookup is an optional feature for querying IPv6 addresses only.
|
|
|
|
//
|
|
|
|
// v2ray:api:beta
|
|
|
|
type IPv6Lookup interface {
|
|
|
|
LookupIPv6(domain string) ([]net.IP, error)
|
|
|
|
}
|
|
|
|
|
2022-12-14 21:38:28 -05:00
|
|
|
// LookupIPWithOption is a helper function for querying DNS information from a dns.Client with dns.IPOption.
|
2021-03-19 03:55:18 -04:00
|
|
|
//
|
|
|
|
// v2ray:api:beta
|
2022-12-14 21:38:28 -05:00
|
|
|
func LookupIPWithOption(client Client, domain string, option IPOption) ([]net.IP, error) {
|
|
|
|
if option.FakeEnable {
|
|
|
|
if clientWithFakeDNS, ok := client.(ClientWithFakeDNS); ok {
|
|
|
|
client = clientWithFakeDNS.AsFakeDNSClient()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if option.IPv4Enable && !option.IPv6Enable {
|
|
|
|
if ipv4Lookup, ok := client.(IPv4Lookup); ok {
|
|
|
|
return ipv4Lookup.LookupIPv4(domain)
|
|
|
|
} else {
|
|
|
|
return nil, errors.New("dns.Client doesn't implement IPv4Lookup")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if option.IPv6Enable && !option.IPv4Enable {
|
|
|
|
if ipv6Lookup, ok := client.(IPv6Lookup); ok {
|
|
|
|
return ipv6Lookup.LookupIPv6(domain)
|
|
|
|
} else {
|
|
|
|
return nil, errors.New("dns.Client doesn't implement IPv6Lookup")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return client.LookupIP(domain)
|
2018-11-19 14:42:02 -05:00
|
|
|
}
|
2018-10-21 04:27:13 -04:00
|
|
|
|
2018-11-19 14:42:02 -05:00
|
|
|
// ClientType returns the type of Client interface. Can be used for implementing common.HasType.
|
2018-12-03 16:44:42 -05:00
|
|
|
//
|
|
|
|
// v2ray:api:beta
|
2018-11-19 14:42:02 -05:00
|
|
|
func ClientType() interface{} {
|
|
|
|
return (*Client)(nil)
|
2018-10-21 04:27:13 -04:00
|
|
|
}
|
2019-02-21 07:43:48 -05:00
|
|
|
|
|
|
|
// ErrEmptyResponse indicates that DNS query succeeded but no answer was returned.
|
|
|
|
var ErrEmptyResponse = errors.New("empty response")
|
|
|
|
|
|
|
|
type RCodeError uint16
|
|
|
|
|
|
|
|
func (e RCodeError) Error() string {
|
|
|
|
return serial.Concat("rcode: ", uint16(e))
|
|
|
|
}
|
|
|
|
|
|
|
|
func RCodeFromError(err error) uint16 {
|
|
|
|
if err == nil {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
cause := errors.Cause(err)
|
|
|
|
if r, ok := cause.(RCodeError); ok {
|
|
|
|
return uint16(r)
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|