|
|
@@ -27,10 +27,28 @@ 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 {
|
|
|
if q.Name == "google.com." && q.Qtype == dns.TypeA {
|
|
|
- rr, _ := dns.NewRR("google.com. IN A 8.8.8.8")
|
|
|
- ans.Answer = append(ans.Answer, rr)
|
|
|
+ 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)
|
|
|
+ }
|
|
|
} else if 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)
|
|
|
@@ -39,6 +57,62 @@ func (*staticHandler) ServeDNS(w dns.ResponseWriter, r *dns.Msg) {
|
|
|
w.WriteMsg(ans)
|
|
|
}
|
|
|
|
|
|
+func TestUDPServerSubnet(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),
|
|
|
+ },
|
|
|
+ },
|
|
|
+ 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)
|
|
|
+ assert(err, IsNil)
|
|
|
+
|
|
|
+ client := v.GetFeature(feature_dns.ClientType()).(feature_dns.Client)
|
|
|
+
|
|
|
+ ips, err := client.LookupIP("google.com")
|
|
|
+ assert(err, IsNil)
|
|
|
+ assert(len(ips), Equals, 1)
|
|
|
+ assert([]byte(ips[0]), Equals, []byte{8, 8, 4, 4})
|
|
|
+}
|
|
|
+
|
|
|
func TestUDPServer(t *testing.T) {
|
|
|
if runtime.GOOS == "windows" {
|
|
|
t.Skip("doesn't work on Windows due to miekg/dns changes.")
|