| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 | 
							- package inbound
 
- import (
 
- 	"context"
 
- 	"sync"
 
- 	"time"
 
- 	"v2ray.com/core/app/proxyman"
 
- 	"v2ray.com/core/common/dice"
 
- 	"v2ray.com/core/common/log"
 
- 	v2net "v2ray.com/core/common/net"
 
- 	"v2ray.com/core/proxy"
 
- )
 
- type DynamicInboundHandler struct {
 
- 	sync.Mutex
 
- 	tag            string
 
- 	ctx            context.Context
 
- 	cancel         context.CancelFunc
 
- 	proxyConfig    interface{}
 
- 	receiverConfig *proxyman.ReceiverConfig
 
- 	portsInUse     map[v2net.Port]bool
 
- 	worker         []worker
 
- 	worker2Recycle []worker
 
- 	lastRefresh    time.Time
 
- }
 
- func NewDynamicInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*DynamicInboundHandler, error) {
 
- 	ctx, cancel := context.WithCancel(ctx)
 
- 	h := &DynamicInboundHandler{
 
- 		ctx:            ctx,
 
- 		tag:            tag,
 
- 		cancel:         cancel,
 
- 		proxyConfig:    proxyConfig,
 
- 		receiverConfig: receiverConfig,
 
- 		portsInUse:     make(map[v2net.Port]bool),
 
- 	}
 
- 	return h, nil
 
- }
 
- func (h *DynamicInboundHandler) allocatePort() v2net.Port {
 
- 	from := int(h.receiverConfig.PortRange.From)
 
- 	delta := int(h.receiverConfig.PortRange.To) - from + 1
 
- 	h.Lock()
 
- 	defer h.Unlock()
 
- 	for {
 
- 		r := dice.Roll(delta)
 
- 		port := v2net.Port(from + r)
 
- 		_, used := h.portsInUse[port]
 
- 		if !used {
 
- 			h.portsInUse[port] = true
 
- 			return port
 
- 		}
 
- 	}
 
- }
 
- func (h *DynamicInboundHandler) refresh() error {
 
- 	h.lastRefresh = time.Now()
 
- 	ports2Del := make([]v2net.Port, 0, 16)
 
- 	for _, worker := range h.worker2Recycle {
 
- 		worker.Close()
 
- 		ports2Del = append(ports2Del, worker.Port())
 
- 	}
 
- 	h.Lock()
 
- 	for _, port := range ports2Del {
 
- 		delete(h.portsInUse, port)
 
- 	}
 
- 	h.Unlock()
 
- 	h.worker2Recycle, h.worker = h.worker, h.worker2Recycle[:0]
 
- 	for i := uint32(0); i < h.receiverConfig.AllocationStrategy.GetConcurrencyValue(); i++ {
 
- 		port := h.allocatePort()
 
- 		p, err := proxy.CreateInboundHandler(h.ctx, h.proxyConfig)
 
- 		if err != nil {
 
- 			log.Warning("Proxyman|DefaultInboundHandler: Failed to create proxy instance: ", err)
 
- 			continue
 
- 		}
 
- 		nl := p.Network()
 
- 		if nl.HasNetwork(v2net.Network_TCP) {
 
- 			worker := &tcpWorker{
 
- 				tag:              h.tag,
 
- 				address:          h.receiverConfig.Listen.AsAddress(),
 
- 				port:             port,
 
- 				proxy:            p,
 
- 				stream:           h.receiverConfig.StreamSettings,
 
- 				recvOrigDest:     h.receiverConfig.ReceiveOriginalDestination,
 
- 				allowPassiveConn: h.receiverConfig.AllowPassiveConnection,
 
- 			}
 
- 			if err := worker.Start(); err != nil {
 
- 				return err
 
- 			}
 
- 			h.worker = append(h.worker, worker)
 
- 		}
 
- 		if nl.HasNetwork(v2net.Network_UDP) {
 
- 			worker := &udpWorker{
 
- 				tag:          h.tag,
 
- 				proxy:        p,
 
- 				address:      h.receiverConfig.Listen.AsAddress(),
 
- 				port:         port,
 
- 				recvOrigDest: h.receiverConfig.ReceiveOriginalDestination,
 
- 			}
 
- 			if err := worker.Start(); err != nil {
 
- 				return err
 
- 			}
 
- 			h.worker = append(h.worker, worker)
 
- 		}
 
- 	}
 
- 	return nil
 
- }
 
- func (h *DynamicInboundHandler) monitor() {
 
- 	for {
 
- 		select {
 
- 		case <-h.ctx.Done():
 
- 			return
 
- 		case <-time.After(time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue())):
 
- 			h.refresh()
 
- 		}
 
- 	}
 
- }
 
- func (h *DynamicInboundHandler) Start() error {
 
- 	err := h.refresh()
 
- 	go h.monitor()
 
- 	return err
 
- }
 
- func (h *DynamicInboundHandler) Close() {
 
- 	h.cancel()
 
- }
 
- func (h *DynamicInboundHandler) GetRandomInboundProxy() (proxy.Inbound, v2net.Port, int) {
 
- 	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)
 
- }
 
 
  |