dynamic.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. package inbound
  2. import (
  3. "context"
  4. "sync"
  5. "time"
  6. "v2ray.com/core/app/proxyman"
  7. "v2ray.com/core/common/dice"
  8. "v2ray.com/core/common/log"
  9. v2net "v2ray.com/core/common/net"
  10. "v2ray.com/core/proxy"
  11. )
  12. type DynamicInboundHandler struct {
  13. sync.Mutex
  14. tag string
  15. ctx context.Context
  16. cancel context.CancelFunc
  17. proxyConfig interface{}
  18. receiverConfig *proxyman.ReceiverConfig
  19. portsInUse map[v2net.Port]bool
  20. worker []worker
  21. worker2Recycle []worker
  22. lastRefresh time.Time
  23. }
  24. func NewDynamicInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*DynamicInboundHandler, error) {
  25. ctx, cancel := context.WithCancel(ctx)
  26. h := &DynamicInboundHandler{
  27. ctx: ctx,
  28. tag: tag,
  29. cancel: cancel,
  30. proxyConfig: proxyConfig,
  31. receiverConfig: receiverConfig,
  32. portsInUse: make(map[v2net.Port]bool),
  33. }
  34. return h, nil
  35. }
  36. func (h *DynamicInboundHandler) allocatePort() v2net.Port {
  37. from := int(h.receiverConfig.PortRange.From)
  38. delta := int(h.receiverConfig.PortRange.To) - from + 1
  39. h.Lock()
  40. defer h.Unlock()
  41. for {
  42. r := dice.Roll(delta)
  43. port := v2net.Port(from + r)
  44. _, used := h.portsInUse[port]
  45. if !used {
  46. h.portsInUse[port] = true
  47. return port
  48. }
  49. }
  50. }
  51. func (h *DynamicInboundHandler) refresh() error {
  52. h.lastRefresh = time.Now()
  53. ports2Del := make([]v2net.Port, 0, 16)
  54. for _, worker := range h.worker2Recycle {
  55. worker.Close()
  56. ports2Del = append(ports2Del, worker.Port())
  57. }
  58. h.Lock()
  59. for _, port := range ports2Del {
  60. delete(h.portsInUse, port)
  61. }
  62. h.Unlock()
  63. h.worker2Recycle, h.worker = h.worker, h.worker2Recycle[:0]
  64. address := h.receiverConfig.Listen.AsAddress()
  65. if address == nil {
  66. address = v2net.AnyIP
  67. }
  68. for i := uint32(0); i < h.receiverConfig.AllocationStrategy.GetConcurrencyValue(); i++ {
  69. port := h.allocatePort()
  70. p, err := proxy.CreateInboundHandler(h.ctx, h.proxyConfig)
  71. if err != nil {
  72. log.Warning("Proxyman|DefaultInboundHandler: Failed to create proxy instance: ", err)
  73. continue
  74. }
  75. nl := p.Network()
  76. if nl.HasNetwork(v2net.Network_TCP) {
  77. worker := &tcpWorker{
  78. tag: h.tag,
  79. address: address,
  80. port: port,
  81. proxy: p,
  82. stream: h.receiverConfig.StreamSettings,
  83. recvOrigDest: h.receiverConfig.ReceiveOriginalDestination,
  84. allowPassiveConn: h.receiverConfig.AllowPassiveConnection,
  85. }
  86. if err := worker.Start(); err != nil {
  87. return err
  88. }
  89. h.worker = append(h.worker, worker)
  90. }
  91. if nl.HasNetwork(v2net.Network_UDP) {
  92. worker := &udpWorker{
  93. tag: h.tag,
  94. proxy: p,
  95. address: address,
  96. port: port,
  97. recvOrigDest: h.receiverConfig.ReceiveOriginalDestination,
  98. }
  99. if err := worker.Start(); err != nil {
  100. return err
  101. }
  102. h.worker = append(h.worker, worker)
  103. }
  104. }
  105. return nil
  106. }
  107. func (h *DynamicInboundHandler) monitor() {
  108. for {
  109. select {
  110. case <-h.ctx.Done():
  111. return
  112. case <-time.After(time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue())):
  113. h.refresh()
  114. }
  115. }
  116. }
  117. func (h *DynamicInboundHandler) Start() error {
  118. err := h.refresh()
  119. go h.monitor()
  120. return err
  121. }
  122. func (h *DynamicInboundHandler) Close() {
  123. h.cancel()
  124. }
  125. func (h *DynamicInboundHandler) GetRandomInboundProxy() (proxy.Inbound, v2net.Port, int) {
  126. w := h.worker[dice.Roll(len(h.worker))]
  127. expire := h.receiverConfig.AllocationStrategy.GetRefreshValue() - uint32(time.Since(h.lastRefresh)/time.Minute)
  128. return w.Proxy(), w.Port(), int(expire)
  129. }