mirror of https://github.com/XTLS/Xray-core
185 lines
4.7 KiB
Go
185 lines
4.7 KiB
Go
package inbound
|
|
|
|
import (
|
|
"context"
|
|
"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/common/net"
|
|
"github.com/xtls/xray-core/common/serial"
|
|
"github.com/xtls/xray-core/common/session"
|
|
"github.com/xtls/xray-core/core"
|
|
"github.com/xtls/xray-core/features/inbound"
|
|
)
|
|
|
|
// Manager manages all inbound handlers.
|
|
type Manager struct {
|
|
access sync.RWMutex
|
|
untaggedHandler []inbound.Handler
|
|
taggedHandlers map[string]inbound.Handler
|
|
running bool
|
|
}
|
|
|
|
// New returns a new Manager for inbound handlers.
|
|
func New(ctx context.Context, config *proxyman.InboundConfig) (*Manager, error) {
|
|
m := &Manager{
|
|
taggedHandlers: make(map[string]inbound.Handler),
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
// Type implements common.HasType.
|
|
func (*Manager) Type() interface{} {
|
|
return inbound.ManagerType()
|
|
}
|
|
|
|
// AddHandler implements inbound.Manager.
|
|
func (m *Manager) AddHandler(ctx context.Context, handler inbound.Handler) error {
|
|
m.access.Lock()
|
|
defer m.access.Unlock()
|
|
|
|
tag := handler.Tag()
|
|
if len(tag) > 0 {
|
|
if _, found := m.taggedHandlers[tag]; found {
|
|
return errors.New("existing tag found: " + tag)
|
|
}
|
|
m.taggedHandlers[tag] = handler
|
|
} else {
|
|
m.untaggedHandler = append(m.untaggedHandler, handler)
|
|
}
|
|
|
|
if m.running {
|
|
return handler.Start()
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GetHandler implements inbound.Manager.
|
|
func (m *Manager) GetHandler(ctx context.Context, tag string) (inbound.Handler, error) {
|
|
m.access.RLock()
|
|
defer m.access.RUnlock()
|
|
|
|
handler, found := m.taggedHandlers[tag]
|
|
if !found {
|
|
return nil, errors.New("handler not found: ", tag)
|
|
}
|
|
return handler, nil
|
|
}
|
|
|
|
// RemoveHandler implements inbound.Manager.
|
|
func (m *Manager) RemoveHandler(ctx context.Context, tag string) error {
|
|
if tag == "" {
|
|
return common.ErrNoClue
|
|
}
|
|
|
|
m.access.Lock()
|
|
defer m.access.Unlock()
|
|
|
|
if handler, found := m.taggedHandlers[tag]; found {
|
|
if err := handler.Close(); err != nil {
|
|
errors.LogWarningInner(ctx, err, "failed to close handler ", tag)
|
|
}
|
|
delete(m.taggedHandlers, tag)
|
|
return nil
|
|
}
|
|
|
|
return common.ErrNoClue
|
|
}
|
|
|
|
// Start implements common.Runnable.
|
|
func (m *Manager) Start() error {
|
|
m.access.Lock()
|
|
defer m.access.Unlock()
|
|
|
|
m.running = true
|
|
|
|
for _, handler := range m.taggedHandlers {
|
|
if err := handler.Start(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, handler := range m.untaggedHandler {
|
|
if err := handler.Start(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Close implements common.Closable.
|
|
func (m *Manager) Close() error {
|
|
m.access.Lock()
|
|
defer m.access.Unlock()
|
|
|
|
m.running = false
|
|
|
|
var errs []interface{}
|
|
for _, handler := range m.taggedHandlers {
|
|
if err := handler.Close(); err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
}
|
|
for _, handler := range m.untaggedHandler {
|
|
if err := handler.Close(); err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
}
|
|
|
|
if len(errs) > 0 {
|
|
return errors.New("failed to close all handlers").Base(errors.New(serial.Concat(errs...)))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// NewHandler creates a new inbound.Handler based on the given config.
|
|
func NewHandler(ctx context.Context, config *core.InboundHandlerConfig) (inbound.Handler, error) {
|
|
rawReceiverSettings, err := config.ReceiverSettings.GetInstance()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
proxySettings, err := config.ProxySettings.GetInstance()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
tag := config.Tag
|
|
|
|
receiverSettings, ok := rawReceiverSettings.(*proxyman.ReceiverConfig)
|
|
if !ok {
|
|
return nil, errors.New("not a ReceiverConfig").AtError()
|
|
}
|
|
|
|
streamSettings := receiverSettings.StreamSettings
|
|
if streamSettings != nil && streamSettings.SocketSettings != nil {
|
|
ctx = session.ContextWithSockopt(ctx, &session.Sockopt{
|
|
Mark: streamSettings.SocketSettings.Mark,
|
|
})
|
|
}
|
|
if streamSettings != nil && streamSettings.ProtocolName == "splithttp" {
|
|
ctx = session.ContextWithAllowedNetwork(ctx, net.Network_UDP)
|
|
}
|
|
|
|
allocStrategy := receiverSettings.AllocationStrategy
|
|
if allocStrategy == nil || allocStrategy.Type == proxyman.AllocationStrategy_Always {
|
|
return NewAlwaysOnInboundHandler(ctx, tag, receiverSettings, proxySettings)
|
|
}
|
|
|
|
if allocStrategy.Type == proxyman.AllocationStrategy_Random {
|
|
return NewDynamicInboundHandler(ctx, tag, receiverSettings, proxySettings)
|
|
}
|
|
return nil, errors.New("unknown allocation strategy: ", receiverSettings.AllocationStrategy.Type).AtError()
|
|
}
|
|
|
|
func init() {
|
|
common.Must(common.RegisterConfig((*proxyman.InboundConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
|
|
return New(ctx, config.(*proxyman.InboundConfig))
|
|
}))
|
|
common.Must(common.RegisterConfig((*core.InboundHandlerConfig)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
|
|
return NewHandler(ctx, config.(*core.InboundHandlerConfig))
|
|
}))
|
|
}
|