1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-02 07:26:24 -05:00

refine stream handling

This commit is contained in:
Darien Raymond 2017-01-04 12:34:01 +01:00
parent 723207158f
commit 49210d8362
No known key found for this signature in database
GPG Key ID: 7251FFA14BB18169
11 changed files with 154 additions and 55 deletions

View File

@ -1,6 +1,8 @@
package impl package impl
import ( import (
"time"
"v2ray.com/core/app" "v2ray.com/core/app"
"v2ray.com/core/app/dispatcher" "v2ray.com/core/app/dispatcher"
"v2ray.com/core/app/proxyman" "v2ray.com/core/app/proxyman"
@ -48,7 +50,6 @@ func (v *DefaultDispatcher) Release() {
} }
func (v *DefaultDispatcher) DispatchToOutbound(session *proxy.SessionInfo) ray.InboundRay { func (v *DefaultDispatcher) DispatchToOutbound(session *proxy.SessionInfo) ray.InboundRay {
direct := ray.NewRay()
dispatcher := v.ohm.GetDefaultHandler() dispatcher := v.ohm.GetDefaultHandler()
destination := session.Destination destination := session.Destination
@ -65,26 +66,32 @@ func (v *DefaultDispatcher) DispatchToOutbound(session *proxy.SessionInfo) ray.I
} }
} }
direct := ray.NewRay()
var waitFunc func() error
if session.Inbound != nil && session.Inbound.AllowPassiveConnection { if session.Inbound != nil && session.Inbound.AllowPassiveConnection {
go dispatcher.Dispatch(destination, buf.NewLocal(32), direct) waitFunc = noOpWait()
} else { } else {
go v.FilterPacketAndDispatch(destination, direct, dispatcher) wdi := &waitDataInspector{
hasData: make(chan bool, 1),
}
direct.AddInspector(wdi)
waitFunc = waitForData(wdi)
} }
go v.waitAndDispatch(waitFunc, destination, direct, dispatcher)
return direct return direct
} }
// FilterPacketAndDispatch waits for a payload from source and starts dispatching. func (v *DefaultDispatcher) waitAndDispatch(wait func() error, destination v2net.Destination, link ray.OutboundRay, dispatcher proxy.OutboundHandler) {
// Private: Visible for testing. if err := wait(); err != nil {
func (v *DefaultDispatcher) FilterPacketAndDispatch(destination v2net.Destination, link ray.OutboundRay, dispatcher proxy.OutboundHandler) { log.Info("DefaultDispatcher: Failed precondition: ", err)
payload, err := link.OutboundInput().Read() link.OutboundInput().ForceClose()
if err != nil { link.OutboundOutput().Close()
log.Info("DefaultDispatcher: No payload towards ", destination, ", stopping now.")
link.OutboundInput().Release()
link.OutboundOutput().Release()
return return
} }
dispatcher.Dispatch(destination, payload, link)
dispatcher.Dispatch(destination, link)
} }
type DefaultDispatcherFactory struct{} type DefaultDispatcherFactory struct{}
@ -100,3 +107,38 @@ func (v DefaultDispatcherFactory) AppId() app.ID {
func init() { func init() {
app.RegisterApplicationFactory(serial.GetMessageType(new(dispatcher.Config)), DefaultDispatcherFactory{}) app.RegisterApplicationFactory(serial.GetMessageType(new(dispatcher.Config)), DefaultDispatcherFactory{})
} }
type waitDataInspector struct {
hasData chan bool
}
func (wdi *waitDataInspector) Input(*buf.Buffer) {
select {
case wdi.hasData <- true:
default:
}
}
func (wdi *waitDataInspector) WaitForData() bool {
select {
case <-wdi.hasData:
return true
case <-time.After(time.Minute):
return false
}
}
func waitForData(wdi *waitDataInspector) func() error {
return func() error {
if wdi.WaitForData() {
return nil
}
return errors.New("DefaultDispatcher: No data.")
}
}
func noOpWait() func() error {
return func() error {
return nil
}
}

View File

@ -50,7 +50,7 @@ func (v *OutboundProxy) Dial(src v2net.Address, dest v2net.Destination, options
} }
log.Info("Proxy: Dialing to ", dest) log.Info("Proxy: Dialing to ", dest)
stream := ray.NewRay() stream := ray.NewRay()
go handler.Dispatch(dest, nil, stream) go handler.Dispatch(dest, stream)
return NewConnection(src, dest, stream), nil return NewConnection(src, dest, stream), nil
} }

