| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 |
- package impl
- import (
- "context"
- "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"
- "v2ray.com/core/proxy"
- "v2ray.com/core/transport/ray"
- )
- type DefaultDispatcher struct {
- ohm proxyman.OutboundHandlerManager
- router *router.Router
- }
- func NewDefaultDispatcher(ctx context.Context, config *dispatcher.Config) (*DefaultDispatcher, error) {
- space := app.SpaceFromContext(ctx)
- if space == nil {
- return nil, errors.New("DefaultDispatcher: No space in context.")
- }
- 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, nil
- }
- func (DefaultDispatcher) Interface() interface{} {
- return (*dispatcher.Interface)(nil)
- }
- func (v *DefaultDispatcher) DispatchToOutbound(ctx context.Context) ray.InboundRay {
- dispatcher := v.ohm.GetDefaultHandler()
- destination := proxy.DestinationFromContext(ctx)
- if !destination.IsValid() {
- panic("Dispatcher: Invalid destination.")
- }
- if v.router != nil {
- if tag, err := v.router.TakeDetour(ctx); 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(ctx)
- var waitFunc func() error
- if allowPassiveConnection, ok := proxy.AllowPassiveConnectionFromContext(ctx); ok && allowPassiveConnection {
- waitFunc = noOpWait()
- } else {
- wdi := &waitDataInspector{
- hasData: make(chan bool, 1),
- }
- direct.AddInspector(wdi)
- waitFunc = waitForData(wdi)
- }
- go v.waitAndDispatch(ctx, waitFunc, direct, dispatcher)
- return direct
- }
- func (v *DefaultDispatcher) waitAndDispatch(ctx context.Context, wait func() error, link ray.OutboundRay, dispatcher proxyman.OutboundHandler) {
- if err := wait(); err != nil {
- log.Info("DefaultDispatcher: Failed precondition: ", err)
- link.OutboundInput().CloseError()
- link.OutboundOutput().CloseError()
- return
- }
- dispatcher.Dispatch(ctx, link)
- }
- func init() {
- common.Must(common.RegisterConfig((*dispatcher.Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
- return NewDefaultDispatcher(ctx, config.(*dispatcher.Config))
- }))
- }
- 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
- }
- }
|