1
0

dynamic.go 5.9 KB

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