View File

@ -3,7 +3,6 @@ package blackhole
import ( import (
"v2ray.com/core/app" "v2ray.com/core/app"
"v2ray.com/core/common/buf"
v2net "v2ray.com/core/common/net" v2net "v2ray.com/core/common/net"
"v2ray.com/core/proxy" "v2ray.com/core/proxy"
"v2ray.com/core/transport/ray" "v2ray.com/core/transport/ray"
@ -28,9 +27,7 @@ func New(space app.Space, config *Config, meta *proxy.OutboundHandlerMeta) (prox
} }
// Dispatch implements OutboundHandler.Dispatch(). // Dispatch implements OutboundHandler.Dispatch().
func (v *Handler) Dispatch(destination v2net.Destination, payload *buf.Buffer, ray ray.OutboundRay) { func (v *Handler) Dispatch(destination v2net.Destination, ray ray.OutboundRay) {
payload.Release()
v.response.WriteTo(ray.OutboundOutput()) v.response.WriteTo(ray.OutboundOutput())
ray.OutboundOutput().Close() ray.OutboundOutput().Close()

View File

@ -67,10 +67,9 @@ func (v *Handler) ResolveIP(destination v2net.Destination) v2net.Destination {
return newDest return newDest
} }
func (v *Handler) Dispatch(destination v2net.Destination, payload *buf.Buffer, ray ray.OutboundRay) { func (v *Handler) Dispatch(destination v2net.Destination, ray ray.OutboundRay) {
log.Info("Freedom: Opening connection to ", destination) log.Info("Freedom: Opening connection to ", destination)
defer payload.Release()
input := ray.OutboundInput() input := ray.OutboundInput()
output := ray.OutboundOutput() output := ray.OutboundOutput()
defer input.ForceClose() defer input.ForceClose()
@ -96,13 +95,6 @@ func (v *Handler) Dispatch(destination v2net.Destination, payload *buf.Buffer, r
conn.SetReusable(false) conn.SetReusable(false)
if !payload.IsEmpty() {
if _, err := conn.Write(payload.Bytes()); err != nil {
log.Warning("Freedom: Failed to write to destination: ", destination, ": ", err)
return
}
}
requestDone := signal.ExecuteAsync(func() error { requestDone := signal.ExecuteAsync(func() error {
defer input.ForceClose() defer input.ForceClose()

View File

@ -53,9 +53,10 @@ func TestSinglePacket(t *testing.T) {
data2Send := "Data to be sent to remote" data2Send := "Data to be sent to remote"
payload := buf.NewLocal(2048) payload := buf.NewLocal(2048)
payload.Append([]byte(data2Send)) payload.Append([]byte(data2Send))
traffic.InboundInput().Write(payload)
fmt.Println(tcpServerAddr.Network, tcpServerAddr.Address, tcpServerAddr.Port) fmt.Println(tcpServerAddr.Network, tcpServerAddr.Address, tcpServerAddr.Port)
go freedom.Dispatch(tcpServerAddr, payload, traffic) go freedom.Dispatch(tcpServerAddr, traffic)
traffic.InboundInput().Close() traffic.InboundInput().Close()
respPayload, err := traffic.InboundOutput().Read() respPayload, err := traffic.InboundOutput().Read()

View File

@ -2,7 +2,6 @@
package proxy package proxy
import ( import (
"v2ray.com/core/common/buf"
v2net "v2ray.com/core/common/net" v2net "v2ray.com/core/common/net"
"v2ray.com/core/common/protocol" "v2ray.com/core/common/protocol"
"v2ray.com/core/transport/internet" "v2ray.com/core/transport/internet"
@ -58,5 +57,5 @@ type InboundHandler interface {
// An OutboundHandler handles outbound network connection for V2Ray. // An OutboundHandler handles outbound network connection for V2Ray.
type OutboundHandler interface { type OutboundHandler interface {
// Dispatch sends one or more Packets to its destination. // Dispatch sends one or more Packets to its destination.
Dispatch(destination v2net.Destination, payload *buf.Buffer, ray ray.OutboundRay) Dispatch(destination v2net.Destination, ray ray.OutboundRay)
} }

View File

@ -35,9 +35,7 @@ func NewClient(config *ClientConfig, space app.Space, meta *proxy.OutboundHandle
} }
// Dispatch implements OutboundHandler.Dispatch(). // Dispatch implements OutboundHandler.Dispatch().
func (v *Client) Dispatch(destination v2net.Destination, payload *buf.Buffer, ray ray.OutboundRay) { func (v *Client) Dispatch(destination v2net.Destination, ray ray.OutboundRay) {
defer payload.Release()
network := destination.Network network := destination.Network
var server *protocol.ServerSpec var server *protocol.ServerSpec
@ -99,13 +97,6 @@ func (v *Client) Dispatch(destination v2net.Destination, payload *buf.Buffer, ra
return return
} }
if !payload.IsEmpty() {
if err := bodyWriter.Write(payload); err != nil {
log.Info("Shadowsocks|Client: Failed to write payload: ", err)
return
}
}
bufferedWriter.SetBuffered(false) bufferedWriter.SetBuffered(false)
requestDone := signal.ExecuteAsync(func() error { requestDone := signal.ExecuteAsync(func() error {
@ -143,12 +134,6 @@ func (v *Client) Dispatch(destination v2net.Destination, payload *buf.Buffer, ra
Writer: conn, Writer: conn,
Request: request, Request: request,
} }
if !payload.IsEmpty() {
if err := writer.Write(payload); err != nil {
log.Info("Shadowsocks|Client: Failed to write payload: ", err)
return
}
}
requestDone := signal.ExecuteAsync(func() error { requestDone := signal.ExecuteAsync(func() error {
defer ray.OutboundInput().ForceClose() defer ray.OutboundInput().ForceClose()

View File

@ -1,10 +1,13 @@
package outbound package outbound
import ( import (
"time"
"v2ray.com/core/app" "v2ray.com/core/app"
"v2ray.com/core/common" "v2ray.com/core/common"
"v2ray.com/core/common/buf" "v2ray.com/core/common/buf"
"v2ray.com/core/common/bufio" "v2ray.com/core/common/bufio"
"v2ray.com/core/common/errors"
"v2ray.com/core/common/log" "v2ray.com/core/common/log"
v2net "v2ray.com/core/common/net" v2net "v2ray.com/core/common/net"
"v2ray.com/core/common/protocol" "v2ray.com/core/common/protocol"
@ -26,10 +29,9 @@ type VMessOutboundHandler struct {
} }
// Dispatch implements OutboundHandler.Dispatch(). // Dispatch implements OutboundHandler.Dispatch().
func (v *VMessOutboundHandler) Dispatch(target v2net.Destination, payload *buf.Buffer, ray ray.OutboundRay) { func (v *VMessOutboundHandler) Dispatch(target v2net.Destination, outboundRay ray.OutboundRay) {
defer payload.Release() defer outboundRay.OutboundInput().ForceClose()
defer ray.OutboundInput().ForceClose() defer outboundRay.OutboundOutput().Close()
defer ray.OutboundOutput().Close()
var rec *protocol.ServerSpec var rec *protocol.ServerSpec
var conn internet.Connection var conn internet.Connection
@ -77,8 +79,8 @@ func (v *VMessOutboundHandler) Dispatch(target v2net.Destination, payload *buf.B
request.Option.Set(protocol.RequestOptionConnectionReuse) request.Option.Set(protocol.RequestOptionConnectionReuse)
} }
input := ray.OutboundInput() input := outboundRay.OutboundInput()
output := ray.OutboundOutput() output := outboundRay.OutboundOutput()
session := encoding.NewClientSession(protocol.DefaultIDHash) session := encoding.NewClientSession(protocol.DefaultIDHash)
@ -93,11 +95,17 @@ func (v *VMessOutboundHandler) Dispatch(target v2net.Destination, payload *buf.B
bodyWriter := session.EncodeRequestBody(request, writer) bodyWriter := session.EncodeRequestBody(request, writer)
defer bodyWriter.Release() defer bodyWriter.Release()
if !payload.IsEmpty() { firstPayload, err := input.ReadTimeout(time.Millisecond * 500)
if err := bodyWriter.Write(payload); err != nil { if err != nil && err != ray.ErrReadTimeout {
return err return errors.Base(err).Message("VMess|Outbound: Failed to get first payload.")
}
} }
if !firstPayload.IsEmpty() {
if err := bodyWriter.Write(firstPayload); err != nil {
return errors.Base(err).Message("VMess|Outbound: Failed to write first payload.")
}
firstPayload.Release()
}
writer.SetBuffered(false) writer.SetBuffered(false)
if err := buf.PipeUntilEOF(input, bodyWriter); err != nil { if err := buf.PipeUntilEOF(input, bodyWriter); err != nil {

View File

@ -1,8 +1,11 @@
package ray package ray
import ( import (
"errors"
"io" "io"
"time"
"v2ray.com/core/common/buf" "v2ray.com/core/common/buf"
) )
@ -10,6 +13,8 @@ const (
bufferSize = 512 bufferSize = 512
) )
var ErrReadTimeout = errors.New("Ray: timeout.")
// NewRay creates a new Ray for direct traffic transport. // NewRay creates a new Ray for direct traffic transport.
func NewRay() Ray { func NewRay() Ray {
return &directRay{ return &directRay{
@ -39,10 +44,19 @@ func (v *directRay) InboundOutput() InputStream {
return v.Output return v.Output
} }
func (v *directRay) AddInspector(inspector Inspector) {
if inspector == nil {
return
}
v.Input.inspector.AddInspector(inspector)
v.Output.inspector.AddInspector(inspector)
}
type Stream struct { type Stream struct {
buffer chan *buf.Buffer buffer chan *buf.Buffer
srcClose chan bool srcClose chan bool
destClose chan bool destClose chan bool
inspector *InspectorChain
} }
func NewStream() *Stream { func NewStream() *Stream {
@ -50,6 +64,7 @@ func NewStream() *Stream {
buffer: make(chan *buf.Buffer, bufferSize), buffer: make(chan *buf.Buffer, bufferSize),
srcClose: make(chan bool), srcClose: make(chan bool),
destClose: make(chan bool), destClose: make(chan bool),
inspector: &InspectorChain{},
} }
} }
@ -71,6 +86,26 @@ func (v *Stream) Read() (*buf.Buffer, error) {
} }
} }
func (v *Stream) ReadTimeout(timeout time.Duration) (*buf.Buffer, error) {
select {
case <-v.destClose:
return nil, io.ErrClosedPipe
case b := <-v.buffer:
return b, nil
default:
select {
case b := <-v.buffer:
return b, nil
case <-v.srcClose:
return nil, io.EOF
case <-v.destClose:
return nil, io.ErrClosedPipe
case <-time.After(timeout):
return nil, ErrReadTimeout
}
}
}
func (v *Stream) Write(data *buf.Buffer) (err error) { func (v *Stream) Write(data *buf.Buffer) (err error) {
if data.IsEmpty() { if data.IsEmpty() {
return return
@ -88,6 +123,7 @@ func (v *Stream) Write(data *buf.Buffer) (err error) {
case <-v.srcClose: case <-v.srcClose:
return io.ErrClosedPipe return io.ErrClosedPipe
case v.buffer <- data: case v.buffer <- data:
v.inspector.Input(data)
return nil return nil
} }
} }

View File

@ -0,0 +1,36 @@
package ray
import (
"sync"
"v2ray.com/core/common/buf"
)
type Inspector interface {
Input(*buf.Buffer)
}
type NoOpInspector struct{}
func (NoOpInspector) Input(*buf.Buffer) {}
type InspectorChain struct {
sync.RWMutex
chain []Inspector
}
func (ic *InspectorChain) AddInspector(inspector Inspector) {
ic.Lock()
defer ic.Unlock()
ic.chain = append(ic.chain, inspector)
}
func (ic *InspectorChain) Input(b *buf.Buffer) {
ic.RLock()
defer ic.RUnlock()
for _, inspector := range ic.chain {
inspector.Input(b)
}
}

View File

@ -1,6 +1,7 @@
package ray package ray
import "v2ray.com/core/common/buf" import "v2ray.com/core/common/buf"
import "time"
// OutboundRay is a transport interface for outbound connections. // OutboundRay is a transport interface for outbound connections.
type OutboundRay interface { type OutboundRay interface {
@ -31,10 +32,12 @@ type InboundRay interface {
type Ray interface { type Ray interface {
InboundRay InboundRay
OutboundRay OutboundRay
AddInspector(Inspector)
} }
type InputStream interface { type InputStream interface {
buf.Reader buf.Reader
ReadTimeout(time.Duration) (*buf.Buffer, error)
ForceClose() ForceClose()
} }