dynamic.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. package inbound
  2. import (
  3. "context"
  4. "sync"
  5. "time"
  6. "github.com/xtls/xray-core/app/proxyman"
  7. "github.com/xtls/xray-core/common/dice"
  8. "github.com/xtls/xray-core/common/errors"
  9. "github.com/xtls/xray-core/common/mux"
  10. "github.com/xtls/xray-core/common/net"
  11. "github.com/xtls/xray-core/common/task"
  12. "github.com/xtls/xray-core/core"
  13. "github.com/xtls/xray-core/proxy"
  14. "github.com/xtls/xray-core/transport/internet"
  15. )
  16. type DynamicInboundHandler struct {
  17. tag string
  18. v *core.Instance
  19. proxyConfig interface{}
  20. receiverConfig *proxyman.ReceiverConfig
  21. streamSettings *internet.MemoryStreamConfig
  22. portMutex sync.Mutex
  23. portsInUse map[net.Port]struct{}
  24. workerMutex sync.RWMutex
  25. worker []worker
  26. lastRefresh time.Time
  27. mux *mux.Server
  28. task *task.Periodic
  29. ctx context.Context
  30. }
  31. func NewDynamicInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*DynamicInboundHandler, error) {
  32. v := core.MustFromContext(ctx)
  33. h := &DynamicInboundHandler{
  34. tag: tag,
  35. proxyConfig: proxyConfig,
  36. receiverConfig: receiverConfig,
  37. portsInUse: make(map[net.Port]struct{}),
  38. mux: mux.NewServer(ctx),
  39. v: v,
  40. ctx: ctx,
  41. }
  42. mss, err := internet.ToMemoryStreamConfig(receiverConfig.StreamSettings)
  43. if err != nil {
  44. return nil, errors.New("failed to parse stream settings").Base(err).AtWarning()
  45. }
  46. if receiverConfig.ReceiveOriginalDestination {
  47. if mss.SocketSettings == nil {
  48. mss.SocketSettings = &internet.SocketConfig{}
  49. }
  50. if mss.SocketSettings.Tproxy == internet.SocketConfig_Off {
  51. mss.SocketSettings.Tproxy = internet.SocketConfig_Redirect
  52. }
  53. mss.SocketSettings.ReceiveOriginalDestAddress = true
  54. }
  55. h.streamSettings = mss
  56. h.task = &task.Periodic{
  57. Interval: time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()),
  58. Execute: h.refresh,
  59. }
  60. return h, nil
  61. }
  62. func (h *DynamicInboundHandler) allocatePort() net.Port {
  63. allPorts := []int32{}
  64. for _, pr := range h.receiverConfig.PortList.Range {
  65. for i := pr.From; i <= pr.To; i++ {
  66. allPorts = append(allPorts, int32(i))
  67. }
  68. }
  69. h.portMutex.Lock()
  70. defer h.portMutex.Unlock()
  71. for {
  72. r := dice.Roll(len(allPorts))
  73. port := net.Port(allPorts[r])
  74. _, used := h.portsInUse[port]
  75. if !used {
  76. h.portsInUse[port] = struct{}{}
  77. return port
  78. }
  79. }
  80. }
  81. func (h *DynamicInboundHandler) closeWorkers(workers []worker) {
  82. ports2Del := make([]net.Port, len(workers))
  83. for idx, worker := range workers {
  84. ports2Del[idx] = worker.Port()
  85. if err := worker.Close(); err != nil {
  86. errors.LogInfoInner(h.ctx, err, "failed to close worker")
  87. }
  88. }
  89. h.portMutex.Lock()
  90. for _, port := range ports2Del {
  91. delete(h.portsInUse, port)
  92. }
  93. h.portMutex.Unlock()
  94. }
  95. func (h *DynamicInboundHandler) refresh() error {
  96. h.lastRefresh = time.Now()
  97. timeout := time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()) * 2
  98. concurrency := h.receiverConfig.AllocationStrategy.GetConcurrencyValue()
  99. workers := make([]worker, 0, concurrency)
  100. address := h.receiverConfig.Listen.AsAddress()
  101. if address == nil {
  102. address = net.AnyIP
  103. }
  104. uplinkCounter, downlinkCounter := getStatCounter(h.v, h.tag)
  105. for i := uint32(0); i < concurrency; i++ {
  106. port := h.allocatePort()
  107. rawProxy, err := core.CreateObject(h.v, h.proxyConfig)
  108. if err != nil {
  109. errors.LogWarningInner(h.ctx, err, "failed to create proxy instance")
  110. continue
  111. }
  112. p := rawProxy.(proxy.Inbound)
  113. nl := p.Network()
  114. if net.HasNetwork(nl, net.Network_TCP) {
  115. worker := &tcpWorker{
  116. tag: h.tag,
  117. address: address,
  118. port: port,
  119. proxy: p,
  120. stream: h.streamSettings,
  121. recvOrigDest: h.receiverConfig.ReceiveOriginalDestination,
  122. dispatcher: h.mux,
  123. sniffingConfig: h.receiverConfig.GetEffectiveSniffingSettings(),
  124. uplinkCounter: uplinkCounter,
  125. downlinkCounter: downlinkCounter,
  126. ctx: h.ctx,
  127. }
  128. if err := worker.Start(); err != nil {
  129. errors.LogWarningInner(h.ctx, err, "failed to create TCP worker")
  130. continue
  131. }
  132. workers = append(workers, worker)
  133. }
  134. if net.HasNetwork(nl, net.Network_UDP) {
  135. worker := &udpWorker{
  136. tag: h.tag,
  137. proxy: p,
  138. address: address,
  139. port: port,
  140. dispatcher: h.mux,
  141. sniffingConfig: h.receiverConfig.GetEffectiveSniffingSettings(),
  142. uplinkCounter: uplinkCounter,
  143. downlinkCounter: downlinkCounter,
  144. stream: h.streamSettings,
  145. ctx: h.ctx,
  146. }
  147. if err := worker.Start(); err != nil {
  148. errors.LogWarningInner(h.ctx, err, "failed to create UDP worker")
  149. continue
  150. }
  151. workers = append(workers, worker)
  152. }
  153. }
  154. h.workerMutex.Lock()
  155. h.worker = workers
  156. h.workerMutex.Unlock()
  157. time.AfterFunc(timeout, func() {
  158. h.closeWorkers(workers)
  159. })
  160. return nil
  161. }
  162. func (h *DynamicInboundHandler) Start() error {
  163. return h.task.Start()
  164. }
  165. func (h *DynamicInboundHandler) Close() error {
  166. return h.task.Close()
  167. }
  168. func (h *DynamicInboundHandler) GetRandomInboundProxy() (interface{}, net.Port, int) {
  169. h.workerMutex.RLock()
  170. defer h.workerMutex.RUnlock()
  171. if len(h.worker) == 0 {
  172. return nil, 0, 0
  173. }
  174. w := h.worker[dice.Roll(len(h.worker))]
  175. expire := h.receiverConfig.AllocationStrategy.GetRefreshValue() - uint32(time.Since(h.lastRefresh)/time.Minute)
  176. return w.Proxy(), w.Port(), int(expire)
  177. }
  178. func (h *DynamicInboundHandler) Tag() string {
  179. return h.tag
  180. }