bridge.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. package reverse
  2. import (
  3. "context"
  4. "time"
  5. "github.com/golang/protobuf/proto"
  6. "github.com/xtls/xray-core/common/mux"
  7. "github.com/xtls/xray-core/common/net"
  8. "github.com/xtls/xray-core/common/session"
  9. "github.com/xtls/xray-core/common/task"
  10. "github.com/xtls/xray-core/features/routing"
  11. "github.com/xtls/xray-core/transport"
  12. "github.com/xtls/xray-core/transport/pipe"
  13. )
  14. // Bridge is a component in reverse proxy, that relays connections from Portal to local address.
  15. type Bridge struct {
  16. dispatcher routing.Dispatcher
  17. tag string
  18. domain string
  19. workers []*BridgeWorker
  20. monitorTask *task.Periodic
  21. }
  22. // NewBridge creates a new Bridge instance.
  23. func NewBridge(config *BridgeConfig, dispatcher routing.Dispatcher) (*Bridge, error) {
  24. if config.Tag == "" {
  25. return nil, newError("bridge tag is empty")
  26. }
  27. if config.Domain == "" {
  28. return nil, newError("bridge domain is empty")
  29. }
  30. b := &Bridge{
  31. dispatcher: dispatcher,
  32. tag: config.Tag,
  33. domain: config.Domain,
  34. }
  35. b.monitorTask = &task.Periodic{
  36. Execute: b.monitor,
  37. Interval: time.Second * 2,
  38. }
  39. return b, nil
  40. }
  41. func (b *Bridge) cleanup() {
  42. var activeWorkers []*BridgeWorker
  43. for _, w := range b.workers {
  44. if w.IsActive() {
  45. activeWorkers = append(activeWorkers, w)
  46. }
  47. }
  48. if len(activeWorkers) != len(b.workers) {
  49. b.workers = activeWorkers
  50. }
  51. }
  52. func (b *Bridge) monitor() error {
  53. b.cleanup()
  54. var numConnections uint32
  55. var numWorker uint32
  56. for _, w := range b.workers {
  57. if w.IsActive() {
  58. numConnections += w.Connections()
  59. numWorker++
  60. }
  61. }
  62. if numWorker == 0 || numConnections/numWorker > 16 {
  63. worker, err := NewBridgeWorker(b.domain, b.tag, b.dispatcher)
  64. if err != nil {
  65. newError("failed to create bridge worker").Base(err).AtWarning().WriteToLog()
  66. return nil
  67. }
  68. b.workers = append(b.workers, worker)
  69. }
  70. return nil
  71. }
  72. func (b *Bridge) Start() error {
  73. return b.monitorTask.Start()
  74. }
  75. func (b *Bridge) Close() error {
  76. return b.monitorTask.Close()
  77. }
  78. type BridgeWorker struct {
  79. tag string
  80. worker *mux.ServerWorker
  81. dispatcher routing.Dispatcher
  82. state Control_State
  83. }
  84. func NewBridgeWorker(domain string, tag string, d routing.Dispatcher) (*BridgeWorker, error) {
  85. ctx := context.Background()
  86. ctx = session.ContextWithInbound(ctx, &session.Inbound{
  87. Tag: tag,
  88. })
  89. link, err := d.Dispatch(ctx, net.Destination{
  90. Network: net.Network_TCP,
  91. Address: net.DomainAddress(domain),
  92. Port: 0,
  93. })
  94. if err != nil {
  95. return nil, err
  96. }
  97. w := &BridgeWorker{
  98. dispatcher: d,
  99. tag: tag,
  100. }
  101. worker, err := mux.NewServerWorker(context.Background(), w, link)
  102. if err != nil {
  103. return nil, err
  104. }
  105. w.worker = worker
  106. return w, nil
  107. }
  108. func (w *BridgeWorker) Type() interface{} {
  109. return routing.DispatcherType()
  110. }
  111. func (w *BridgeWorker) Start() error {
  112. return nil
  113. }
  114. func (w *BridgeWorker) Close() error {
  115. return nil
  116. }
  117. func (w *BridgeWorker) IsActive() bool {
  118. return w.state == Control_ACTIVE && !w.worker.Closed()
  119. }
  120. func (w *BridgeWorker) Connections() uint32 {
  121. return w.worker.ActiveConnections()
  122. }
  123. func (w *BridgeWorker) handleInternalConn(link transport.Link) {
  124. go func() {
  125. reader := link.Reader
  126. for {
  127. mb, err := reader.ReadMultiBuffer()
  128. if err != nil {
  129. break
  130. }
  131. for _, b := range mb {
  132. var ctl Control
  133. if err := proto.Unmarshal(b.Bytes(), &ctl); err != nil {
  134. newError("failed to parse proto message").Base(err).WriteToLog()
  135. break
  136. }
  137. if ctl.State != w.state {
  138. w.state = ctl.State
  139. }
  140. }
  141. }
  142. }()
  143. }
  144. func (w *BridgeWorker) Dispatch(ctx context.Context, dest net.Destination) (*transport.Link, error) {
  145. if !isInternalDomain(dest) {
  146. ctx = session.ContextWithInbound(ctx, &session.Inbound{
  147. Tag: w.tag,
  148. })
  149. return w.dispatcher.Dispatch(ctx, dest)
  150. }
  151. opt := []pipe.Option{pipe.WithSizeLimit(16 * 1024)}
  152. uplinkReader, uplinkWriter := pipe.New(opt...)
  153. downlinkReader, downlinkWriter := pipe.New(opt...)
  154. w.handleInternalConn(transport.Link{
  155. Reader: downlinkReader,
  156. Writer: uplinkWriter,
  157. })
  158. return &transport.Link{
  159. Reader: uplinkReader,
  160. Writer: downlinkWriter,
  161. }, nil
  162. }