| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- package outbound
- //go:generate go run github.com/xtls/xray-core/common/errors/errorgen
- import (
- "context"
- "strings"
- "sync"
- "github.com/xtls/xray-core/app/proxyman"
- "github.com/xtls/xray-core/common"
- "github.com/xtls/xray-core/common/errors"
- "github.com/xtls/xray-core/core"
- "github.com/xtls/xray-core/features/outbound"
- )
- // Manager is to manage all outbound handlers.
- type Manager struct {
- access sync.RWMutex
- defaultHandler outbound.Handler
- taggedHandler map[string]outbound.Handler
- untaggedHandlers []outbound.Handler
- running bool
- }
- // New creates a new Manager.
- func New(ctx context.Context, config *proxyman.OutboundConfig) (*Manager, error) {
- m := &Manager{
- taggedHandler: make(map[string]outbound.Handler),
- }
- return m, nil
- }
- // Type implements common.HasType.
- func (m *Manager) Type() interface{} {
- return outbound.ManagerType()
- }
- // Start implements core.Feature
- func (m *Manager) Start() error {
- m.access.Lock()
- defer m.access.Unlock()
- m.running = true
- for _, h := range m.taggedHandler {
- if err := h.Start(); err != nil {
- return err
- }
- }
- for _, h := range m.untaggedHandlers {
- if err := h.Start(); err != nil {
- return err
- }
- }
- return nil
- }
- // Close implements core.Feature
- func (m *Manager) Close() error {
- m.access.Lock()
- defer m.access.Unlock()
- m.running = false
- var errs []error
- for _, h := range m.taggedHandler {
- errs = append(errs, h.Close())
- }
- for _, h := range m.untaggedHandlers {
- errs = append(errs, h.Close())
- }
- return errors.Combine(errs...)
- }
- // GetDefaultHandler implements outbound.Manager.
- func (m *Manager) GetDefaultHandler() outbound.Handler {
- m.access.RLock()
- defer m.access.RUnlock()
- if m.defaultHandler == nil {
- return nil
- }
- return m.defaultHandler
- }
- // GetHandler implements outbound.Manager.
- func (m *Manager) GetHandler(tag string) outbound.Handler {
- m.access.RLock()
- defer m.access.RUnlock()
- if handler, found := m.taggedHandler[tag]; found {
- return handler
- }
- return nil
- }
- // AddHandler implements outbound.Manager.
- func (m *Manager) AddHandler(ctx context.Context, handler outbound.Handler) error {
- m.access.Lock()
- defer m.access.Unlock()
- if m.defaultHandler == nil {
- m.defaultHandler = handler
- }
- tag := handler.Tag()
- if len(tag) > 0 {
- if _, found := m.taggedHandler[tag]; found {
- return newError("existing tag found: " + tag)
- }
- m.taggedHandler[tag] = handler
- } else {
- m.untaggedHandlers = append(m.untaggedHandlers, handler)
- }
- if m.running {
- return handler.Start()
- }
- return nil
- }
- // RemoveHandler implements outbound.Manager.
- func (m *Manager) RemoveHandler(ctx context.Context, tag string) error {
- if tag == "" {
- return common.ErrNoClue
- }
- m.access.Lock()
- defer m.access.Unlock()
- delete(m.taggedHandler, tag)
- if m.defaultHandler != nil && m.defaultHandler.Tag() == tag {
- m.defaultHandler = nil
- }
- return nil
- }
- // Select implements outbound.HandlerSelector.
- func (m *Manager) Select(selectors []string) []string {
- m.access.RLock()
- defer m.access.RUnlock()
- tags := make([]string, 0, len(selectors))
- for tag := range m.taggedHandler {
- match := false
- for _, selector := range selectors {
- if strings.HasPrefix(tag, selector) {
- match = true
- break
- }
- }
- if match {
- tags = append(tags, tag)
- }
- }
- return tags
- }
- func init() {
- common.Must(common.RegisterConfig((*proxyman.OutboundConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
- return New(ctx, config.(*proxyman.OutboundConfig))
- }))
- common.Must(common.RegisterConfig((*core.OutboundHandlerConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
- return NewHandler(ctx, config.(*core.OutboundHandlerConfig))
- }))
- }
|