123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 |
- package inbound
- import (
- "context"
- "sync"
- "time"
- "github.com/xtls/xray-core/app/proxyman"
- "github.com/xtls/xray-core/common/dice"
- "github.com/xtls/xray-core/common/errors"
- "github.com/xtls/xray-core/common/mux"
- "github.com/xtls/xray-core/common/net"
- "github.com/xtls/xray-core/common/serial"
- "github.com/xtls/xray-core/common/task"
- "github.com/xtls/xray-core/core"
- "github.com/xtls/xray-core/proxy"
- "github.com/xtls/xray-core/transport/internet"
- "google.golang.org/protobuf/proto"
- )
- type DynamicInboundHandler struct {
- tag string
- v *core.Instance
- proxyConfig interface{}
- receiverConfig *proxyman.ReceiverConfig
- streamSettings *internet.MemoryStreamConfig
- portMutex sync.Mutex
- portsInUse map[net.Port]struct{}
- workerMutex sync.RWMutex
- worker []worker
- lastRefresh time.Time
- mux *mux.Server
- task *task.Periodic
- ctx context.Context
- }
- func NewDynamicInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*DynamicInboundHandler, error) {
- v := core.MustFromContext(ctx)
- h := &DynamicInboundHandler{
- tag: tag,
- proxyConfig: proxyConfig,
- receiverConfig: receiverConfig,
- portsInUse: make(map[net.Port]struct{}),
- mux: mux.NewServer(ctx),
- v: v,
- ctx: ctx,
- }
- mss, err := internet.ToMemoryStreamConfig(receiverConfig.StreamSettings)
- if err != nil {
- return nil, errors.New("failed to parse stream settings").Base(err).AtWarning()
- }
- if receiverConfig.ReceiveOriginalDestination {
- if mss.SocketSettings == nil {
- mss.SocketSettings = &internet.SocketConfig{}
- }
- if mss.SocketSettings.Tproxy == internet.SocketConfig_Off {
- mss.SocketSettings.Tproxy = internet.SocketConfig_Redirect
- }
- mss.SocketSettings.ReceiveOriginalDestAddress = true
- }
- h.streamSettings = mss
- h.task = &task.Periodic{
- Interval: time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()),
- Execute: h.refresh,
- }
- return h, nil
- }
- func (h *DynamicInboundHandler) allocatePort() net.Port {
- allPorts := []int32{}
- for _, pr := range h.receiverConfig.PortList.Range {
- for i := pr.From; i <= pr.To; i++ {
- allPorts = append(allPorts, int32(i))
- }
- }
- h.portMutex.Lock()
- defer h.portMutex.Unlock()
- for {
- r := dice.Roll(len(allPorts))
- port := net.Port(allPorts[r])
- _, used := h.portsInUse[port]
- if !used {
- h.portsInUse[port] = struct{}{}
- return port
- }
- }
- }
- func (h *DynamicInboundHandler) closeWorkers(workers []worker) {
- ports2Del := make([]net.Port, len(workers))
- for idx, worker := range workers {
- ports2Del[idx] = worker.Port()
- if err := worker.Close(); err != nil {
- errors.LogInfoInner(h.ctx, err, "failed to close worker")
- }
- }
- h.portMutex.Lock()
- for _, port := range ports2Del {
- delete(h.portsInUse, port)
- }
- h.portMutex.Unlock()
- }
- func (h *DynamicInboundHandler) refresh() error {
- h.lastRefresh = time.Now()
- timeout := time.Minute * time.Duration(h.receiverConfig.AllocationStrategy.GetRefreshValue()) * 2
- concurrency := h.receiverConfig.AllocationStrategy.GetConcurrencyValue()
- workers := make([]worker, 0, concurrency)
- address := h.receiverConfig.Listen.AsAddress()
- if address == nil {
- address = net.AnyIP
- }
- uplinkCounter, downlinkCounter := getStatCounter(h.v, h.tag)
- for i := uint32(0); i < concurrency; i++ {
- port := h.allocatePort()
- rawProxy, err := core.CreateObject(h.v, h.proxyConfig)
- if err != nil {
- errors.LogWarningInner(h.ctx, err, "failed to create proxy instance")
- continue
- }
- p := rawProxy.(proxy.Inbound)
- nl := p.Network()
- if net.HasNetwork(nl, net.Network_TCP) {
- worker := &tcpWorker{
- tag: h.tag,
- address: address,
- port: port,
- proxy: p,
- stream: h.streamSettings,
- recvOrigDest: h.receiverConfig.ReceiveOriginalDestination,
- dispatcher: h.mux,
- sniffingConfig: h.receiverConfig.GetEffectiveSniffingSettings(),
- uplinkCounter: uplinkCounter,
- downlinkCounter: downlinkCounter,
- ctx: h.ctx,
- }
- if err := worker.Start(); err != nil {
- errors.LogWarningInner(h.ctx, err, "failed to create TCP worker")
- continue
- }
- workers = append(workers, worker)
- }
- if net.HasNetwork(nl, net.Network_UDP) {
- worker := &udpWorker{
- tag: h.tag,
- proxy: p,
- address: address,
- port: port,
- dispatcher: h.mux,
- sniffingConfig: h.receiverConfig.GetEffectiveSniffingSettings(),
- uplinkCounter: uplinkCounter,
- downlinkCounter: downlinkCounter,
- stream: h.streamSettings,
- ctx: h.ctx,
- }
- if err := worker.Start(); err != nil {
- errors.LogWarningInner(h.ctx, err, "failed to create UDP worker")
- continue
- }
- workers = append(workers, worker)
- }
- }
- h.workerMutex.Lock()
- h.worker = workers
- h.workerMutex.Unlock()
- time.AfterFunc(timeout, func() {
- h.closeWorkers(workers)
- })
- return nil
- }
- func (h *DynamicInboundHandler) Start() error {
- return h.task.Start()
- }
- func (h *DynamicInboundHandler) Close() error {
- return h.task.Close()
- }
- func (h *DynamicInboundHandler) GetRandomInboundProxy() (interface{}, net.Port, int) {
- h.workerMutex.RLock()
- defer h.workerMutex.RUnlock()
- if len(h.worker) == 0 {
- return nil, 0, 0
- }
- w := h.worker[dice.Roll(len(h.worker))]
- expire := h.receiverConfig.AllocationStrategy.GetRefreshValue() - uint32(time.Since(h.lastRefresh)/time.Minute)
- return w.Proxy(), w.Port(), int(expire)
- }
- func (h *DynamicInboundHandler) Tag() string {
- return h.tag
- }
- // ReceiverSettings implements inbound.Handler.
- func (h *DynamicInboundHandler) ReceiverSettings() *serial.TypedMessage {
- return serial.ToTypedMessage(h.receiverConfig)
- }
- // ProxySettings implements inbound.Handler.
- func (h *DynamicInboundHandler) ProxySettings() *serial.TypedMessage {
- if v, ok := h.proxyConfig.(proto.Message); ok {
- return serial.ToTypedMessage(v)
- }
- return nil
- }
|