| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 | package outboundimport (	"context"	"io"	"net"	"time"	"v2ray.com/core/app"	"v2ray.com/core/app/log"	"v2ray.com/core/app/proxyman"	"v2ray.com/core/common/buf"	"v2ray.com/core/common/errors"	v2net "v2ray.com/core/common/net"	"v2ray.com/core/proxy"	"v2ray.com/core/transport/internet"	"v2ray.com/core/transport/ray")type Handler struct {	config          *proxyman.OutboundHandlerConfig	senderSettings  *proxyman.SenderConfig	proxy           proxy.Outbound	outboundManager proxyman.OutboundHandlerManager}func NewHandler(ctx context.Context, config *proxyman.OutboundHandlerConfig) (*Handler, error) {	h := &Handler{		config: config,	}	space := app.SpaceFromContext(ctx)	if space == nil {		return nil, errors.New("Proxyman|OutboundHandler: No space in context.")	}	space.OnInitialize(func() error {		ohm := proxyman.OutboundHandlerManagerFromSpace(space)		if ohm == nil {			return errors.New("Proxyman|OutboundHandler: No OutboundManager in space.")		}		h.outboundManager = ohm		return nil	})	if config.SenderSettings != nil {		senderSettings, err := config.SenderSettings.GetInstance()		if err != nil {			return nil, err		}		switch s := senderSettings.(type) {		case *proxyman.SenderConfig:			h.senderSettings = s		default:			return nil, errors.New("Proxyman|DefaultOutboundHandler: settings is not SenderConfig.")		}	}	proxyHandler, err := config.GetProxyHandler(ctx)	if err != nil {		return nil, err	}	h.proxy = proxyHandler	return h, nil}func (h *Handler) Dispatch(ctx context.Context, outboundRay ray.OutboundRay) {	err := h.proxy.Process(ctx, outboundRay, h)	// Ensure outbound ray is properly closed.	if err != nil && errors.Cause(err) != io.EOF {		log.Info("Proxyman|OutboundHandler: Failed to process outbound traffic: ", err)		outboundRay.OutboundOutput().CloseError()	} else {		outboundRay.OutboundOutput().Close()	}	outboundRay.OutboundInput().CloseError()}// Dial implements proxy.Dialer.Dial().func (h *Handler) Dial(ctx context.Context, dest v2net.Destination) (internet.Connection, error) {	if h.senderSettings != nil {		if h.senderSettings.ProxySettings.HasTag() {			tag := h.senderSettings.ProxySettings.Tag			handler := h.outboundManager.GetHandler(tag)			if handler != nil {				log.Info("Proxyman|OutboundHandler: Proxying to ", tag)				ctx = proxy.ContextWithTarget(ctx, dest)				stream := ray.NewRay(ctx)				go handler.Dispatch(ctx, stream)				return NewConnection(stream), nil			}			log.Warning("Proxyman|OutboundHandler: Failed to get outbound handler with tag: ", tag)		}		if h.senderSettings.Via != nil {			ctx = internet.ContextWithDialerSource(ctx, h.senderSettings.Via.AsAddress())		}		if h.senderSettings.StreamSettings != nil {			ctx = internet.ContextWithStreamSettings(ctx, h.senderSettings.StreamSettings)		}	}	return internet.Dial(ctx, dest)}type Connection struct {	stream     ray.Ray	closed     bool	localAddr  net.Addr	remoteAddr net.Addr	reader io.Reader	writer io.Writer}func NewConnection(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.ToBytesReader(stream.InboundOutput()),		writer: buf.ToBytesWriter(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}// SetDeadline implements net.Conn.SetDeadline().func (v *Connection) SetDeadline(t time.Time) error {	return nil}// SetReadDeadline implements net.Conn.SetReadDeadline().func (v *Connection) SetReadDeadline(t time.Time) error {	return nil}// SetWriteDeadline implement net.Conn.SetWriteDeadline().func (v *Connection) SetWriteDeadline(t time.Time) error {	return nil}func (v *Connection) Reusable() bool {	return false}func (v *Connection) SetReusable(bool) {}
 |