1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-17 23:06:30 -05:00

support querying either IPv4 or IPv6 dns

This commit is contained in:
Darien Raymond 2018-11-19 20:42:02 +01:00
parent 8d8eb0f35a
commit bb1efdebd1
No known key found for this signature in database
GPG Key ID: 7251FFA14BB18169
10 changed files with 242 additions and 57 deletions

View File

@ -73,11 +73,24 @@ func NewStaticHosts(hosts []*Config_HostMapping, legacy map[string]*net.IPOrDoma
return sh, nil return sh, nil
} }
func filterIP(ips []net.IP, option IPOption) []net.IP {
filtered := make([]net.IP, 0, len(ips))
for _, ip := range ips {
if (len(ip) == net.IPv4len && option.IPv4Enable) || (len(ip) == net.IPv6len && option.IPv6Enable) {
filtered = append(filtered, ip)
}
}
if len(filtered) == 0 {
return nil
}
return filtered
}
// LookupIP returns IP address for the given domain, if exists in this StaticHosts. // LookupIP returns IP address for the given domain, if exists in this StaticHosts.
func (h *StaticHosts) LookupIP(domain string) []net.IP { func (h *StaticHosts) LookupIP(domain string, option IPOption) []net.IP {
id := h.matchers.Match(domain) id := h.matchers.Match(domain)
if id == 0 { if id == 0 {
return nil return nil
} }
return h.ips[id] return filterIP(h.ips[id], option)
} }

View File

