| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972 |
- package dns_test
- import (
- "testing"
- "time"
- "github.com/google/go-cmp/cmp"
- "github.com/miekg/dns"
- "github.com/xtls/xray-core/v1/app/dispatcher"
- . "github.com/xtls/xray-core/v1/app/dns"
- "github.com/xtls/xray-core/v1/app/policy"
- "github.com/xtls/xray-core/v1/app/proxyman"
- _ "github.com/xtls/xray-core/v1/app/proxyman/outbound"
- "github.com/xtls/xray-core/v1/app/router"
- "github.com/xtls/xray-core/v1/common"
- "github.com/xtls/xray-core/v1/common/net"
- "github.com/xtls/xray-core/v1/common/serial"
- "github.com/xtls/xray-core/v1/core"
- feature_dns "github.com/xtls/xray-core/v1/features/dns"
- "github.com/xtls/xray-core/v1/proxy/freedom"
- "github.com/xtls/xray-core/v1/testing/servers/udp"
- )
- type staticHandler struct {
- }
- func (*staticHandler) ServeDNS(w dns.ResponseWriter, r *dns.Msg) {
- ans := new(dns.Msg)
- ans.Id = r.Id
- var clientIP net.IP
- opt := r.IsEdns0()
- if opt != nil {
- for _, o := range opt.Option {
- if o.Option() == dns.EDNS0SUBNET {
- subnet := o.(*dns.EDNS0_SUBNET)
- clientIP = subnet.Address
- }
- }
- }
- for _, q := range r.Question {
- switch {
- case q.Name == "google.com." && q.Qtype == dns.TypeA:
- if clientIP == nil {
- rr, _ := dns.NewRR("google.com. IN A 8.8.8.8")
- ans.Answer = append(ans.Answer, rr)
- } else {
- rr, _ := dns.NewRR("google.com. IN A 8.8.4.4")
- ans.Answer = append(ans.Answer, rr)
- }
- case q.Name == "api.google.com." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("api.google.com. IN A 8.8.7.7")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "v2.api.google.com." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("v2.api.google.com. IN A 8.8.7.8")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "facebook.com." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("facebook.com. IN A 9.9.9.9")
- ans.Answer = append(ans.Answer, rr)
- case 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)
- case 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)
- case q.Name == "notexist.google.com." && q.Qtype == dns.TypeAAAA:
- ans.MsgHdr.Rcode = dns.RcodeNameError
- case q.Name == "hostname." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("hostname. IN A 127.0.0.1")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "hostname.local." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("hostname.local. IN A 127.0.0.1")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "hostname.localdomain." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("hostname.localdomain. IN A 127.0.0.1")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "localhost." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("localhost. IN A 127.0.0.2")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "localhost-a." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("localhost-a. IN A 127.0.0.3")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "localhost-b." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("localhost-b. IN A 127.0.0.4")
- ans.Answer = append(ans.Answer, rr)
- case q.Name == "Mijia\\ Cloud." && q.Qtype == dns.TypeA:
- rr, _ := dns.NewRR("Mijia\\ Cloud. IN A 127.0.0.1")
- ans.Answer = append(ans.Answer, rr)
- }
- }
- w.WriteMsg(ans)
- }
- func TestUDPServerSubnet(t *testing.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),
- },
- },
- ClientIp: []byte{7, 8, 9, 10},
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 4, 4}}); r != "" {
- t.Fatal(r)
- }
- }
- func TestUDPServer(t *testing.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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- {
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- {
- ips, err := client.LookupIP("facebook.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{9, 9, 9, 9}}); r != "" {
- t.Fatal(r)
- }
- }
- {
- _, err := client.LookupIP("notexist.google.com")
- if err == nil {
- t.Fatal("nil error")
- }
- if r := feature_dns.RCodeFromError(err); r != uint16(dns.RcodeNameError) {
- t.Fatal("expected NameError, but got ", r)
- }
- }
- {
- clientv6 := client.(feature_dns.IPv6Lookup)
- ips, err := clientv6.LookupIPv6("ipv4only.google.com")
- if err != feature_dns.ErrEmptyResponse {
- t.Fatal("error: ", err)
- }
- if len(ips) != 0 {
- t.Fatal("ips: ", ips)
- }
- }
- dnsServer.Shutdown()
- {
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- }
- func TestPrioritizedDomain(t *testing.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: 9999, /* unreachable */
- },
- },
- NameServer: []*NameServer{
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- {
- Type: DomainMatchingType_Full,
- Domain: "google.com",
- },
- },
- },
- },
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- startTime := time.Now()
- {
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- endTime := time.Now()
- if startTime.After(endTime.Add(time.Second * 2)) {
- t.Error("DNS query doesn't finish in 2 seconds.")
- }
- }
- func TestUDPServerIPv6(t *testing.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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- client6 := client.(feature_dns.IPv6Lookup)
- {
- ips, err := client6.LookupIPv6("ipv6.google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{32, 1, 72, 96, 72, 96, 0, 0, 0, 0, 0, 0, 0, 0, 136, 136}}); r != "" {
- t.Fatal(r)
- }
- }
- }
- func TestStaticHostDomain(t *testing.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),
- },
- },
- StaticHosts: []*Config_HostMapping{
- {
- Type: DomainMatchingType_Full,
- Domain: "example.com",
- ProxiedDomain: "google.com",
- },
- },
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- {
- ips, err := client.LookupIP("example.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- dnsServer.Shutdown()
- }
- func TestIPMatch(t *testing.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{
- NameServer: []*NameServer{
- // private dns, not match
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- Geoip: []*router.GeoIP{
- {
- CountryCode: "local",
- Cidr: []*router.CIDR{
- {
- // inner ip, will not match
- Ip: []byte{192, 168, 11, 1},
- Prefix: 32,
- },
- },
- },
- },
- },
- // second dns, match ip
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- Geoip: []*router.GeoIP{
- {
- CountryCode: "test",
- Cidr: []*router.CIDR{
- {
- Ip: []byte{8, 8, 8, 8},
- Prefix: 32,
- },
- },
- },
- {
- CountryCode: "test",
- Cidr: []*router.CIDR{
- {
- Ip: []byte{8, 8, 8, 4},
- Prefix: 32,
- },
- },
- },
- },
- },
- },
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- startTime := time.Now()
- {
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- endTime := time.Now()
- if startTime.After(endTime.Add(time.Second * 2)) {
- t.Error("DNS query doesn't finish in 2 seconds.")
- }
- }
- func TestLocalDomain(t *testing.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: 9999, /* unreachable */
- },
- },
- NameServer: []*NameServer{
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- // Equivalent of dotless:localhost
- {Type: DomainMatchingType_Regex, Domain: "^[^.]*localhost[^.]*$"},
- },
- Geoip: []*router.GeoIP{
- { // Will match localhost, localhost-a and localhost-b,
- CountryCode: "local",
- Cidr: []*router.CIDR{
- {Ip: []byte{127, 0, 0, 2}, Prefix: 32},
- {Ip: []byte{127, 0, 0, 3}, Prefix: 32},
- {Ip: []byte{127, 0, 0, 4}, Prefix: 32},
- },
- },
- },
- },
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- // Equivalent of dotless: and domain:local
- {Type: DomainMatchingType_Regex, Domain: "^[^.]*$"},
- {Type: DomainMatchingType_Subdomain, Domain: "local"},
- {Type: DomainMatchingType_Subdomain, Domain: "localdomain"},
- },
- },
- },
- StaticHosts: []*Config_HostMapping{
- {
- Type: DomainMatchingType_Full,
- Domain: "hostnamestatic",
- Ip: [][]byte{{127, 0, 0, 53}},
- },
- {
- Type: DomainMatchingType_Full,
- Domain: "hostnamealias",
- ProxiedDomain: "hostname.localdomain",
- },
- },
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- startTime := time.Now()
- { // Will match dotless:
- ips, err := client.LookupIP("hostname")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 1}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match domain:local
- ips, err := client.LookupIP("hostname.local")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 1}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match static ip
- ips, err := client.LookupIP("hostnamestatic")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 53}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match domain replacing
- ips, err := client.LookupIP("hostnamealias")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 1}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match dotless:localhost, but not expectIPs: 127.0.0.2, 127.0.0.3, then matches at dotless:
- ips, err := client.LookupIP("localhost")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 2}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match dotless:localhost, and expectIPs: 127.0.0.2, 127.0.0.3
- ips, err := client.LookupIP("localhost-a")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 3}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match dotless:localhost, and expectIPs: 127.0.0.2, 127.0.0.3
- ips, err := client.LookupIP("localhost-b")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 4}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match dotless:
- ips, err := client.LookupIP("Mijia Cloud")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{127, 0, 0, 1}}); r != "" {
- t.Fatal(r)
- }
- }
- endTime := time.Now()
- if startTime.After(endTime.Add(time.Second * 2)) {
- t.Error("DNS query doesn't finish in 2 seconds.")
- }
- }
- func TestMultiMatchPrioritizedDomain(t *testing.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: 9999, /* unreachable */
- },
- },
- NameServer: []*NameServer{
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- {
- Type: DomainMatchingType_Subdomain,
- Domain: "google.com",
- },
- },
- Geoip: []*router.GeoIP{
- { // Will only match 8.8.8.8 and 8.8.4.4
- Cidr: []*router.CIDR{
- {Ip: []byte{8, 8, 8, 8}, Prefix: 32},
- {Ip: []byte{8, 8, 4, 4}, Prefix: 32},
- },
- },
- },
- },
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- {
- Type: DomainMatchingType_Subdomain,
- Domain: "google.com",
- },
- },
- Geoip: []*router.GeoIP{
- { // Will match 8.8.8.8 and 8.8.8.7, etc
- Cidr: []*router.CIDR{
- {Ip: []byte{8, 8, 8, 7}, Prefix: 24},
- },
- },
- },
- },
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- {
- Type: DomainMatchingType_Subdomain,
- Domain: "api.google.com",
- },
- },
- Geoip: []*router.GeoIP{
- { // Will only match 8.8.7.7 (api.google.com)
- Cidr: []*router.CIDR{
- {Ip: []byte{8, 8, 7, 7}, Prefix: 32},
- },
- },
- },
- },
- {
- Address: &net.Endpoint{
- Network: net.Network_UDP,
- Address: &net.IPOrDomain{
- Address: &net.IPOrDomain_Ip{
- Ip: []byte{127, 0, 0, 1},
- },
- },
- Port: uint32(port),
- },
- PrioritizedDomain: []*NameServer_PriorityDomain{
- {
- Type: DomainMatchingType_Full,
- Domain: "v2.api.google.com",
- },
- },
- Geoip: []*router.GeoIP{
- { // Will only match 8.8.7.8 (v2.api.google.com)
- Cidr: []*router.CIDR{
- {Ip: []byte{8, 8, 7, 8}, Prefix: 32},
- },
- },
- },
- },
- },
- }),
- 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)
- common.Must(err)
- client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
- startTime := time.Now()
- { // Will match server 1,2 and server 1 returns expected ip
- ips, err := client.LookupIP("google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match server 1,2 and server 1 returns unexpected ip, then server 2 returns expected one
- clientv4 := client.(feature_dns.IPv4Lookup)
- ips, err := clientv4.LookupIPv4("ipv6.google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 7}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match server 3,1,2 and server 3 returns expected one
- ips, err := client.LookupIP("api.google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 7, 7}}); r != "" {
- t.Fatal(r)
- }
- }
- { // Will match server 4,3,1,2 and server 4 returns expected one
- ips, err := client.LookupIP("v2.api.google.com")
- if err != nil {
- t.Fatal("unexpected error: ", err)
- }
- if r := cmp.Diff(ips, []net.IP{{8, 8, 7, 8}}); r != "" {
- t.Fatal(r)
- }
- }
- endTime := time.Now()
- if startTime.After(endTime.Add(time.Second * 2)) {
- t.Error("DNS query doesn't finish in 2 seconds.")
- }
- }
|