handle inbound detour in vmess inbound

pull/73/head
v2ray 2016-01-19 23:41:40 +01:00
parent 573d0a9c29
commit b0adb24003
13 changed files with 163 additions and 75 deletions

View File

@ -10,6 +10,7 @@ import (
"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"
) )
type DokodemoDoor struct { type DokodemoDoor struct {
@ -22,6 +23,7 @@ type DokodemoDoor struct {
space app.Space space app.Space
tcpListener *net.TCPListener tcpListener *net.TCPListener
udpConn *net.UDPConn udpConn *net.UDPConn
listeningPort v2net.Port
} }
func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor { func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor {
@ -33,6 +35,10 @@ func NewDokodemoDoor(space app.Space, config *Config) *DokodemoDoor {
} }
} }
func (this *DokodemoDoor) Port() v2net.Port {
return this.listeningPort
}
func (this *DokodemoDoor) Close() { func (this *DokodemoDoor) Close() {
this.accepting = false this.accepting = false
if this.tcpListener != nil { if this.tcpListener != nil {
@ -50,6 +56,14 @@ func (this *DokodemoDoor) Close() {
} }
func (this *DokodemoDoor) Listen(port v2net.Port) error { func (this *DokodemoDoor) Listen(port v2net.Port) error {
if this.accepting {
if this.listeningPort == port {
return nil
} else {
return proxy.ErrorAlreadyListening
}
}
this.listeningPort = port
this.accepting = true this.accepting = true
if this.config.Network.HasNetwork(v2net.TCPNetwork) { if this.config.Network.HasNetwork(v2net.TCPNetwork) {

View File

@ -7,4 +7,5 @@ import (
var ( var (
InvalidAuthentication = errors.New("Invalid authentication.") InvalidAuthentication = errors.New("Invalid authentication.")
InvalidProtocolVersion = errors.New("Invalid protocol version.") InvalidProtocolVersion = errors.New("Invalid protocol version.")
ErrorAlreadyListening = errors.New("Already listening on another port.")
) )

View File

@ -15,6 +15,7 @@ import (
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/common/serial" "github.com/v2ray/v2ray-core/common/serial"
"github.com/v2ray/v2ray-core/proxy"
"github.com/v2ray/v2ray-core/transport/ray" "github.com/v2ray/v2ray-core/transport/ray"
) )
@ -24,6 +25,7 @@ type HttpProxyServer struct {
space app.Space space app.Space
config *Config config *Config
tcpListener *net.TCPListener tcpListener *net.TCPListener
listeningPort v2net.Port
} }
func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer { func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer {
@ -33,6 +35,10 @@ func NewHttpProxyServer(space app.Space, config *Config) *HttpProxyServer {
} }
} }
func (this *HttpProxyServer) Port() v2net.Port {
return this.listeningPort
}
func (this *HttpProxyServer) Close() { func (this *HttpProxyServer) Close() {
this.accepting = false this.accepting = false
if this.tcpListener != nil { if this.tcpListener != nil {
@ -44,6 +50,15 @@ func (this *HttpProxyServer) Close() {
} }
func (this *HttpProxyServer) Listen(port v2net.Port) error { func (this *HttpProxyServer) Listen(port v2net.Port) error {
if this.accepting {
if this.listeningPort == port {
return nil
} else {
return proxy.ErrorAlreadyListening
}
}
this.listeningPort = port
tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{ tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{
Port: int(port.Value()), Port: int(port.Value()),
IP: []byte{0, 0, 0, 0}, IP: []byte{0, 0, 0, 0},

View File

@ -13,6 +13,8 @@ type InboundConnectionHandler interface {
Listen(port v2net.Port) error Listen(port v2net.Port) error
// Close stops the handler to accepting anymore inbound connections. // Close stops the handler to accepting anymore inbound connections.
Close() Close()
// Port returns the port that the handler is listening on.
Port() v2net.Port
} }
// An OutboundConnectionHandler handles outbound network connection for V2Ray. // An OutboundConnectionHandler handles outbound network connection for V2Ray.

View File

@ -31,6 +31,7 @@ type SocksServer struct {
tcpListener *net.TCPListener tcpListener *net.TCPListener
udpConn *net.UDPConn udpConn *net.UDPConn
udpAddress v2net.Destination udpAddress v2net.Destination
listeningPort v2net.Port
} }
func NewSocksServer(space app.Space, config *Config) *SocksServer { func NewSocksServer(space app.Space, config *Config) *SocksServer {
@ -40,6 +41,10 @@ func NewSocksServer(space app.Space, config *Config) *SocksServer {
} }
} }
func (this *SocksServer) Port() v2net.Port {
return this.listeningPort
}
func (this *SocksServer) Close() { func (this *SocksServer) Close() {
this.accepting = false this.accepting = false
if this.tcpListener != nil { if this.tcpListener != nil {
@ -57,6 +62,15 @@ func (this *SocksServer) Close() {
} }
func (this *SocksServer) Listen(port v2net.Port) error { func (this *SocksServer) Listen(port v2net.Port) error {
if this.accepting {
if this.listeningPort == port {
return nil
} else {
return proxy.ErrorAlreadyListening
}
}
this.listeningPort = port
listener, err := net.ListenTCP("tcp", &net.TCPAddr{ listener, err := net.ListenTCP("tcp", &net.TCPAddr{
IP: []byte{0, 0, 0, 0}, IP: []byte{0, 0, 0, 0},
Port: int(port), Port: int(port),

View File

@ -9,17 +9,21 @@ import (
) )
type InboundConnectionHandler struct { type InboundConnectionHandler struct {
Port v2net.Port port v2net.Port
Space app.Space Space app.Space
ConnInput io.Reader ConnInput io.Reader
ConnOutput io.Writer ConnOutput io.Writer
} }
func (this *InboundConnectionHandler) Listen(port v2net.Port) error { func (this *InboundConnectionHandler) Listen(port v2net.Port) error {
this.Port = port this.port = port
return nil return nil
} }
func (this *InboundConnectionHandler) Port() v2net.Port {
return this.port
}
func (this *InboundConnectionHandler) Close() { func (this *InboundConnectionHandler) Close() {
} }

View File

@ -2,7 +2,6 @@ package command
import ( import (
"io" "io"
"time"
v2net "github.com/v2ray/v2ray-core/common/net" v2net "github.com/v2ray/v2ray-core/common/net"
"github.com/v2ray/v2ray-core/common/serial" "github.com/v2ray/v2ray-core/common/serial"
@ -26,40 +25,29 @@ type SwitchAccount struct {
Port v2net.Port Port v2net.Port
ID *uuid.UUID ID *uuid.UUID
AlterIds serial.Uint16Literal AlterIds serial.Uint16Literal
ValidUntil time.Time ValidSec serial.Uint16Literal
} }
func (this *SwitchAccount) Marshal(writer io.Writer) (int, error) { func (this *SwitchAccount) Marshal(writer io.Writer) {
outBytes := 0
hostStr := "" hostStr := ""
if this.Host != nil { if this.Host != nil {
hostStr = this.Host.String() hostStr = this.Host.String()
} }
writer.Write([]byte{byte(len(hostStr))}) writer.Write([]byte{byte(len(hostStr))})
outBytes++
if len(hostStr) > 0 { if len(hostStr) > 0 {
writer.Write([]byte(hostStr)) writer.Write([]byte(hostStr))
outBytes += len(hostStr)
} }
writer.Write(this.Port.Bytes()) writer.Write(this.Port.Bytes())
outBytes += 2
idBytes := this.ID.Bytes() idBytes := this.ID.Bytes()
writer.Write(idBytes) writer.Write(idBytes)
outBytes += len(idBytes)
writer.Write(this.AlterIds.Bytes()) writer.Write(this.AlterIds.Bytes())
outBytes += 2
timestamp := this.ValidUntil.Unix()
timeBytes := serial.Int64Literal(timestamp).Bytes()
timeBytes := this.ValidSec.Bytes()
writer.Write(timeBytes) writer.Write(timeBytes)
outBytes += len(timeBytes)
return outBytes, nil
} }
func (this *SwitchAccount) Unmarshal(data []byte) error { func (this *SwitchAccount) Unmarshal(data []byte) error {
@ -84,9 +72,9 @@ func (this *SwitchAccount) Unmarshal(data []byte) error {
} }
this.AlterIds = serial.ParseUint16(data[alterIdStart : alterIdStart+2]) this.AlterIds = serial.ParseUint16(data[alterIdStart : alterIdStart+2])
timeStart := alterIdStart + 2 timeStart := alterIdStart + 2
if len(data) < timeStart+8 { if len(data) < timeStart+2 {
return transport.CorruptedPacket return transport.CorruptedPacket
} }
this.ValidUntil = time.Unix(serial.BytesLiteral(data[timeStart:timeStart+8]).Int64Value(), 0) this.ValidSec = serial.ParseUint16(data[timeStart : timeStart+2])
return nil return nil
} }

View File

@ -3,7 +3,6 @@ package command_test
import ( import (
"bytes" "bytes"
"testing" "testing"
"time"
netassert "github.com/v2ray/v2ray-core/common/net/testing/assert" netassert "github.com/v2ray/v2ray-core/common/net/testing/assert"
"github.com/v2ray/v2ray-core/common/uuid" "github.com/v2ray/v2ray-core/common/uuid"
@ -19,16 +18,14 @@ func TestSwitchAccount(t *testing.T) {
Port: 1234, Port: 1234,
ID: uuid.New(), ID: uuid.New(),
AlterIds: 1024, AlterIds: 1024,
ValidUntil: time.Now(), ValidSec: 8080,
} }
cmd, err := CreateResponseCommand(1) cmd, err := CreateResponseCommand(1)
assert.Error(err).IsNil() assert.Error(err).IsNil()
buffer := bytes.NewBuffer(make([]byte, 0, 1024)) buffer := bytes.NewBuffer(make([]byte, 0, 1024))
nBytes, err := sa.Marshal(buffer) sa.Marshal(buffer)
assert.Error(err).IsNil()
assert.Int(nBytes).Equals(buffer.Len())
cmd.Unmarshal(buffer.Bytes()) cmd.Unmarshal(buffer.Bytes())
sa2, ok := cmd.(*SwitchAccount) sa2, ok := cmd.(*SwitchAccount)
@ -36,5 +33,5 @@ func TestSwitchAccount(t *testing.T) {
netassert.Port(sa.Port).Equals(sa2.Port) netassert.Port(sa.Port).Equals(sa2.Port)
assert.String(sa.ID).Equals(sa2.ID.String()) assert.String(sa.ID).Equals(sa2.ID.String())
assert.Uint16(sa.AlterIds.Value()).Equals(sa2.AlterIds.Value()) assert.Uint16(sa.AlterIds.Value()).Equals(sa2.AlterIds.Value())
assert.Int64(sa.ValidUntil.Unix()).Equals(sa2.ValidUntil.Unix()) assert.Uint16(sa.ValidSec.Value()).Equals(sa2.ValidSec.Value())
} }

View File

@ -10,7 +10,7 @@ var (
) )
type Command interface { type Command interface {
Marshal(io.Writer) (int, error) Marshal(io.Writer)
Unmarshal([]byte) error Unmarshal([]byte) error
} }

View File

@ -26,20 +26,16 @@ type CacheDns struct {
Address v2net.Address Address v2net.Address
} }
func (this *CacheDns) Marshal(writer io.Writer) (int, error) { func (this *CacheDns) Marshal(writer io.Writer) {
if this.Address.IsIPv4() { if this.Address.IsIPv4() {
writer.Write([]byte{typeIPv4}) writer.Write([]byte{typeIPv4})
writer.Write(this.Address.IP()) writer.Write(this.Address.IP())
return 5, nil
} }
if this.Address.IsIPv6() { if this.Address.IsIPv6() {
writer.Write([]byte{typeIPv6}) writer.Write([]byte{typeIPv6})
writer.Write(this.Address.IP()) writer.Write(this.Address.IP())
return 17, nil
} }
return 0, ErrDomainAddress
} }
func (this *CacheDns) Unmarshal(data []byte) error { func (this *CacheDns) Unmarshal(data []byte) error {

View File

@ -21,12 +21,10 @@ func TestCacheDnsIPv4(t *testing.T) {
buffer := alloc.NewBuffer().Clear() buffer := alloc.NewBuffer().Clear()
defer buffer.Release() defer buffer.Release()
nBytes, err := cd.Marshal(buffer) cd.Marshal(buffer)
assert.Error(err).IsNil()
assert.Int(nBytes).Equals(buffer.Len())
cd2 := &CacheDns{} cd2 := &CacheDns{}
err = cd2.Unmarshal(buffer.Value) err := cd2.Unmarshal(buffer.Value)
assert.Error(err).IsNil() assert.Error(err).IsNil()
netassert.Address(cd.Address).Equals(cd2.Address) netassert.Address(cd.Address).Equals(cd2.Address)
} }

View File

@ -0,0 +1,45 @@
package inbound
import (
"github.com/v2ray/v2ray-core/common/alloc"
"github.com/v2ray/v2ray-core/common/serial"
"github.com/v2ray/v2ray-core/proxy/vmess/command"
)
func (this *VMessInboundHandler) generateCommand(buffer *alloc.Buffer) {
cmd := byte(0)
commandBytes := alloc.NewSmallBuffer().Clear()
defer commandBytes.Release()
if this.features != nil && this.features.Detour != nil {
tag := this.features.Detour.ToTag
if this.space.HasInboundHandlerManager() {
handlerManager := this.space.InboundHandlerManager()
handler, availableSec := handlerManager.GetHandler(tag)
inboundHandler, ok := handler.(*VMessInboundHandler)
if ok {
user := inboundHandler.GetUser()
availableSecUint16 := uint16(65535)
if availableSec < 65535 {
availableSecUint16 = uint16(availableSec)
}
saCmd := &command.SwitchAccount{
Port: inboundHandler.Port(),
ID: user.ID.UUID(),
AlterIds: serial.Uint16Literal(len(user.AlterIDs)),
ValidSec: serial.Uint16Literal(availableSecUint16),
}
saCmd.Marshal(commandBytes)
}
}
}
if commandBytes.Len() > 256 {
buffer.AppendBytes(byte(0), byte(0))
} else {
buffer.AppendBytes(cmd, byte(commandBytes.Len()))
buffer.Append(commandBytes.Value)
}
}

View File

@ -24,15 +24,15 @@ type VMessInboundHandler struct {
sync.Mutex sync.Mutex
space app.Space space app.Space
clients protocol.UserSet clients protocol.UserSet
user *vmess.User
accepting bool accepting bool
listener *net.TCPListener listener *net.TCPListener
features *FeaturesConfig
listeningPort v2net.Port
} }
func NewVMessInboundHandler(space app.Space, clients protocol.UserSet) *VMessInboundHandler { func (this *VMessInboundHandler) Port() v2net.Port {
return &VMessInboundHandler{ return this.listeningPort
space: space,
clients: clients,
}
} }
func (this *VMessInboundHandler) Close() { func (this *VMessInboundHandler) Close() {
@ -45,11 +45,20 @@ func (this *VMessInboundHandler) Close() {
} }
} }
func (this *VMessInboundHandler) AddUser(user vmess.User) { func (this *VMessInboundHandler) GetUser() *vmess.User {
return this.user
} }
func (this *VMessInboundHandler) Listen(port v2net.Port) error { func (this *VMessInboundHandler) Listen(port v2net.Port) error {
if this.accepting {
if this.listeningPort == port {
return nil
} else {
return proxy.ErrorAlreadyListening
}
}
this.listeningPort = port
listener, err := net.ListenTCP("tcp", &net.TCPAddr{ listener, err := net.ListenTCP("tcp", &net.TCPAddr{
IP: []byte{0, 0, 0, 0}, IP: []byte{0, 0, 0, 0},
Port: int(port), Port: int(port),
@ -175,6 +184,11 @@ func init() {
allowedClients.AddUser(user) allowedClients.AddUser(user)
} }
return NewVMessInboundHandler(space, allowedClients), nil return &VMessInboundHandler{
space: space,
clients: allowedClients,
features: config.Features,
user: config.AllowedUsers[0],
}, nil
}) })
} }