| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- package inbound
- import (
- "context"
- "github.com/xtls/xray-core/app/proxyman"
- "github.com/xtls/xray-core/common"
- "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/core"
- "github.com/xtls/xray-core/features/policy"
- "github.com/xtls/xray-core/features/stats"
- "github.com/xtls/xray-core/proxy"
- "github.com/xtls/xray-core/transport/internet"
- )
- func getStatCounter(v *core.Instance, tag string) (stats.Counter, stats.Counter) {
- var uplinkCounter stats.Counter
- var downlinkCounter stats.Counter
- policy := v.GetFeature(policy.ManagerType()).(policy.Manager)
- if len(tag) > 0 && policy.ForSystem().Stats.InboundUplink {
- statsManager := v.GetFeature(stats.ManagerType()).(stats.Manager)
- name := "inbound>>>" + tag + ">>>traffic>>>uplink"
- c, _ := stats.GetOrRegisterCounter(statsManager, name)
- if c != nil {
- uplinkCounter = c
- }
- }
- if len(tag) > 0 && policy.ForSystem().Stats.InboundDownlink {
- statsManager := v.GetFeature(stats.ManagerType()).(stats.Manager)
- name := "inbound>>>" + tag + ">>>traffic>>>downlink"
- c, _ := stats.GetOrRegisterCounter(statsManager, name)
- if c != nil {
- downlinkCounter = c
- }
- }
- return uplinkCounter, downlinkCounter
- }
- type AlwaysOnInboundHandler struct {
- proxy proxy.Inbound
- workers []worker
- mux *mux.Server
- tag string
- }
- func NewAlwaysOnInboundHandler(ctx context.Context, tag string, receiverConfig *proxyman.ReceiverConfig, proxyConfig interface{}) (*AlwaysOnInboundHandler, error) {
- rawProxy, err := common.CreateObject(ctx, proxyConfig)
- if err != nil {
- return nil, err
- }
- p, ok := rawProxy.(proxy.Inbound)
- if !ok {
- return nil, newError("not an inbound proxy.")
- }
- h := &AlwaysOnInboundHandler{
- proxy: p,
- mux: mux.NewServer(ctx),
- tag: tag,
- }
- uplinkCounter, downlinkCounter := getStatCounter(core.MustFromContext(ctx), tag)
- nl := p.Network()
- pr := receiverConfig.PortRange
- address := receiverConfig.Listen.AsAddress()
- if address == nil {
- address = net.AnyIP
- }
- mss, err := internet.ToMemoryStreamConfig(receiverConfig.StreamSettings)
- if err != nil {
- return nil, newError("failed to parse stream config").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
- }
- if pr == nil {
- if net.HasNetwork(nl, net.Network_UNIX) {
- newError("creating unix domain socket worker on ", address).AtDebug().WriteToLog()
- worker := &dsWorker{
- address: address,
- proxy: p,
- stream: mss,
- tag: tag,
- dispatcher: h.mux,
- sniffingConfig: receiverConfig.GetEffectiveSniffingSettings(),
- uplinkCounter: uplinkCounter,
- downlinkCounter: downlinkCounter,
- ctx: ctx,
- }
- h.workers = append(h.workers, worker)
- }
- }
- if pr != nil {
- for port := pr.From; port <= pr.To; port++ {
- if net.HasNetwork(nl, net.Network_TCP) {
- newError("creating stream worker on ", address, ":", port).AtDebug().WriteToLog()
- worker := &tcpWorker{
- address: address,
- port: net.Port(port),
- proxy: p,
- stream: mss,
- recvOrigDest: receiverConfig.ReceiveOriginalDestination,
- tag: tag,
- dispatcher: h.mux,
- sniffingConfig: receiverConfig.GetEffectiveSniffingSettings(),
- uplinkCounter: uplinkCounter,
- downlinkCounter: downlinkCounter,
- ctx: ctx,
- }
- h.workers = append(h.workers, worker)
- }
- if net.HasNetwork(nl, net.Network_UDP) {
- worker := &udpWorker{
- tag: tag,
- proxy: p,
- address: address,
- port: net.Port(port),
- dispatcher: h.mux,
- sniffingConfig: receiverConfig.GetEffectiveSniffingSettings(),
- uplinkCounter: uplinkCounter,
- downlinkCounter: downlinkCounter,
- stream: mss,
- ctx: ctx,
- }
- h.workers = append(h.workers, worker)
- }
- }
- }
- return h, nil
- }
- // Start implements common.Runnable.
- func (h *AlwaysOnInboundHandler) Start() error {
- for _, worker := range h.workers {
- if err := worker.Start(); err != nil {
- return err
- }
- }
- return nil
- }
- // Close implements common.Closable.
- func (h *AlwaysOnInboundHandler) Close() error {
- var errs []error
- for _, worker := range h.workers {
- errs = append(errs, worker.Close())
- }
- errs = append(errs, h.mux.Close())
- if err := errors.Combine(errs...); err != nil {
- return newError("failed to close all resources").Base(err)
- }
- return nil
- }
- func (h *AlwaysOnInboundHandler) GetRandomInboundProxy() (interface{}, net.Port, int) {
- if len(h.workers) == 0 {
- return nil, 0, 0
- }
- w := h.workers[dice.Roll(len(h.workers))]
- return w.Proxy(), w.Port(), 9999
- }
- func (h *AlwaysOnInboundHandler) Tag() string {
- return h.tag
- }
- func (h *AlwaysOnInboundHandler) GetInbound() proxy.Inbound {
- return h.proxy
- }
|