github.com/v2fly/v2ray-core/v5@v5.16.2-0.20240507031116-8191faa6e095/proxy/dns/dns_test.go (about) 1 package dns_test 2 3 import ( 4 "strconv" 5 "testing" 6 "time" 7 8 "github.com/google/go-cmp/cmp" 9 "github.com/miekg/dns" 10 "google.golang.org/protobuf/types/known/anypb" 11 12 core "github.com/v2fly/v2ray-core/v5" 13 "github.com/v2fly/v2ray-core/v5/app/dispatcher" 14 dnsapp "github.com/v2fly/v2ray-core/v5/app/dns" 15 "github.com/v2fly/v2ray-core/v5/app/policy" 16 "github.com/v2fly/v2ray-core/v5/app/proxyman" 17 _ "github.com/v2fly/v2ray-core/v5/app/proxyman/inbound" 18 _ "github.com/v2fly/v2ray-core/v5/app/proxyman/outbound" 19 "github.com/v2fly/v2ray-core/v5/common" 20 "github.com/v2fly/v2ray-core/v5/common/net" 21 "github.com/v2fly/v2ray-core/v5/common/serial" 22 dns_proxy "github.com/v2fly/v2ray-core/v5/proxy/dns" 23 "github.com/v2fly/v2ray-core/v5/proxy/dokodemo" 24 "github.com/v2fly/v2ray-core/v5/testing/servers/tcp" 25 "github.com/v2fly/v2ray-core/v5/testing/servers/udp" 26 ) 27 28 type staticHandler struct{} 29 30 func (*staticHandler) ServeDNS(w dns.ResponseWriter, r *dns.Msg) { 31 ans := new(dns.Msg) 32 ans.Id = r.Id 33 34 var clientIP net.IP 35 36 opt := r.IsEdns0() 37 if opt != nil { 38 for _, o := range opt.Option { 39 if o.Option() == dns.EDNS0SUBNET { 40 subnet := o.(*dns.EDNS0_SUBNET) 41 clientIP = subnet.Address 42 } 43 } 44 } 45 46 for _, q := range r.Question { 47 switch { 48 case q.Name == "google.com." && q.Qtype == dns.TypeA: 49 if clientIP == nil { 50 rr, _ := dns.NewRR("google.com. IN A 8.8.8.8") 51 ans.Answer = append(ans.Answer, rr) 52 } else { 53 rr, _ := dns.NewRR("google.com. IN A 8.8.4.4") 54 ans.Answer = append(ans.Answer, rr) 55 } 56 57 case q.Name == "facebook.com." && q.Qtype == dns.TypeA: 58 rr, _ := dns.NewRR("facebook.com. IN A 9.9.9.9") 59 ans.Answer = append(ans.Answer, rr) 60 61 case q.Name == "ipv6.google.com." && q.Qtype == dns.TypeA: 62 rr, err := dns.NewRR("ipv6.google.com. IN A 8.8.8.7") 63 common.Must(err) 64 ans.Answer = append(ans.Answer, rr) 65 66 case q.Name == "ipv6.google.com." && q.Qtype == dns.TypeAAAA: 67 rr, err := dns.NewRR("ipv6.google.com. IN AAAA 2001:4860:4860::8888") 68 common.Must(err) 69 ans.Answer = append(ans.Answer, rr) 70 71 case q.Name == "notexist.google.com." && q.Qtype == dns.TypeAAAA: 72 ans.MsgHdr.Rcode = dns.RcodeNameError 73 } 74 } 75 w.WriteMsg(ans) 76 } 77 78 func TestUDPDNSTunnel(t *testing.T) { 79 port := udp.PickPort() 80 81 dnsServer := dns.Server{ 82 Addr: "127.0.0.1:" + port.String(), 83 Net: "udp", 84 Handler: &staticHandler{}, 85 UDPSize: 1200, 86 } 87 defer dnsServer.Shutdown() 88 89 go dnsServer.ListenAndServe() 90 time.Sleep(time.Second) 91 92 serverPort := udp.PickPort() 93 config := &core.Config{ 94 App: []*anypb.Any{ 95 serial.ToTypedMessage(&dnsapp.Config{ 96 NameServers: []*net.Endpoint{ 97 { 98 Network: net.Network_UDP, 99 Address: &net.IPOrDomain{ 100 Address: &net.IPOrDomain_Ip{ 101 Ip: []byte{127, 0, 0, 1}, 102 }, 103 }, 104 Port: uint32(port), 105 }, 106 }, 107 }), 108 serial.ToTypedMessage(&dispatcher.Config{}), 109 serial.ToTypedMessage(&proxyman.OutboundConfig{}), 110 serial.ToTypedMessage(&proxyman.InboundConfig{}), 111 serial.ToTypedMessage(&policy.Config{}), 112 }, 113 Inbound: []*core.InboundHandlerConfig{ 114 { 115 ProxySettings: serial.ToTypedMessage(&dokodemo.Config{ 116 Address: net.NewIPOrDomain(net.LocalHostIP), 117 Port: uint32(port), 118 Networks: []net.Network{net.Network_UDP}, 119 }), 120 ReceiverSettings: serial.ToTypedMessage(&proxyman.ReceiverConfig{ 121 PortRange: net.SinglePortRange(serverPort), 122 Listen: net.NewIPOrDomain(net.LocalHostIP), 123 }), 124 }, 125 }, 126 Outbound: []*core.OutboundHandlerConfig{ 127 { 128 ProxySettings: serial.ToTypedMessage(&dns_proxy.Config{}), 129 }, 130 }, 131 } 132 133 v, err := core.New(config) 134 common.Must(err) 135 common.Must(v.Start()) 136 defer v.Close() 137 138 { 139 m1 := new(dns.Msg) 140 m1.Id = dns.Id() 141 m1.RecursionDesired = true 142 m1.Question = make([]dns.Question, 1) 143 m1.Question[0] = dns.Question{Name: "google.com.", Qtype: dns.TypeA, Qclass: dns.ClassINET} 144 145 c := new(dns.Client) 146 in, _, err := c.Exchange(m1, "127.0.0.1:"+strconv.Itoa(int(serverPort))) 147 common.Must(err) 148 149 if len(in.Answer) != 1 { 150 t.Fatal("len(answer): ", len(in.Answer)) 151 } 152 153 rr, ok := in.Answer[0].(*dns.A) 154 if !ok { 155 t.Fatal("not A record") 156 } 157 if r := cmp.Diff(rr.A[:], net.IP{8, 8, 8, 8}); r != "" { 158 t.Error(r) 159 } 160 } 161 162 { 163 m1 := new(dns.Msg) 164 m1.Id = dns.Id() 165 m1.RecursionDesired = true 166 m1.Question = make([]dns.Question, 1) 167 m1.Question[0] = dns.Question{Name: "ipv4only.google.com.", Qtype: dns.TypeAAAA, Qclass: dns.ClassINET} 168 169 c := new(dns.Client) 170 c.Timeout = 10 * time.Second 171 in, _, err := c.Exchange(m1, "127.0.0.1:"+strconv.Itoa(int(serverPort))) 172 common.Must(err) 173 174 if len(in.Answer) != 0 { 175 t.Fatal("len(answer): ", len(in.Answer)) 176 } 177 } 178 179 { 180 m1 := new(dns.Msg) 181 m1.Id = dns.Id() 182 m1.RecursionDesired = true 183 m1.Question = make([]dns.Question, 1) 184 m1.Question[0] = dns.Question{Name: "notexist.google.com.", Qtype: dns.TypeAAAA, Qclass: dns.ClassINET} 185 186 c := new(dns.Client) 187 in, _, err := c.Exchange(m1, "127.0.0.1:"+strconv.Itoa(int(serverPort))) 188 common.Must(err) 189 190 if in.Rcode != dns.RcodeNameError { 191 t.Error("expected NameError, but got ", in.Rcode) 192 } 193 } 194 } 195 196 func TestTCPDNSTunnel(t *testing.T) { 197 port := udp.PickPort() 198 199 dnsServer := dns.Server{ 200 Addr: "127.0.0.1:" + port.String(), 201 Net: "udp", 202 Handler: &staticHandler{}, 203 } 204 defer dnsServer.Shutdown() 205 206 go dnsServer.ListenAndServe() 207 time.Sleep(time.Second) 208 209 serverPort := tcp.PickPort() 210 config := &core.Config{ 211 App: []*anypb.Any{ 212 serial.ToTypedMessage(&dnsapp.Config{ 213 NameServer: []*dnsapp.NameServer{ 214 { 215 Address: &net.Endpoint{ 216 Network: net.Network_UDP, 217 Address: &net.IPOrDomain{ 218 Address: &net.IPOrDomain_Ip{ 219 Ip: []byte{127, 0, 0, 1}, 220 }, 221 }, 222 Port: uint32(port), 223 }, 224 }, 225 }, 226 }), 227 serial.ToTypedMessage(&dispatcher.Config{}), 228 serial.ToTypedMessage(&proxyman.OutboundConfig{}), 229 serial.ToTypedMessage(&proxyman.InboundConfig{}), 230 serial.ToTypedMessage(&policy.Config{}), 231 }, 232 Inbound: []*core.InboundHandlerConfig{ 233 { 234 ProxySettings: serial.ToTypedMessage(&dokodemo.Config{ 235 Address: net.NewIPOrDomain(net.LocalHostIP), 236 Port: uint32(port), 237 Networks: []net.Network{net.Network_TCP}, 238 }), 239 ReceiverSettings: serial.ToTypedMessage(&proxyman.ReceiverConfig{ 240 PortRange: net.SinglePortRange(serverPort), 241 Listen: net.NewIPOrDomain(net.LocalHostIP), 242 }), 243 }, 244 }, 245 Outbound: []*core.OutboundHandlerConfig{ 246 { 247 ProxySettings: serial.ToTypedMessage(&dns_proxy.Config{}), 248 }, 249 }, 250 } 251 252 v, err := core.New(config) 253 common.Must(err) 254 common.Must(v.Start()) 255 defer v.Close() 256 257 m1 := new(dns.Msg) 258 m1.Id = dns.Id() 259 m1.RecursionDesired = true 260 m1.Question = make([]dns.Question, 1) 261 m1.Question[0] = dns.Question{Name: "google.com.", Qtype: dns.TypeA, Qclass: dns.ClassINET} 262 263 c := &dns.Client{ 264 Net: "tcp", 265 } 266 in, _, err := c.Exchange(m1, "127.0.0.1:"+serverPort.String()) 267 common.Must(err) 268 269 if len(in.Answer) != 1 { 270 t.Fatal("len(answer): ", len(in.Answer)) 271 } 272 273 rr, ok := in.Answer[0].(*dns.A) 274 if !ok { 275 t.Fatal("not A record") 276 } 277 if r := cmp.Diff(rr.A[:], net.IP{8, 8, 8, 8}); r != "" { 278 t.Error(r) 279 } 280 } 281 282 func TestUDP2TCPDNSTunnel(t *testing.T) { 283 port := tcp.PickPort() 284 285 dnsServer := dns.Server{ 286 Addr: "127.0.0.1:" + port.String(), 287 Net: "tcp", 288 Handler: &staticHandler{}, 289 } 290 defer dnsServer.Shutdown() 291 292 go dnsServer.ListenAndServe() 293 time.Sleep(time.Second) 294 295 serverPort := tcp.PickPort() 296 config := &core.Config{ 297 App: []*anypb.Any{ 298 serial.ToTypedMessage(&dnsapp.Config{ 299 NameServer: []*dnsapp.NameServer{ 300 { 301 Address: &net.Endpoint{ 302 Network: net.Network_UDP, 303 Address: &net.IPOrDomain{ 304 Address: &net.IPOrDomain_Ip{ 305 Ip: []byte{127, 0, 0, 1}, 306 }, 307 }, 308 Port: uint32(port), 309 }, 310 }, 311 }, 312 }), 313 serial.ToTypedMessage(&dispatcher.Config{}), 314 serial.ToTypedMessage(&proxyman.OutboundConfig{}), 315 serial.ToTypedMessage(&proxyman.InboundConfig{}), 316 serial.ToTypedMessage(&policy.Config{}), 317 }, 318 Inbound: []*core.InboundHandlerConfig{ 319 { 320 ProxySettings: serial.ToTypedMessage(&dokodemo.Config{ 321 Address: net.NewIPOrDomain(net.LocalHostIP), 322 Port: uint32(port), 323 Networks: []net.Network{net.Network_TCP}, 324 }), 325 ReceiverSettings: serial.ToTypedMessage(&proxyman.ReceiverConfig{ 326 PortRange: net.SinglePortRange(serverPort), 327 Listen: net.NewIPOrDomain(net.LocalHostIP), 328 }), 329 }, 330 }, 331 Outbound: []*core.OutboundHandlerConfig{ 332 { 333 ProxySettings: serial.ToTypedMessage(&dns_proxy.Config{ 334 Server: &net.Endpoint{ 335 Network: net.Network_TCP, 336 }, 337 }), 338 }, 339 }, 340 } 341 342 v, err := core.New(config) 343 common.Must(err) 344 common.Must(v.Start()) 345 defer v.Close() 346 347 m1 := new(dns.Msg) 348 m1.Id = dns.Id() 349 m1.RecursionDesired = true 350 m1.Question = make([]dns.Question, 1) 351 m1.Question[0] = dns.Question{Name: "google.com.", Qtype: dns.TypeA, Qclass: dns.ClassINET} 352 353 c := &dns.Client{ 354 Net: "tcp", 355 } 356 in, _, err := c.Exchange(m1, "127.0.0.1:"+serverPort.String()) 357 common.Must(err) 358 359 if len(in.Answer) != 1 { 360 t.Fatal("len(answer): ", len(in.Answer)) 361 } 362 363 rr, ok := in.Answer[0].(*dns.A) 364 if !ok { 365 t.Fatal("not A record") 366 } 367 if r := cmp.Diff(rr.A[:], net.IP{8, 8, 8, 8}); r != "" { 368 t.Error(r) 369 } 370 }