server_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. package dns_test
  2. import (
  3. "testing"
  4. "time"
  5. "github.com/google/go-cmp/cmp"
  6. "github.com/miekg/dns"
  7. "v2ray.com/core"
  8. "v2ray.com/core/app/dispatcher"
  9. . "v2ray.com/core/app/dns"
  10. "v2ray.com/core/app/policy"
  11. "v2ray.com/core/app/proxyman"
  12. _ "v2ray.com/core/app/proxyman/outbound"
  13. "v2ray.com/core/app/router"
  14. "v2ray.com/core/common"
  15. "v2ray.com/core/common/net"
  16. "v2ray.com/core/common/serial"
  17. feature_dns "v2ray.com/core/features/dns"
  18. "v2ray.com/core/proxy/freedom"
  19. "v2ray.com/core/testing/servers/udp"
  20. )
  21. type staticHandler struct {
  22. }
  23. func (*staticHandler) ServeDNS(w dns.ResponseWriter, r *dns.Msg) {
  24. ans := new(dns.Msg)
  25. ans.Id = r.Id
  26. var clientIP net.IP
  27. opt := r.IsEdns0()
  28. if opt != nil {
  29. for _, o := range opt.Option {
  30. if o.Option() == dns.EDNS0SUBNET {
  31. subnet := o.(*dns.EDNS0_SUBNET)
  32. clientIP = subnet.Address
  33. }
  34. }
  35. }
  36. for _, q := range r.Question {
  37. if q.Name == "google.com." && q.Qtype == dns.TypeA {
  38. if clientIP == nil {
  39. rr, _ := dns.NewRR("google.com. IN A 8.8.8.8")
  40. ans.Answer = append(ans.Answer, rr)
  41. } else {
  42. rr, _ := dns.NewRR("google.com. IN A 8.8.4.4")
  43. ans.Answer = append(ans.Answer, rr)
  44. }
  45. } else if q.Name == "facebook.com." && q.Qtype == dns.TypeA {
  46. rr, _ := dns.NewRR("facebook.com. IN A 9.9.9.9")
  47. ans.Answer = append(ans.Answer, rr)
  48. } else if q.Name == "ipv6.google.com." && q.Qtype == dns.TypeA {
  49. rr, err := dns.NewRR("ipv6.google.com. IN A 8.8.8.7")
  50. common.Must(err)
  51. ans.Answer = append(ans.Answer, rr)
  52. } else if q.Name == "ipv6.google.com." && q.Qtype == dns.TypeAAAA {
  53. rr, err := dns.NewRR("ipv6.google.com. IN AAAA 2001:4860:4860::8888")
  54. common.Must(err)
  55. ans.Answer = append(ans.Answer, rr)
  56. } else if q.Name == "notexist.google.com." && q.Qtype == dns.TypeAAAA {
  57. ans.MsgHdr.Rcode = dns.RcodeNameError
  58. }
  59. }
  60. w.WriteMsg(ans)
  61. }
  62. func TestUDPServerSubnet(t *testing.T) {
  63. port := udp.PickPort()
  64. dnsServer := dns.Server{
  65. Addr: "127.0.0.1:" + port.String(),
  66. Net: "udp",
  67. Handler: &staticHandler{},
  68. UDPSize: 1200,
  69. }
  70. go dnsServer.ListenAndServe()
  71. time.Sleep(time.Second)
  72. config := &core.Config{
  73. App: []*serial.TypedMessage{
  74. serial.ToTypedMessage(&Config{
  75. NameServers: []*net.Endpoint{
  76. {
  77. Network: net.Network_UDP,
  78. Address: &net.IPOrDomain{
  79. Address: &net.IPOrDomain_Ip{
  80. Ip: []byte{127, 0, 0, 1},
  81. },
  82. },
  83. Port: uint32(port),
  84. },
  85. },
  86. ClientIp: []byte{7, 8, 9, 10},
  87. }),
  88. serial.ToTypedMessage(&dispatcher.Config{}),
  89. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  90. serial.ToTypedMessage(&policy.Config{}),
  91. },
  92. Outbound: []*core.OutboundHandlerConfig{
  93. {
  94. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  95. },
  96. },
  97. }
  98. v, err := core.New(config)
  99. common.Must(err)
  100. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  101. ips, err := client.LookupIP("google.com")
  102. if err != nil {
  103. t.Fatal("unexpected error: ", err)
  104. }
  105. if r := cmp.Diff(ips, []net.IP{{8, 8, 4, 4}}); r != "" {
  106. t.Fatal(r)
  107. }
  108. }
  109. func TestUDPServer(t *testing.T) {
  110. port := udp.PickPort()
  111. dnsServer := dns.Server{
  112. Addr: "127.0.0.1:" + port.String(),
  113. Net: "udp",
  114. Handler: &staticHandler{},
  115. UDPSize: 1200,
  116. }
  117. go dnsServer.ListenAndServe()
  118. time.Sleep(time.Second)
  119. config := &core.Config{
  120. App: []*serial.TypedMessage{
  121. serial.ToTypedMessage(&Config{
  122. NameServers: []*net.Endpoint{
  123. {
  124. Network: net.Network_UDP,
  125. Address: &net.IPOrDomain{
  126. Address: &net.IPOrDomain_Ip{
  127. Ip: []byte{127, 0, 0, 1},
  128. },
  129. },
  130. Port: uint32(port),
  131. },
  132. },
  133. }),
  134. serial.ToTypedMessage(&dispatcher.Config{}),
  135. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  136. serial.ToTypedMessage(&policy.Config{}),
  137. },
  138. Outbound: []*core.OutboundHandlerConfig{
  139. {
  140. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  141. },
  142. },
  143. }
  144. v, err := core.New(config)
  145. common.Must(err)
  146. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  147. {
  148. ips, err := client.LookupIP("google.com")
  149. if err != nil {
  150. t.Fatal("unexpected error: ", err)
  151. }
  152. if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
  153. t.Fatal(r)
  154. }
  155. }
  156. {
  157. ips, err := client.LookupIP("facebook.com")
  158. if err != nil {
  159. t.Fatal("unexpected error: ", err)
  160. }
  161. if r := cmp.Diff(ips, []net.IP{{9, 9, 9, 9}}); r != "" {
  162. t.Fatal(r)
  163. }
  164. }
  165. {
  166. _, err := client.LookupIP("notexist.google.com")
  167. if err == nil {
  168. t.Fatal("nil error")
  169. }
  170. if r := feature_dns.RCodeFromError(err); r != uint16(dns.RcodeNameError) {
  171. t.Fatal("expected NameError, but got ", r)
  172. }
  173. }
  174. {
  175. clientv6 := client.(feature_dns.IPv6Lookup)
  176. ips, err := clientv6.LookupIPv6("ipv4only.google.com")
  177. if err != feature_dns.ErrEmptyResponse {
  178. t.Fatal("error: ", err)
  179. }
  180. if len(ips) != 0 {
  181. t.Fatal("ips: ", ips)
  182. }
  183. }
  184. dnsServer.Shutdown()
  185. {
  186. ips, err := client.LookupIP("google.com")
  187. if err != nil {
  188. t.Fatal("unexpected error: ", err)
  189. }
  190. if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
  191. t.Fatal(r)
  192. }
  193. }
  194. }
  195. func TestPrioritizedDomain(t *testing.T) {
  196. port := udp.PickPort()
  197. dnsServer := dns.Server{
  198. Addr: "127.0.0.1:" + port.String(),
  199. Net: "udp",
  200. Handler: &staticHandler{},
  201. UDPSize: 1200,
  202. }
  203. go dnsServer.ListenAndServe()
  204. time.Sleep(time.Second)
  205. config := &core.Config{
  206. App: []*serial.TypedMessage{
  207. serial.ToTypedMessage(&Config{
  208. NameServers: []*net.Endpoint{
  209. {
  210. Network: net.Network_UDP,
  211. Address: &net.IPOrDomain{
  212. Address: &net.IPOrDomain_Ip{
  213. Ip: []byte{127, 0, 0, 1},
  214. },
  215. },
  216. Port: 9999, /* unreachable */
  217. },
  218. },
  219. NameServer: []*NameServer{
  220. {
  221. Address: &net.Endpoint{
  222. Network: net.Network_UDP,
  223. Address: &net.IPOrDomain{
  224. Address: &net.IPOrDomain_Ip{
  225. Ip: []byte{127, 0, 0, 1},
  226. },
  227. },
  228. Port: uint32(port),
  229. },
  230. PrioritizedDomain: []*NameServer_PriorityDomain{
  231. {
  232. Type: DomainMatchingType_Full,
  233. Domain: "google.com",
  234. },
  235. },
  236. },
  237. },
  238. }),
  239. serial.ToTypedMessage(&dispatcher.Config{}),
  240. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  241. serial.ToTypedMessage(&policy.Config{}),
  242. },
  243. Outbound: []*core.OutboundHandlerConfig{
  244. {
  245. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  246. },
  247. },
  248. }
  249. v, err := core.New(config)
  250. common.Must(err)
  251. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  252. startTime := time.Now()
  253. {
  254. ips, err := client.LookupIP("google.com")
  255. if err != nil {
  256. t.Fatal("unexpected error: ", err)
  257. }
  258. if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
  259. t.Fatal(r)
  260. }
  261. }
  262. endTime := time.Now()
  263. if startTime.After(endTime.Add(time.Second * 2)) {
  264. t.Error("DNS query doesn't finish in 2 seconds.")
  265. }
  266. }
  267. func TestUDPServerIPv6(t *testing.T) {
  268. port := udp.PickPort()
  269. dnsServer := dns.Server{
  270. Addr: "127.0.0.1:" + port.String(),
  271. Net: "udp",
  272. Handler: &staticHandler{},
  273. UDPSize: 1200,
  274. }
  275. go dnsServer.ListenAndServe()
  276. time.Sleep(time.Second)
  277. config := &core.Config{
  278. App: []*serial.TypedMessage{
  279. serial.ToTypedMessage(&Config{
  280. NameServers: []*net.Endpoint{
  281. {
  282. Network: net.Network_UDP,
  283. Address: &net.IPOrDomain{
  284. Address: &net.IPOrDomain_Ip{
  285. Ip: []byte{127, 0, 0, 1},
  286. },
  287. },
  288. Port: uint32(port),
  289. },
  290. },
  291. }),
  292. serial.ToTypedMessage(&dispatcher.Config{}),
  293. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  294. serial.ToTypedMessage(&policy.Config{}),
  295. },
  296. Outbound: []*core.OutboundHandlerConfig{
  297. {
  298. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  299. },
  300. },
  301. }
  302. v, err := core.New(config)
  303. common.Must(err)
  304. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  305. client6 := client.(feature_dns.IPv6Lookup)
  306. {
  307. ips, err := client6.LookupIPv6("ipv6.google.com")
  308. if err != nil {
  309. t.Fatal("unexpected error: ", err)
  310. }
  311. if r := cmp.Diff(ips, []net.IP{{32, 1, 72, 96, 72, 96, 0, 0, 0, 0, 0, 0, 0, 0, 136, 136}}); r != "" {
  312. t.Fatal(r)
  313. }
  314. }
  315. }
  316. func TestStaticHostDomain(t *testing.T) {
  317. port := udp.PickPort()
  318. dnsServer := dns.Server{
  319. Addr: "127.0.0.1:" + port.String(),
  320. Net: "udp",
  321. Handler: &staticHandler{},
  322. UDPSize: 1200,
  323. }
  324. go dnsServer.ListenAndServe()
  325. time.Sleep(time.Second)
  326. config := &core.Config{
  327. App: []*serial.TypedMessage{
  328. serial.ToTypedMessage(&Config{
  329. NameServers: []*net.Endpoint{
  330. {
  331. Network: net.Network_UDP,
  332. Address: &net.IPOrDomain{
  333. Address: &net.IPOrDomain_Ip{
  334. Ip: []byte{127, 0, 0, 1},
  335. },
  336. },
  337. Port: uint32(port),
  338. },
  339. },
  340. StaticHosts: []*Config_HostMapping{
  341. {
  342. Type: DomainMatchingType_Full,
  343. Domain: "example.com",
  344. ProxiedDomain: "google.com",
  345. },
  346. },
  347. }),
  348. serial.ToTypedMessage(&dispatcher.Config{}),
  349. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  350. serial.ToTypedMessage(&policy.Config{}),
  351. },
  352. Outbound: []*core.OutboundHandlerConfig{
  353. {
  354. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  355. },
  356. },
  357. }
  358. v, err := core.New(config)
  359. common.Must(err)
  360. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  361. {
  362. ips, err := client.LookupIP("example.com")
  363. if err != nil {
  364. t.Fatal("unexpected error: ", err)
  365. }
  366. if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
  367. t.Fatal(r)
  368. }
  369. }
  370. dnsServer.Shutdown()
  371. }
  372. func TestIPMatch(t *testing.T) {
  373. port := udp.PickPort()
  374. dnsServer := dns.Server{
  375. Addr: "127.0.0.1:" + port.String(),
  376. Net: "udp",
  377. Handler: &staticHandler{},
  378. UDPSize: 1200,
  379. }
  380. go dnsServer.ListenAndServe()
  381. time.Sleep(time.Second)
  382. config := &core.Config{
  383. App: []*serial.TypedMessage{
  384. serial.ToTypedMessage(&Config{
  385. NameServer: []*NameServer{
  386. // private dns, not match
  387. {
  388. Address: &net.Endpoint{
  389. Network: net.Network_UDP,
  390. Address: &net.IPOrDomain{
  391. Address: &net.IPOrDomain_Ip{
  392. Ip: []byte{127, 0, 0, 1},
  393. },
  394. },
  395. Port: uint32(port),
  396. },
  397. Geoip: []*router.GeoIP{
  398. {
  399. CountryCode: "local",
  400. Cidr: []*router.CIDR{
  401. {
  402. // inner ip, will not match
  403. Ip: []byte{192, 168, 11, 1},
  404. Prefix: 32,
  405. },
  406. },
  407. },
  408. },
  409. },
  410. // second dns, match ip
  411. {
  412. Address: &net.Endpoint{
  413. Network: net.Network_UDP,
  414. Address: &net.IPOrDomain{
  415. Address: &net.IPOrDomain_Ip{
  416. Ip: []byte{127, 0, 0, 1},
  417. },
  418. },
  419. Port: uint32(port),
  420. },
  421. Geoip: []*router.GeoIP{
  422. {
  423. CountryCode: "test",
  424. Cidr: []*router.CIDR{
  425. {
  426. Ip: []byte{8, 8, 8, 8},
  427. Prefix: 32,
  428. },
  429. },
  430. },
  431. {
  432. CountryCode: "test",
  433. Cidr: []*router.CIDR{
  434. {
  435. Ip: []byte{8, 8, 8, 4},
  436. Prefix: 32,
  437. },
  438. },
  439. },
  440. },
  441. },
  442. },
  443. }),
  444. serial.ToTypedMessage(&dispatcher.Config{}),
  445. serial.ToTypedMessage(&proxyman.OutboundConfig{}),
  446. serial.ToTypedMessage(&policy.Config{}),
  447. },
  448. Outbound: []*core.OutboundHandlerConfig{
  449. {
  450. ProxySettings: serial.ToTypedMessage(&freedom.Config{}),
  451. },
  452. },
  453. }
  454. v, err := core.New(config)
  455. common.Must(err)
  456. client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
  457. startTime := time.Now()
  458. {
  459. ips, err := client.LookupIP("google.com")
  460. if err != nil {
  461. t.Fatal("unexpected error: ", err)
  462. }
  463. if r := cmp.Diff(ips, []net.IP{{8, 8, 8, 8}}); r != "" {
  464. t.Fatal(r)
  465. }
  466. }
  467. endTime := time.Now()
  468. if startTime.After(endTime.Add(time.Second * 2)) {
  469. t.Error("DNS query doesn't finish in 2 seconds.")
  470. }
  471. }