mirror of
https://github.com/v2fly/v2ray-core.git
synced 2025-01-02 15:36:41 -05:00
Allow freedom to consume DNS settings
This commit is contained in:
parent
bb503c6954
commit
0ea2678e72
@ -39,7 +39,7 @@ func TestDokodemoTCP(t *testing.T) {
|
|||||||
space := app.NewSpace()
|
space := app.NewSpace()
|
||||||
space.BindApp(dispatcher.APP_ID, dispatchers.NewDefaultDispatcher(space))
|
space.BindApp(dispatcher.APP_ID, dispatchers.NewDefaultDispatcher(space))
|
||||||
ohm := proxyman.NewDefaultOutboundHandlerManager()
|
ohm := proxyman.NewDefaultOutboundHandlerManager()
|
||||||
ohm.SetDefaultHandler(&freedom.FreedomConnection{})
|
ohm.SetDefaultHandler(freedom.NewFreedomConnection(&freedom.Config{}, space))
|
||||||
space.BindApp(proxyman.APP_ID_OUTBOUND_MANAGER, ohm)
|
space.BindApp(proxyman.APP_ID_OUTBOUND_MANAGER, ohm)
|
||||||
|
|
||||||
data2Send := "Data to be sent to remote."
|
data2Send := "Data to be sent to remote."
|
||||||
@ -97,7 +97,7 @@ func TestDokodemoUDP(t *testing.T) {
|
|||||||
space := app.NewSpace()
|
space := app.NewSpace()
|
||||||
space.BindApp(dispatcher.APP_ID, dispatchers.NewDefaultDispatcher(space))
|
space.BindApp(dispatcher.APP_ID, dispatchers.NewDefaultDispatcher(space))
|
||||||
ohm := proxyman.NewDefaultOutboundHandlerManager()
|
ohm := proxyman.NewDefaultOutboundHandlerManager()
|
||||||
ohm.SetDefaultHandler(&freedom.FreedomConnection{})
|
ohm.SetDefaultHandler(freedom.NewFreedomConnection(&freedom.Config{}, space))
|
||||||
space.BindApp(proxyman.APP_ID_OUTBOUND_MANAGER, ohm)
|
space.BindApp(proxyman.APP_ID_OUTBOUND_MANAGER, ohm)
|
||||||
|
|
||||||
data2Send := "Data to be sent to remote."
|
data2Send := "Data to be sent to remote."
|
||||||
|
@ -1,4 +1,12 @@
|
|||||||
package freedom
|
package freedom
|
||||||
|
|
||||||
|
type DomainStrategy int
|
||||||
|
|
||||||
|
const (
|
||||||
|
DomainStrategyAsIs = DomainStrategy(0)
|
||||||
|
DomainStrategyUseIP = DomainStrategy(1)
|
||||||
|
)
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
DomainStrategy DomainStrategy
|
||||||
}
|
}
|
||||||
|
@ -3,12 +3,35 @@
|
|||||||
package freedom
|
package freedom
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
"github.com/v2ray/v2ray-core/proxy/internal/config"
|
"github.com/v2ray/v2ray-core/proxy/internal/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func (this *Config) UnmarshalJSON(data []byte) error {
|
||||||
|
type JsonConfig struct {
|
||||||
|
DomainStrategy string `json:"domainStrategy"`
|
||||||
|
}
|
||||||
|
jsonConfig := new(JsonConfig)
|
||||||
|
if err := json.Unmarshal(data, jsonConfig); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
this.DomainStrategy = DomainStrategyAsIs
|
||||||
|
domainStrategy := serial.StringLiteral(jsonConfig.DomainStrategy).ToLower()
|
||||||
|
if domainStrategy.String() == "useip" {
|
||||||
|
this.DomainStrategy = DomainStrategyUseIP
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
config.RegisterOutboundConfig("freedom",
|
config.RegisterOutboundConfig("freedom",
|
||||||
func(data []byte) (interface{}, error) {
|
func(data []byte) (interface{}, error) {
|
||||||
return new(Config), nil
|
c := new(Config)
|
||||||
|
if err := json.Unmarshal(data, c); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return c, nil
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -5,16 +5,64 @@ import (
|
|||||||
"net"
|
"net"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/v2ray/v2ray-core/app"
|
||||||
|
"github.com/v2ray/v2ray-core/app/dns"
|
||||||
"github.com/v2ray/v2ray-core/common/alloc"
|
"github.com/v2ray/v2ray-core/common/alloc"
|
||||||
|
"github.com/v2ray/v2ray-core/common/dice"
|
||||||
v2io "github.com/v2ray/v2ray-core/common/io"
|
v2io "github.com/v2ray/v2ray-core/common/io"
|
||||||
"github.com/v2ray/v2ray-core/common/log"
|
"github.com/v2ray/v2ray-core/common/log"
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
"github.com/v2ray/v2ray-core/common/retry"
|
"github.com/v2ray/v2ray-core/common/retry"
|
||||||
|
"github.com/v2ray/v2ray-core/proxy"
|
||||||
|
"github.com/v2ray/v2ray-core/proxy/internal"
|
||||||
"github.com/v2ray/v2ray-core/transport/dialer"
|
"github.com/v2ray/v2ray-core/transport/dialer"
|
||||||
"github.com/v2ray/v2ray-core/transport/ray"
|
"github.com/v2ray/v2ray-core/transport/ray"
|
||||||
)
|
)
|
||||||
|
|
||||||
type FreedomConnection struct {
|
type FreedomConnection struct {
|
||||||
|
domainStrategy DomainStrategy
|
||||||
|
dns dns.Server
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewFreedomConnection(config *Config, space app.Space) *FreedomConnection {
|
||||||
|
f := &FreedomConnection{
|
||||||
|
domainStrategy: config.DomainStrategy,
|
||||||
|
}
|
||||||
|
log.Info("Freedom: Domain strategy: ", f.domainStrategy)
|
||||||
|
space.InitializeApplication(func() error {
|
||||||
|
if config.DomainStrategy == DomainStrategyUseIP {
|
||||||
|
if !space.HasApp(dns.APP_ID) {
|
||||||
|
log.Error("Freedom: DNS server is not found in the space.")
|
||||||
|
return app.ErrorMissingApplication
|
||||||
|
}
|
||||||
|
f.dns = space.GetApp(dns.APP_ID).(dns.Server)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return f
|
||||||
|
}
|
||||||
|
|
||||||
|
// @Private
|
||||||
|
func (this *FreedomConnection) ResolveIP(destination v2net.Destination) v2net.Destination {
|
||||||
|
if !destination.Address().IsDomain() {
|
||||||
|
return destination
|
||||||
|
}
|
||||||
|
|
||||||
|
ips := this.dns.Get(destination.Address().Domain())
|
||||||
|
if len(ips) == 0 {
|
||||||
|
log.Info("Freedom: DNS returns nil answer. Keep domain as is.")
|
||||||
|
return destination
|
||||||
|
}
|
||||||
|
|
||||||
|
ip := ips[dice.Roll(len(ips))]
|
||||||
|
var newDest v2net.Destination
|
||||||
|
if destination.IsTCP() {
|
||||||
|
newDest = v2net.TCPDestination(v2net.IPAddress(ip), destination.Port())
|
||||||
|
} else {
|
||||||
|
newDest = v2net.UDPDestination(v2net.IPAddress(ip), destination.Port())
|
||||||
|
}
|
||||||
|
log.Info("Freedom: Changing destination from ", destination, " to ", newDest)
|
||||||
|
return newDest
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *FreedomConnection) Dispatch(destination v2net.Destination, payload *alloc.Buffer, ray ray.OutboundRay) error {
|
func (this *FreedomConnection) Dispatch(destination v2net.Destination, payload *alloc.Buffer, ray ray.OutboundRay) error {
|
||||||
@ -25,6 +73,9 @@ func (this *FreedomConnection) Dispatch(destination v2net.Destination, payload *
|
|||||||
defer ray.OutboundOutput().Close()
|
defer ray.OutboundOutput().Close()
|
||||||
|
|
||||||
var conn net.Conn
|
var conn net.Conn
|
||||||
|
if this.domainStrategy == DomainStrategyUseIP && destination.Address().IsDomain() {
|
||||||
|
destination = this.ResolveIP(destination)
|
||||||
|
}
|
||||||
err := retry.Timed(5, 100).On(func() error {
|
err := retry.Timed(5, 100).On(func() error {
|
||||||
rawConn, err := dialer.Dial(destination)
|
rawConn, err := dialer.Dial(destination)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -79,3 +130,10 @@ func (this *FreedomConnection) Dispatch(destination v2net.Destination, payload *
|
|||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
internal.MustRegisterOutboundHandlerCreator("freedom",
|
||||||
|
func(space app.Space, config interface{}) (proxy.OutboundHandler, error) {
|
||||||
|
return NewFreedomConnection(config.(*Config), space), nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
@ -1,11 +1,20 @@
|
|||||||
package freedom_test
|
package freedom_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"net"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/v2ray/v2ray-core/app"
|
||||||
|
"github.com/v2ray/v2ray-core/app/dispatcher"
|
||||||
|
dispatchers "github.com/v2ray/v2ray-core/app/dispatcher/impl"
|
||||||
|
"github.com/v2ray/v2ray-core/app/dns"
|
||||||
|
"github.com/v2ray/v2ray-core/app/proxyman"
|
||||||
|
"github.com/v2ray/v2ray-core/app/router"
|
||||||
|
"github.com/v2ray/v2ray-core/app/router/rules"
|
||||||
"github.com/v2ray/v2ray-core/common/alloc"
|
"github.com/v2ray/v2ray-core/common/alloc"
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
v2nettesting "github.com/v2ray/v2ray-core/common/net/testing"
|
v2nettesting "github.com/v2ray/v2ray-core/common/net/testing"
|
||||||
|
netassert "github.com/v2ray/v2ray-core/common/net/testing/assert"
|
||||||
. "github.com/v2ray/v2ray-core/proxy/freedom"
|
. "github.com/v2ray/v2ray-core/proxy/freedom"
|
||||||
v2testing "github.com/v2ray/v2ray-core/testing"
|
v2testing "github.com/v2ray/v2ray-core/testing"
|
||||||
"github.com/v2ray/v2ray-core/testing/assert"
|
"github.com/v2ray/v2ray-core/testing/assert"
|
||||||
@ -29,7 +38,10 @@ func TestSinglePacket(t *testing.T) {
|
|||||||
_, err := tcpServer.Start()
|
_, err := tcpServer.Start()
|
||||||
assert.Error(err).IsNil()
|
assert.Error(err).IsNil()
|
||||||
|
|
||||||
freedom := &FreedomConnection{}
|
space := app.NewSpace()
|
||||||
|
freedom := NewFreedomConnection(&Config{}, space)
|
||||||
|
space.Initialize()
|
||||||
|
|
||||||
traffic := ray.NewRay()
|
traffic := ray.NewRay()
|
||||||
data2Send := "Data to be sent to remote"
|
data2Send := "Data to be sent to remote"
|
||||||
payload := alloc.NewSmallBuffer().Clear().Append([]byte(data2Send))
|
payload := alloc.NewSmallBuffer().Clear().Append([]byte(data2Send))
|
||||||
@ -47,7 +59,7 @@ func TestSinglePacket(t *testing.T) {
|
|||||||
func TestUnreachableDestination(t *testing.T) {
|
func TestUnreachableDestination(t *testing.T) {
|
||||||
v2testing.Current(t)
|
v2testing.Current(t)
|
||||||
|
|
||||||
freedom := &FreedomConnection{}
|
freedom := NewFreedomConnection(&Config{}, app.NewSpace())
|
||||||
traffic := ray.NewRay()
|
traffic := ray.NewRay()
|
||||||
data2Send := "Data to be sent to remote"
|
data2Send := "Data to be sent to remote"
|
||||||
payload := alloc.NewSmallBuffer().Clear().Append([]byte(data2Send))
|
payload := alloc.NewSmallBuffer().Clear().Append([]byte(data2Send))
|
||||||
@ -55,3 +67,27 @@ func TestUnreachableDestination(t *testing.T) {
|
|||||||
err := freedom.Dispatch(v2net.TCPDestination(v2net.IPAddress([]byte{127, 0, 0, 1}), 128), payload, traffic)
|
err := freedom.Dispatch(v2net.TCPDestination(v2net.IPAddress([]byte{127, 0, 0, 1}), 128), payload, traffic)
|
||||||
assert.Error(err).IsNotNil()
|
assert.Error(err).IsNotNil()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestIPResolution(t *testing.T) {
|
||||||
|
v2testing.Current(t)
|
||||||
|
|
||||||
|
space := app.NewSpace()
|
||||||
|
space.BindApp(proxyman.APP_ID_OUTBOUND_MANAGER, proxyman.NewDefaultOutboundHandlerManager())
|
||||||
|
space.BindApp(dispatcher.APP_ID, dispatchers.NewDefaultDispatcher(space))
|
||||||
|
r, _ := router.CreateRouter("rules", &rules.RouterRuleConfig{}, space)
|
||||||
|
space.BindApp(router.APP_ID, r)
|
||||||
|
dnsServer := dns.NewCacheServer(space, &dns.Config{
|
||||||
|
Hosts: map[string]net.IP{
|
||||||
|
"v2ray.com": net.IP([]byte{127, 0, 0, 1}),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
space.BindApp(dns.APP_ID, dnsServer)
|
||||||
|
|
||||||
|
freedom := NewFreedomConnection(&Config{DomainStrategy: DomainStrategyUseIP}, space)
|
||||||
|
|
||||||
|
space.Initialize()
|
||||||
|
|
||||||
|
ipDest := freedom.ResolveIP(v2net.TCPDestination(v2net.DomainAddress("v2ray.com"), v2net.Port(80)))
|
||||||
|
netassert.Destination(ipDest).IsTCP()
|
||||||
|
netassert.Address(ipDest.Address()).Equals(v2net.LocalHostIP)
|
||||||
|
}
|
||||||
|
@ -1,14 +0,0 @@
|
|||||||
package freedom
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/v2ray/v2ray-core/app"
|
|
||||||
"github.com/v2ray/v2ray-core/proxy"
|
|
||||||
"github.com/v2ray/v2ray-core/proxy/internal"
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
internal.MustRegisterOutboundHandlerCreator("freedom",
|
|
||||||
func(space app.Space, config interface{}) (proxy.OutboundHandler, error) {
|
|
||||||
return &FreedomConnection{}, nil
|
|
||||||
})
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user