| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 | package implimport (	"v2ray.com/core/app"	"v2ray.com/core/app/dispatcher"	"v2ray.com/core/app/proxyman"	"v2ray.com/core/app/router"	"v2ray.com/core/common/buf"	"v2ray.com/core/common/errors"	"v2ray.com/core/common/log"	v2net "v2ray.com/core/common/net"	"v2ray.com/core/common/serial"	"v2ray.com/core/proxy"	"v2ray.com/core/transport/ray")type DefaultDispatcher struct {	ohm    proxyman.OutboundHandlerManager	router *router.Router}func NewDefaultDispatcher(space app.Space) *DefaultDispatcher {	d := &DefaultDispatcher{}	space.InitializeApplication(func() error {		return d.Initialize(space)	})	return d}// Private: Used by app.Space only.func (v *DefaultDispatcher) Initialize(space app.Space) error {	if !space.HasApp(proxyman.APP_ID_OUTBOUND_MANAGER) {		return errors.New("DefaultDispatcher: OutboundHandlerManager is not found in the space.")	}	v.ohm = space.GetApp(proxyman.APP_ID_OUTBOUND_MANAGER).(proxyman.OutboundHandlerManager)	if space.HasApp(router.APP_ID) {		v.router = space.GetApp(router.APP_ID).(*router.Router)	}	return nil}func (v *DefaultDispatcher) Release() {}func (v *DefaultDispatcher) DispatchToOutbound(session *proxy.SessionInfo) ray.InboundRay {	direct := ray.NewRay()	dispatcher := v.ohm.GetDefaultHandler()	destination := session.Destination	if v.router != nil {		if tag, err := v.router.TakeDetour(session); err == nil {			if handler := v.ohm.GetHandler(tag); handler != nil {				log.Info("DefaultDispatcher: Taking detour [", tag, "] for [", destination, "].")				dispatcher = handler			} else {				log.Warning("DefaultDispatcher: Nonexisting tag: ", tag)			}		} else {			log.Info("DefaultDispatcher: Default route for ", destination)		}	}	if session.Inbound != nil && session.Inbound.AllowPassiveConnection {		go dispatcher.Dispatch(destination, buf.NewLocal(32), direct)	} else {		go v.FilterPacketAndDispatch(destination, direct, dispatcher)	}	return direct}// FilterPacketAndDispatch waits for a payload from source and starts dispatching.// Private: Visible for testing.func (v *DefaultDispatcher) FilterPacketAndDispatch(destination v2net.Destination, link ray.OutboundRay, dispatcher proxy.OutboundHandler) {	payload, err := link.OutboundInput().Read()	if err != nil {		log.Info("DefaultDispatcher: No payload towards ", destination, ", stopping now.")		link.OutboundInput().Release()		link.OutboundOutput().Release()		return	}	dispatcher.Dispatch(destination, payload, link)}func (v *DefaultDispatcher) FixDestination(payload *buf.Buffer, dest v2net.Destination) v2net.Destination {	if dest.Address.Family().IsDomain() || dest.Network != v2net.Network_TCP {		return dest	}	switch dest.Port {	case 80:		addr := DetectHTTPHost(payload)		if addr != nil {			dest.Address = addr		}	}	return dest}type DefaultDispatcherFactory struct{}func (v DefaultDispatcherFactory) Create(space app.Space, config interface{}) (app.Application, error) {	return NewDefaultDispatcher(space), nil}func (v DefaultDispatcherFactory) AppId() app.ID {	return dispatcher.APP_ID}func init() {	app.RegisterApplicationFactory(serial.GetMessageType(new(dispatcher.Config)), DefaultDispatcherFactory{})}
 |