package proxy import ( "io" "net" "time" "context" "v2ray.com/core/app" "v2ray.com/core/app/proxyman" "v2ray.com/core/common" "v2ray.com/core/common/buf" "v2ray.com/core/common/errors" "v2ray.com/core/common/log" v2net "v2ray.com/core/common/net" "v2ray.com/core/transport/internet" "v2ray.com/core/transport/ray" ) type OutboundProxy struct { outboundManager proxyman.OutboundHandlerManager } func NewOutboundProxy(ctx context.Context, config *Config) (*OutboundProxy, error) { space := app.SpaceFromContext(ctx) if space == nil { return nil, errors.New("OutboundProxy: No space in context.") } proxy := new(OutboundProxy) space.OnInitialize(func() error { proxy.outboundManager = proxyman.OutboundHandlerManagerFromSpace(space) if proxy.outboundManager == nil { return errors.New("Proxy: Outbound handler manager not found in space.") } return nil }) return proxy, nil } func (OutboundProxy) Interface() interface{} { return (*OutboundProxy)(nil) } func (v *OutboundProxy) RegisterDialer() { internet.ProxyDialer = v.Dial } // Dial implements internet.Dialer. func (v *OutboundProxy) Dial(src v2net.Address, dest v2net.Destination, options internet.DialerOptions) (internet.Connection, error) { handler := v.outboundManager.GetHandler(options.Proxy.Tag) if handler == nil { log.Warning("Proxy: Failed to get outbound handler with tag: ", options.Proxy.Tag) return internet.Dial(src, dest, internet.DialerOptions{ Stream: options.Stream, }) } log.Info("Proxy: Dialing to ", dest) stream := ray.NewRay() go handler.Dispatch(dest, stream) return NewConnection(src, dest, stream), nil } type Connection struct { stream ray.Ray closed bool localAddr net.Addr remoteAddr net.Addr reader *buf.BufferToBytesReader writer *buf.BytesToBufferWriter } func NewConnection(src v2net.Address, dest v2net.Destination, stream ray.Ray) *Connection { return &Connection{ 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: buf.NewBytesReader(stream.InboundOutput()), writer: buf.NewBytesWriter(stream.InboundInput()), } } // Read implements net.Conn.Read(). func (v *Connection) Read(b []byte) (int, error) { if v.closed { return 0, io.EOF } return v.reader.Read(b) } // Write implements net.Conn.Write(). func (v *Connection) Write(b []byte) (int, error) { if v.closed { return 0, io.ErrClosedPipe } return v.writer.Write(b) } // Close implements net.Conn.Close(). func (v *Connection) Close() error { v.closed = true v.stream.InboundInput().Close() v.stream.InboundOutput().CloseError() return nil } // LocalAddr implements net.Conn.LocalAddr(). func (v *Connection) LocalAddr() net.Addr { return v.localAddr } // RemoteAddr implements net.Conn.RemoteAddr(). func (v *Connection) RemoteAddr() net.Addr { return v.remoteAddr } func (v *Connection) SetDeadline(t time.Time) error { return nil } func (v *Connection) SetReadDeadline(t time.Time) error { return nil } func (v *Connection) SetWriteDeadline(t time.Time) error { return nil } func (v *Connection) Reusable() bool { return false } func (v *Connection) SetReusable(bool) { } func OutboundProxyFromSpace(space app.Space) *OutboundProxy { app := space.GetApplication((*OutboundProxy)(nil)) if app == nil { return nil } return app.(*OutboundProxy) } func init() { common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) { return NewOutboundProxy(ctx, config.(*Config)) })) }