| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 | package implimport (	"time"	"v2ray.com/core/app"	"v2ray.com/core/app/dispatcher"	"v2ray.com/core/app/proxyman"	"v2ray.com/core/app/router"	"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/proxy"	"v2ray.com/core/transport/ray")type DefaultDispatcher struct {	ohm    proxyman.OutboundHandlerManager	router *router.Router}func NewDefaultDispatcher(space app.Space) *DefaultDispatcher {	d := &DefaultDispatcher{}	space.OnInitialize(func() error {		d.ohm = proxyman.OutboundHandlerManagerFromSpace(space)		if d.ohm == nil {			return errors.New("DefaultDispatcher: OutboundHandlerManager is not found in the space.")		}		d.router = router.FromSpace(space)		return nil	})	return d}func (v *DefaultDispatcher) DispatchToOutbound(session *proxy.SessionInfo) ray.InboundRay {	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)		}	}	direct := ray.NewRay()	var waitFunc func() error	if session.Inbound != nil && session.Inbound.AllowPassiveConnection {		waitFunc = noOpWait()	} else {		wdi := &waitDataInspector{			hasData: make(chan bool, 1),		}		direct.AddInspector(wdi)		waitFunc = waitForData(wdi)	}	go v.waitAndDispatch(waitFunc, destination, direct, dispatcher)	return direct}func (v *DefaultDispatcher) waitAndDispatch(wait func() error, destination v2net.Destination, link ray.OutboundRay, dispatcher proxy.OutboundHandler) {	if err := wait(); err != nil {		log.Info("DefaultDispatcher: Failed precondition: ", err)		link.OutboundInput().CloseError()		link.OutboundOutput().CloseError()		return	}	dispatcher.Dispatch(destination, link)}type DefaultDispatcherFactory struct{}func (v DefaultDispatcherFactory) Create(space app.Space, config interface{}) (app.Application, error) {	return NewDefaultDispatcher(space), nil}func init() {	common.Must(app.RegisterApplicationFactory((*dispatcher.Config)(nil), 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	}}
 |