outbound.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package outbound
  2. //go:generate go run github.com/xtls/xray-core/common/errors/errorgen
  3. import (
  4. "context"
  5. "sort"
  6. "strings"
  7. "sync"
  8. "github.com/xtls/xray-core/app/proxyman"
  9. "github.com/xtls/xray-core/common"
  10. "github.com/xtls/xray-core/common/errors"
  11. "github.com/xtls/xray-core/core"
  12. "github.com/xtls/xray-core/features/outbound"
  13. )
  14. // Manager is to manage all outbound handlers.
  15. type Manager struct {
  16. access sync.RWMutex
  17. defaultHandler outbound.Handler
  18. taggedHandler map[string]outbound.Handler
  19. untaggedHandlers []outbound.Handler
  20. running bool
  21. tagsCache *sync.Map
  22. }
  23. // New creates a new Manager.
  24. func New(ctx context.Context, config *proxyman.OutboundConfig) (*Manager, error) {
  25. m := &Manager{
  26. taggedHandler: make(map[string]outbound.Handler),
  27. tagsCache: &sync.Map{},
  28. }
  29. return m, nil
  30. }
  31. // Type implements common.HasType.
  32. func (m *Manager) Type() interface{} {
  33. return outbound.ManagerType()
  34. }
  35. // Start implements core.Feature
  36. func (m *Manager) Start() error {
  37. m.access.Lock()
  38. defer m.access.Unlock()
  39. m.running = true
  40. for _, h := range m.taggedHandler {
  41. if err := h.Start(); err != nil {
  42. return err
  43. }
  44. }
  45. for _, h := range m.untaggedHandlers {
  46. if err := h.Start(); err != nil {
  47. return err
  48. }
  49. }
  50. return nil
  51. }
  52. // Close implements core.Feature
  53. func (m *Manager) Close() error {
  54. m.access.Lock()
  55. defer m.access.Unlock()
  56. m.running = false
  57. var errs []error
  58. for _, h := range m.taggedHandler {
  59. errs = append(errs, h.Close())
  60. }
  61. for _, h := range m.untaggedHandlers {
  62. errs = append(errs, h.Close())
  63. }
  64. return errors.Combine(errs...)
  65. }
  66. // GetDefaultHandler implements outbound.Manager.
  67. func (m *Manager) GetDefaultHandler() outbound.Handler {
  68. m.access.RLock()
  69. defer m.access.RUnlock()
  70. if m.defaultHandler == nil {
  71. return nil
  72. }
  73. return m.defaultHandler
  74. }
  75. // GetHandler implements outbound.Manager.
  76. func (m *Manager) GetHandler(tag string) outbound.Handler {
  77. m.access.RLock()
  78. defer m.access.RUnlock()
  79. if handler, found := m.taggedHandler[tag]; found {
  80. return handler
  81. }
  82. return nil
  83. }
  84. // AddHandler implements outbound.Manager.
  85. func (m *Manager) AddHandler(ctx context.Context, handler outbound.Handler) error {
  86. m.access.Lock()
  87. defer m.access.Unlock()
  88. m.tagsCache = &sync.Map{}
  89. if m.defaultHandler == nil {
  90. m.defaultHandler = handler
  91. }
  92. tag := handler.Tag()
  93. if len(tag) > 0 {
  94. if _, found := m.taggedHandler[tag]; found {
  95. return newError("existing tag found: " + tag)
  96. }
  97. m.taggedHandler[tag] = handler
  98. } else {
  99. m.untaggedHandlers = append(m.untaggedHandlers, handler)
  100. }
  101. if m.running {
  102. return handler.Start()
  103. }
  104. return nil
  105. }
  106. // RemoveHandler implements outbound.Manager.
  107. func (m *Manager) RemoveHandler(ctx context.Context, tag string) error {
  108. if tag == "" {
  109. return common.ErrNoClue
  110. }
  111. m.access.Lock()
  112. defer m.access.Unlock()
  113. m.tagsCache = &sync.Map{}
  114. delete(m.taggedHandler, tag)
  115. if m.defaultHandler != nil && m.defaultHandler.Tag() == tag {
  116. m.defaultHandler = nil
  117. }
  118. return nil
  119. }
  120. // Select implements outbound.HandlerSelector.
  121. func (m *Manager) Select(selectors []string) []string {
  122. key := strings.Join(selectors, ",")
  123. if cache, ok := m.tagsCache.Load(key); ok {
  124. return cache.([]string)
  125. }
  126. m.access.RLock()
  127. defer m.access.RUnlock()
  128. tags := make([]string, 0, len(selectors))
  129. for tag := range m.taggedHandler {
  130. for _, selector := range selectors {
  131. if strings.HasPrefix(tag, selector) {
  132. tags = append(tags, tag)
  133. break
  134. }
  135. }
  136. }
  137. sort.Strings(tags)
  138. m.tagsCache.Store(key, tags)
  139. return tags
  140. }
  141. func init() {
  142. common.Must(common.RegisterConfig((*proxyman.OutboundConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  143. return New(ctx, config.(*proxyman.OutboundConfig))
  144. }))
  145. common.Must(common.RegisterConfig((*core.OutboundHandlerConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  146. return NewHandler(ctx, config.(*core.OutboundHandlerConfig))
  147. }))
  148. }