|
|
@@ -0,0 +1,131 @@
|
|
|
+package proxy
|
|
|
+
|
|
|
+import (
|
|
|
+ "errors"
|
|
|
+ "io"
|
|
|
+ "net"
|
|
|
+ "time"
|
|
|
+
|
|
|
+ "v2ray.com/core/app"
|
|
|
+ "v2ray.com/core/app/proxyman"
|
|
|
+ "v2ray.com/core/common/alloc"
|
|
|
+ v2io "v2ray.com/core/common/io"
|
|
|
+ "v2ray.com/core/common/log"
|
|
|
+ v2net "v2ray.com/core/common/net"
|
|
|
+ "v2ray.com/core/transport/internet"
|
|
|
+ "v2ray.com/core/transport/ray"
|
|
|
+)
|
|
|
+
|
|
|
+const (
|
|
|
+ APP_ID = 7
|
|
|
+)
|
|
|
+
|
|
|
+type OutboundProxy struct {
|
|
|
+ outboundManager proxyman.OutboundHandlerManager
|
|
|
+}
|
|
|
+
|
|
|
+func NewOutboundProxy(space app.Space) *OutboundProxy {
|
|
|
+ proxy := new(OutboundProxy)
|
|
|
+ space.InitializeApplication(func() error {
|
|
|
+ if !space.HasApp(proxyman.APP_ID_OUTBOUND_MANAGER) {
|
|
|
+ return errors.New("Proxy: Outbound handler manager not found.")
|
|
|
+ }
|
|
|
+ proxy.outboundManager = space.GetApp(proxyman.APP_ID_OUTBOUND_MANAGER).(proxyman.OutboundHandlerManager)
|
|
|
+ return nil
|
|
|
+ })
|
|
|
+ return proxy
|
|
|
+}
|
|
|
+
|
|
|
+func (this *OutboundProxy) Dial(src v2net.Address, dest v2net.Destination, options internet.DialerOptions) (internet.Connection, error) {
|
|
|
+ handler := this.outboundManager.GetHandler(options.ProxyTag)
|
|
|
+ if handler == nil {
|
|
|
+ log.Warning("Proxy: Failed to get outbound handler with tag: ", options.ProxyTag)
|
|
|
+ return internet.Dial(src, dest, internet.DialerOptions{
|
|
|
+ Stream: options.Stream,
|
|
|
+ })
|
|
|
+ }
|
|
|
+ stream := ray.NewRay()
|
|
|
+ if err := handler.Dispatch(dest, alloc.NewLocalBuffer(32).Clear(), stream); err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ return NewProxyConnection(src, dest, stream), nil
|
|
|
+}
|
|
|
+
|
|
|
+func (this *OutboundProxy) Release() {
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+type ProxyConnection struct {
|
|
|
+ stream ray.Ray
|
|
|
+ closed bool
|
|
|
+ localAddr net.Addr
|
|
|
+ remoteAddr net.Addr
|
|
|
+
|
|
|
+ reader *v2io.ChanReader
|
|
|
+ writer *v2io.ChainWriter
|
|
|
+}
|
|
|
+
|
|
|
+func NewProxyConnection(src v2net.Address, dest v2net.Destination, stream ray.Ray) *ProxyConnection {
|
|
|
+ return &ProxyConnection{
|
|
|
+ stream: stream,
|
|
|
+ localAddr: &net.TCPAddr{
|
|
|
+ IP: []byte{0, 0, 0, 0},
|
|
|
+ Port: 0,
|
|
|
+ },
|
|
|
+ remoteAddr: &net.TCPAddr{
|
|
|
+ IP: []byte{0, 0, 0, 0},
|
|
|
+ Port: 0,
|
|
|
+ },
|
|
|
+ reader: v2io.NewChanReader(stream.InboundOutput()),
|
|
|
+ writer: v2io.NewChainWriter(stream.InboundInput()),
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) Read(b []byte) (int, error) {
|
|
|
+ if this.closed {
|
|
|
+ return 0, io.EOF
|
|
|
+ }
|
|
|
+ return this.reader.Read(b)
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) Write(b []byte) (int, error) {
|
|
|
+ if this.closed {
|
|
|
+ return 0, io.EOF
|
|
|
+ }
|
|
|
+ return this.writer.Write(b)
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) Close() error {
|
|
|
+ this.closed = true
|
|
|
+ this.stream.InboundInput().Close()
|
|
|
+ this.stream.InboundOutput().Release()
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) LocalAddr() net.Addr {
|
|
|
+ return this.localAddr
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) RemoteAddr() net.Addr {
|
|
|
+ return this.remoteAddr
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) SetDeadline(t time.Time) error {
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) SetReadDeadline(t time.Time) error {
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) SetWriteDeadline(t time.Time) error {
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) Reusable() bool {
|
|
|
+ return false
|
|
|
+}
|
|
|
+
|
|
|
+func (this *ProxyConnection) SetReusable(bool) {
|
|
|
+
|
|
|
+}
|