| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 | 
							- package inbound
 
- import (
 
- 	"context"
 
- 	"sync"
 
- 	"time"
 
- 	"v2ray.com/core"
 
- 	"v2ray.com/core/app/proxyman"
 
- 	"v2ray.com/core/app/proxyman/mux"
 
- 	"v2ray.com/core/common/dice"
 
- 	"v2ray.com/core/common/net"
 
- 	"v2ray.com/core/common/task"
 
- 	"v2ray.com/core/proxy"
 
- 	"v2ray.com/core/transport/internet"
 
- )
 
- type DynamicInboundHandler struct {
 
- 	tag            string
 
- 	v              *core.Instance
 
- 	proxyConfig    interface{}
 
- 	receiverConfig *proxyman.ReceiverConfig
 
- 	streamSettings *internet.MemoryStreamConfig
 
- 	portMutex      sync.Mutex
 
- 	portsInUse     map[net.Port]bool
 
- 	workerMutex    sync.RWMutex
 
- 	worker         []worker
 
- 	lastRefresh    time.Time
 
- 	mux            *mux.Server
 
- 	task           *task.Periodic
 
- }
 
- func NewDynamicInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*DynamicInboundHandler, error) {
 
- 	v := core.MustFromContext(ctx)
 
- 	h := &DynamicInboundHandler{
 
- 		tag:            tag,
 
- 		proxyConfig:    proxyConfig,
 
- 		receiverConfig: receiverConfig,
 
- 		portsInUse:     make(map[net.Port]bool),
 
- 		mux:            mux.NewServer(ctx),
 
- 		v:              v,
 
- 	}
 
- 	mss, err := internet.ToMemoryStreamConfig(receiverConfig.StreamSettings)
 
- 	if err != nil {
 
- 		return nil, newError("failed to parse stream settings").Base(err).AtWarning()
 
- 	}
 
- 	if receiverConfig.ReceiveOriginalDestination {
 
- 		if mss.SocketSettings == nil {
 
- 			mss.SocketSettings = &internet.SocketConfig{}
 
- 		}
 
- 		if mss.SocketSettings.Tproxy == internet.SocketConfig_Off {
 
- 			mss.SocketSettings.Tproxy = internet.SocketConfig_Redirect
 
- 		}
 
- 		mss.SocketSettings.ReceiveOriginalDestAddress = true
 
- 	}
 
- 	h.streamSettings = mss
 
- 	h.task = &task.Periodic{
 
- 		Interval: time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()),
 
- 		Execute:  h.refresh,
 
- 	}
 
- 	return h, nil
 
- }
 
- func (h *DynamicInboundHandler) allocatePort() net.Port {
 
- 	from := int(h.receiverConfig.PortRange.From)
 
- 	delta := int(h.receiverConfig.PortRange.To) - from + 1
 
- 	h.portMutex.Lock()
 
- 	defer h.portMutex.Unlock()
 
- 	for {
 
- 		r := dice.Roll(delta)
 
- 		port := net.Port(from + r)
 
- 		_, used := h.portsInUse[port]
 
- 		if !used {
 
- 			h.portsInUse[port] = true
 
- 			return port
 
- 		}
 
- 	}
 
- }
 
- func (h *DynamicInboundHandler) closeWorkers(workers []worker) {
 
- 	ports2Del := make([]net.Port, len(workers))
 
- 	for idx, worker := range workers {
 
- 		ports2Del[idx] = worker.Port()
 
- 		if err := worker.Close(); err != nil {
 
- 			newError("failed to close worker").Base(err).WriteToLog()
 
- 		}
 
- 	}
 
- 	h.portMutex.Lock()
 
- 	for _, port := range ports2Del {
 
- 		delete(h.portsInUse, port)
 
- 	}
 
- 	h.portMutex.Unlock()
 
- }
 
- func (h *DynamicInboundHandler) refresh() error {
 
- 	h.lastRefresh = time.Now()
 
- 	timeout := time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()) * 2
 
- 	concurrency := h.receiverConfig.AllocationStrategy.GetConcurrencyValue()
 
- 	workers := make([]worker, 0, concurrency)
 
- 	address := h.receiverConfig.Listen.AsAddress()
 
- 	if address == nil {
 
- 		address = net.AnyIP
 
- 	}
 
- 	uplinkCounter, downlinkCounter := getStatCounter(h.v, h.tag)
 
- 	for i := uint32(0); i < concurrency; i++ {
 
- 		port := h.allocatePort()
 
- 		rawProxy, err := core.CreateObject(h.v, h.proxyConfig)
 
- 		if err != nil {
 
- 			newError("failed to create proxy instance").Base(err).AtWarning().WriteToLog()
 
- 			continue
 
- 		}
 
- 		p := rawProxy.(proxy.Inbound)
 
- 		nl := p.Network()
 
- 		if nl.HasNetwork(net.Network_TCP) {
 
- 			worker := &tcpWorker{
 
- 				tag:             h.tag,
 
- 				address:         address,
 
- 				port:            port,
 
- 				proxy:           p,
 
- 				stream:          h.streamSettings,
 
- 				recvOrigDest:    h.receiverConfig.ReceiveOriginalDestination,
 
- 				dispatcher:      h.mux,
 
- 				sniffingConfig:  h.receiverConfig.GetEffectiveSniffingSettings(),
 
- 				uplinkCounter:   uplinkCounter,
 
- 				downlinkCounter: downlinkCounter,
 
- 			}
 
- 			if err := worker.Start(); err != nil {
 
- 				newError("failed to create TCP worker").Base(err).AtWarning().WriteToLog()
 
- 				continue
 
- 			}
 
- 			workers = append(workers, worker)
 
- 		}
 
- 		if nl.HasNetwork(net.Network_UDP) {
 
- 			worker := &udpWorker{
 
- 				tag:             h.tag,
 
- 				proxy:           p,
 
- 				address:         address,
 
- 				port:            port,
 
- 				dispatcher:      h.mux,
 
- 				uplinkCounter:   uplinkCounter,
 
- 				downlinkCounter: downlinkCounter,
 
- 				stream:          h.streamSettings,
 
- 			}
 
- 			if err := worker.Start(); err != nil {
 
- 				newError("failed to create UDP worker").Base(err).AtWarning().WriteToLog()
 
- 				continue
 
- 			}
 
- 			workers = append(workers, worker)
 
- 		}
 
- 	}
 
- 	h.workerMutex.Lock()
 
- 	h.worker = workers
 
- 	h.workerMutex.Unlock()
 
- 	time.AfterFunc(timeout, func() {
 
- 		h.closeWorkers(workers)
 
- 	})
 
- 	return nil
 
- }
 
- func (h *DynamicInboundHandler) Start() error {
 
- 	return h.task.Start()
 
- }
 
- func (h *DynamicInboundHandler) Close() error {
 
- 	return h.task.Close()
 
- }
 
- func (h *DynamicInboundHandler) GetRandomInboundProxy() (interface{}, net.Port, int) {
 
- 	h.workerMutex.RLock()
 
- 	defer h.workerMutex.RUnlock()
 
- 	if len(h.worker) == 0 {
 
- 		return nil, 0, 0
 
- 	}
 
- 	w := h.worker[dice.Roll(len(h.worker))]
 
- 	expire := h.receiverConfig.AllocationStrategy.GetRefreshValue() - uint32(time.Since(h.lastRefresh)/time.Minute)
 
- 	return w.Proxy(), w.Port(), int(expire)
 
- }
 
- func (h *DynamicInboundHandler) Tag() string {
 
- 	return h.tag
 
- }
 
 
  |