@ -31,7 +31,10 @@ func TestStaticHosts(t *testing.T) {
common.Must(err) common.Must(err)
{ {
ips := hosts.LookupIP("v2ray.com") ips := hosts.LookupIP("v2ray.com", IPOption{
IPv4Enable: true,
IPv6Enable: true,
})
if len(ips) != 1 { if len(ips) != 1 {
t.Error("expect 1 IP, but got ", len(ips)) t.Error("expect 1 IP, but got ", len(ips))
} }
@ -41,7 +44,10 @@ func TestStaticHosts(t *testing.T) {
} }
{ {
ips := hosts.LookupIP("www.v2ray.cn") ips := hosts.LookupIP("www.v2ray.cn", IPOption{
IPv4Enable: true,
IPv6Enable: true,
})
if len(ips) != 1 { if len(ips) != 1 {
t.Error("expect 1 IP, but got ", len(ips)) t.Error("expect 1 IP, but got ", len(ips))
} }

View File

@ -4,32 +4,40 @@ import (
"context" "context"
"v2ray.com/core/common/net" "v2ray.com/core/common/net"
"v2ray.com/core/features/dns/localdns"
) )
type IPOption struct {
IPv4Enable bool
IPv6Enable bool
}
type NameServerInterface interface { type NameServerInterface interface {
QueryIP(ctx context.Context, domain string) ([]net.IP, error) QueryIP(ctx context.Context, domain string, option IPOption) ([]net.IP, error)
} }
type localNameServer struct { type localNameServer struct {
resolver net.Resolver client *localdns.Client
} }
func (s *localNameServer) QueryIP(ctx context.Context, domain string) ([]net.IP, error) { func (s *localNameServer) QueryIP(ctx context.Context, domain string, option IPOption) ([]net.IP, error) {
ipAddr, err := s.resolver.LookupIPAddr(ctx, domain) if option.IPv4Enable && option.IPv6Enable {
if err != nil { return s.client.LookupIP(domain)
return nil, err
} }
var ips []net.IP
for _, addr := range ipAddr { if option.IPv4Enable {
ips = append(ips, addr.IP) return s.client.LookupIPv4(domain)
} }
return ips, nil
if option.IPv6Enable {
return s.client.LookupIPv6(domain)
}
return nil, newError("neither IPv4 nor IPv6 is enabled")
} }
func NewLocalNameServer() *localNameServer { func NewLocalNameServer() *localNameServer {
return &localNameServer{ return &localNameServer{
resolver: net.Resolver{ client: localdns.New(),
PreferGo: true,
},
} }
} }

View File

@ -12,7 +12,10 @@ import (
func TestLocalNameServer(t *testing.T) { func TestLocalNameServer(t *testing.T) {
s := NewLocalNameServer() s := NewLocalNameServer()
ctx, cancel := context.WithTimeout(context.Background(), time.Second*2) ctx, cancel := context.WithTimeout(context.Background(), time.Second*2)
ips, err := s.QueryIP(ctx, "google.com") ips, err := s.QueryIP(ctx, "google.com", IPOption{
IPv4Enable: true,
IPv6Enable: true,
})
cancel() cancel()
common.Must(err) common.Must(err)
if len(ips) == 0 { if len(ips) == 0 {

View File

@ -116,16 +116,39 @@ func (s *Server) Close() error {
return nil return nil
} }
func (s *Server) queryIPTimeout(server NameServerInterface, domain string) ([]net.IP, error) { func (s *Server) queryIPTimeout(server NameServerInterface, domain string, option IPOption) ([]net.IP, error) {
ctx, cancel := context.WithTimeout(context.Background(), time.Second*4) ctx, cancel := context.WithTimeout(context.Background(), time.Second*4)
ips, err := server.QueryIP(ctx, domain) ips, err := server.QueryIP(ctx, domain, option)
cancel() cancel()
return ips, err return ips, err
} }
// LookupIP implements dns.Client. // LookupIP implements dns.Client.
func (s *Server) LookupIP(domain string) ([]net.IP, error) { func (s *Server) LookupIP(domain string) ([]net.IP, error) {
if ip := s.hosts.LookupIP(domain); len(ip) > 0 { return s.lookupIPInternal(domain, IPOption{
IPv4Enable: true,
IPv6Enable: true,
})
}
// LookupIPv4 implements dns.IPv4Lookup.
func (s *Server) LookupIPv4(domain string) ([]net.IP, error) {
return s.lookupIPInternal(domain, IPOption{
IPv4Enable: true,
IPv6Enable: false,
})
}
// LookupIPv6 implements dns.IPv6Lookup.
func (s *Server) LookupIPv6(domain string) ([]net.IP, error) {
return s.lookupIPInternal(domain, IPOption{
IPv4Enable: false,
IPv6Enable: true,
})
}
func (s *Server) lookupIPInternal(domain string, option IPOption) ([]net.IP, error) {
if ip := s.hosts.LookupIP(domain, option); len(ip) > 0 {
return ip, nil return ip, nil
} }
@ -134,7 +157,7 @@ func (s *Server) LookupIP(domain string) ([]net.IP, error) {
idx := s.domainMatcher.Match(domain) idx := s.domainMatcher.Match(domain)
if idx > 0 { if idx > 0 {
ns := s.servers[s.domainIndexMap[idx]] ns := s.servers[s.domainIndexMap[idx]]
ips, err := s.queryIPTimeout(ns, domain) ips, err := s.queryIPTimeout(ns, domain, option)
if len(ips) > 0 { if len(ips) > 0 {
return ips, nil return ips, nil
} }
@ -145,7 +168,7 @@ func (s *Server) LookupIP(domain string) ([]net.IP, error) {
} }
for _, server := range s.servers { for _, server := range s.servers {
ips, err := s.queryIPTimeout(server, domain) ips, err := s.queryIPTimeout(server, domain, option)
if len(ips) > 0 { if len(ips) > 0 {
return ips, nil return ips, nil
} }

View File

@ -11,6 +11,7 @@ import (
"v2ray.com/core/app/policy" "v2ray.com/core/app/policy"
"v2ray.com/core/app/proxyman" "v2ray.com/core/app/proxyman"
_ "v2ray.com/core/app/proxyman/outbound" _ "v2ray.com/core/app/proxyman/outbound"
"v2ray.com/core/common"
"v2ray.com/core/common/net" "v2ray.com/core/common/net"
"v2ray.com/core/common/serial" "v2ray.com/core/common/serial"
feature_dns "v2ray.com/core/features/dns" feature_dns "v2ray.com/core/features/dns"
@ -52,6 +53,14 @@ func (*staticHandler) ServeDNS(w dns.ResponseWriter, r *dns.Msg) {
} else if q.Name == "facebook.com." && q.Qtype == dns.TypeA { } else if q.Name == "facebook.com." && q.Qtype == dns.TypeA {
rr, _ := dns.NewRR("facebook.com. IN A 9.9.9.9") rr, _ := dns.NewRR("facebook.com. IN A 9.9.9.9")
ans.Answer = append(ans.Answer, rr) ans.Answer = append(ans.Answer, rr)
} else if q.Name == "ipv6.google.com." && q.Qtype == dns.TypeA {
rr, err := dns.NewRR("ipv6.google.com. IN A 8.8.8.7")
common.Must(err)
ans.Answer = append(ans.Answer, rr)
} else if q.Name == "ipv6.google.com." && q.Qtype == dns.TypeAAAA {
rr, err := dns.NewRR("ipv6.google.com. IN AAAA 2001:4860:4860::8888")
common.Must(err)
ans.Answer = append(ans.Answer, rr)
} }
} }
w.WriteMsg(ans) w.WriteMsg(ans)
@ -259,3 +268,59 @@ func TestPrioritizedDomain(t *testing.T) {
t.Error("DNS query doesn't finish in 2 seconds.") t.Error("DNS query doesn't finish in 2 seconds.")
} }
} }
func TestUDPServerIPv6(t *testing.T) {
if runtime.GOOS == "windows" {
t.Skip("doesn't work on Windows due to miekg/dns changes.")
}
assert := With(t)
port := udp.PickPort()
dnsServer := dns.Server{
Addr: "127.0.0.1:" + port.String(),
Net: "udp",
Handler: &staticHandler{},
UDPSize: 1200,
}
go dnsServer.ListenAndServe()
time.Sleep(time.Second)
config := &core.Config{
App: []*serial.TypedMessage{
serial.ToTypedMessage(&Config{
NameServers: []*net.Endpoint{
{
Network: net.Network_UDP,
Address: &net.IPOrDomain{
Address: &net.IPOrDomain_Ip{
Ip: []byte{127, 0, 0, 1},
},
},
Port: uint32(port),
},
},
}),
serial.ToTypedMessage(&dispatcher.Config{}),
serial.ToTypedMessage(&proxyman.OutboundConfig{}),
serial.ToTypedMessage(&policy.Config{}),
},
Outbound: []*core.OutboundHandlerConfig{
{
ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
},
},
}
v, err := core.New(config)
assert(err, IsNil)
client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
client6 := client.(feature_dns.IPv6Lookup)
ips, err := client6.LookupIPv6("ipv6.google.com")
assert(err, IsNil)
assert(len(ips), Equals, 1)
assert([]byte(ips[0]), Equals, []byte{32, 1, 72, 96, 72, 96, 0, 0, 0, 0, 0, 0, 0, 0, 136, 136})
}

View File

@ -241,7 +241,7 @@ func (s *ClassicNameServer) addPendingRequest(domain string) uint16 {
return id return id
} }
func (s *ClassicNameServer) buildMsgs(domain string) []*dnsmessage.Message { func (s *ClassicNameServer) buildMsgs(domain string, option IPOption) []*dnsmessage.Message {
qA := dnsmessage.Question{ qA := dnsmessage.Question{
Name: dnsmessage.MustNewName(domain), Name: dnsmessage.MustNewName(domain),
Type: dnsmessage.TypeA, Type: dnsmessage.TypeA,
@ -256,7 +256,7 @@ func (s *ClassicNameServer) buildMsgs(domain string) []*dnsmessage.Message {
var msgs []*dnsmessage.Message var msgs []*dnsmessage.Message
{ if option.IPv4Enable {
msg := new(dnsmessage.Message) msg := new(dnsmessage.Message)
msg.Header.ID = s.addPendingRequest(domain) msg.Header.ID = s.addPendingRequest(domain)
msg.Header.RecursionDesired = true msg.Header.RecursionDesired = true
@ -267,7 +267,7 @@ func (s *ClassicNameServer) buildMsgs(domain string) []*dnsmessage.Message {
msgs = append(msgs, msg) msgs = append(msgs, msg)
} }
{ if option.IPv6Enable {
msg := new(dnsmessage.Message) msg := new(dnsmessage.Message)
msg.Header.ID = s.addPendingRequest(domain) msg.Header.ID = s.addPendingRequest(domain)
msg.Header.RecursionDesired = true msg.Header.RecursionDesired = true
@ -281,7 +281,7 @@ func (s *ClassicNameServer) buildMsgs(domain string) []*dnsmessage.Message {
return msgs return msgs
} }
func msgToBuffer(msg *dnsmessage.Message) (*buf.Buffer, error) { func msgToBuffer2(msg *dnsmessage.Message) (*buf.Buffer, error) {
buffer := buf.New() buffer := buf.New()
rawBytes := buffer.Extend(buf.Size) rawBytes := buffer.Extend(buf.Size)
packed, err := msg.AppendPack(rawBytes[:0]) packed, err := msg.AppendPack(rawBytes[:0])
@ -293,19 +293,19 @@ func msgToBuffer(msg *dnsmessage.Message) (*buf.Buffer, error) {
return buffer, nil return buffer, nil
} }
func (s *ClassicNameServer) sendQuery(ctx context.Context, domain string) { func (s *ClassicNameServer) sendQuery(ctx context.Context, domain string, option IPOption) {
newError("querying DNS for: ", domain).AtDebug().WriteToLog(session.ExportIDToError(ctx)) newError("querying DNS for: ", domain).AtDebug().WriteToLog(session.ExportIDToError(ctx))
msgs := s.buildMsgs(domain) msgs := s.buildMsgs(domain, option)
for _, msg := range msgs { for _, msg := range msgs {
b, err := msgToBuffer(msg) b, err := msgToBuffer2(msg)
common.Must(err) common.Must(err)
s.udpServer.Dispatch(context.Background(), s.address, b) s.udpServer.Dispatch(context.Background(), s.address, b)
} }
} }
func (s *ClassicNameServer) findIPsForDomain(domain string) []net.IP { func (s *ClassicNameServer) findIPsForDomain(domain string, option IPOption) []net.IP {
s.RLock() s.RLock()
records, found := s.ips[domain] records, found := s.ips[domain]
s.RUnlock() s.RUnlock()
@ -318,7 +318,7 @@ func (s *ClassicNameServer) findIPsForDomain(domain string) []net.IP {
ips = append(ips, rec.IP) ips = append(ips, rec.IP)
} }
} }
return ips return filterIP(ips, option)
} }
return nil return nil
} }
@ -330,10 +330,10 @@ func Fqdn(domain string) string {
return domain + "." return domain + "."
} }
func (s *ClassicNameServer) QueryIP(ctx context.Context, domain string) ([]net.IP, error) { func (s *ClassicNameServer) QueryIP(ctx context.Context, domain string, option IPOption) ([]net.IP, error) {
fqdn := Fqdn(domain) fqdn := Fqdn(domain)
ips := s.findIPsForDomain(fqdn) ips := s.findIPsForDomain(fqdn, option)
if len(ips) > 0 { if len(ips) > 0 {
return ips, nil return ips, nil
} }
@ -341,10 +341,10 @@ func (s *ClassicNameServer) QueryIP(ctx context.Context, domain string) ([]net.I
sub := s.pub.Subscribe(fqdn) sub := s.pub.Subscribe(fqdn)
defer sub.Close() defer sub.Close()
s.sendQuery(ctx, fqdn) s.sendQuery(ctx, fqdn, option)
for { for {
ips := s.findIPsForDomain(fqdn) ips := s.findIPsForDomain(fqdn, option)
if len(ips) > 0 { if len(ips) > 0 {
return ips, nil return ips, nil
} }

View File

@ -8,29 +8,22 @@ import (
// Client is a V2Ray feature for querying DNS information. // Client is a V2Ray feature for querying DNS information.
type Client interface { type Client interface {
features.Feature features.Feature
LookupIP(host string) ([]net.IP, error)
// LookupIP returns IP address for the given domain. IPs may contain IPv4 and/or IPv6 addresses.
LookupIP(domain string) ([]net.IP, error)
}
// IPv4Lookup is an optional feature for querying IPv4 addresses only.
type IPv4Lookup interface {
LookupIPv4(domain string) ([]net.IP, error)
}
// IPv6Lookup is an optional feature for querying IPv6 addresses only.
type IPv6Lookup interface {
LookupIPv6(domain string) ([]net.IP, error)
} }
// ClientType returns the type of Client interface. Can be used for implementing common.HasType. // ClientType returns the type of Client interface. Can be used for implementing common.HasType.
func ClientType() interface{} { func ClientType() interface{} {
return (*Client)(nil) return (*Client)(nil)
} }
// LocalClient is an implementation of Client, which queries localhost for DNS.
type LocalClient struct{}
// Type implements common.HasType.
func (LocalClient) Type() interface{} {
return ClientType()
}
// Start implements common.Runnable.
func (LocalClient) Start() error { return nil }
// Close implements common.Closable.
func (LocalClient) Close() error { return nil }
// LookupIP implements Client.
func (LocalClient) LookupIP(host string) ([]net.IP, error) {
return net.LookupIP(host)
}

View File

@ -0,0 +1,73 @@
package localdns
import (
"context"
"net"
"v2ray.com/core/features/dns"
)
// Client is an implementation of dns.Client, which queries localhost for DNS.
type Client struct {
resolver net.Resolver
}
// Type implements common.HasType.
func (*Client) Type() interface{} {
return dns.ClientType()
}
// Start implements common.Runnable.
func (*Client) Start() error { return nil }
// Close implements common.Closable.
func (*Client) Close() error { return nil }
// LookupIP implements Client.
func (c *Client) LookupIP(host string) ([]net.IP, error) {
ipAddr, err := c.resolver.LookupIPAddr(context.Background(), host)
if err != nil {
return nil, err
}
ips := make([]net.IP, 0, len(ipAddr))
for _, addr := range ipAddr {
ips = append(ips, addr.IP)
}
return ips, nil
}
func (c *Client) LookupIPv4(host string) ([]net.IP, error) {
ips, err := c.LookupIP(host)
if err != nil {
return nil, err
}
var ipv4 []net.IP
for _, ip := range ips {
if len(ip) == net.IPv4len {
ipv4 = append(ipv4, ip)
}
}
return ipv4, nil
}
func (c *Client) LookupIPv6(host string) ([]net.IP, error) {
ips, err := c.LookupIP(host)
if err != nil {
return nil, err
}
var ipv6 []net.IP
for _, ip := range ips {
if len(ip) == net.IPv6len {
ipv6 = append(ipv6, ip)
}
}
return ipv6, nil
}
func New() *Client {
return &Client{
resolver: net.Resolver{
PreferGo: true,
},
}
}

View File

@ -9,6 +9,7 @@ import (
"v2ray.com/core/common/serial" "v2ray.com/core/common/serial"
"v2ray.com/core/features" "v2ray.com/core/features"
"v2ray.com/core/features/dns" "v2ray.com/core/features/dns"
"v2ray.com/core/features/dns/localdns"
"v2ray.com/core/features/inbound" "v2ray.com/core/features/inbound"
"v2ray.com/core/features/outbound" "v2ray.com/core/features/outbound"
"v2ray.com/core/features/policy" "v2ray.com/core/features/policy"
@ -183,7 +184,7 @@ func New(config *Config) (*Instance, error) {
Type interface{} Type interface{}
Instance features.Feature Instance features.Feature
}{ }{
{dns.ClientType(), dns.LocalClient{}}, {dns.ClientType(), localdns.New()},
{policy.ManagerType(), policy.DefaultManager{}}, {policy.ManagerType(), policy.DefaultManager{}},
{routing.RouterType(), routing.DefaultRouter{}}, {routing.RouterType(), routing.DefaultRouter{}},
{stats.ManagerType(), stats.NoopManager{}}, {stats.ManagerType(), stats.NoopManager{}},