package router import ( "context" "regexp" "strings" "github.com/xtls/xray-core/common/errors" "github.com/xtls/xray-core/features/outbound" "github.com/xtls/xray-core/features/routing" ) type Rule struct { Tag string RuleTag string Balancer *Balancer Condition Condition } func (r *Rule) GetTag() (string, error) { if r.Balancer != nil { return r.Balancer.PickOutbound() } return r.Tag, nil } // Apply checks rule matching of current routing context. func (r *Rule) Apply(ctx routing.Context) bool { return r.Condition.Apply(ctx) } func (rr *RoutingRule) BuildCondition() (Condition, error) { conds := NewConditionChan() if len(rr.Domain) > 0 { switch rr.DomainMatcher { case "linear": matcher, err := NewDomainMatcher(rr.Domain) if err != nil { return nil, errors.New("failed to build domain condition").Base(err) } conds.Add(matcher) case "mph", "hybrid": fallthrough default: matcher, err := NewMphMatcherGroup(rr.Domain) if err != nil { return nil, errors.New("failed to build domain condition with MphDomainMatcher").Base(err) } errors.LogDebug(context.Background(), "MphDomainMatcher is enabled for ", len(rr.Domain), " domain rule(s)") conds.Add(matcher) } } if len(rr.UserEmail) > 0 { conds.Add(NewUserMatcher(rr.UserEmail)) } if len(rr.InboundTag) > 0 { conds.Add(NewInboundTagMatcher(rr.InboundTag)) } if rr.PortList != nil { conds.Add(NewPortMatcher(rr.PortList, "target")) } if rr.SourcePortList != nil { conds.Add(NewPortMatcher(rr.SourcePortList, "source")) } if rr.LocalPortList != nil { conds.Add(NewPortMatcher(rr.LocalPortList, "local")) } if len(rr.Networks) > 0 { conds.Add(NewNetworkMatcher(rr.Networks)) } if len(rr.Geoip) > 0 { cond, err := NewMultiGeoIPMatcher(rr.Geoip, "target") if err != nil { return nil, err } conds.Add(cond) } if len(rr.SourceGeoip) > 0 { cond, err := NewMultiGeoIPMatcher(rr.SourceGeoip, "source") if err != nil { return nil, err } conds.Add(cond) } if len(rr.LocalGeoip) > 0 { cond, err := NewMultiGeoIPMatcher(rr.LocalGeoip, "local") if err != nil { return nil, err } conds.Add(cond) errors.LogWarning(context.Background(), "Due to some limitations, in UDP connections, localIP is always equal to listen interface IP, so \"localIP\" rule condition does not work properly on UDP inbound connections that listen on all interfaces") } if len(rr.Protocol) > 0 { conds.Add(NewProtocolMatcher(rr.Protocol)) } if len(rr.Attributes) > 0 { configuredKeys := make(map[string]*regexp.Regexp) for key, value := range rr.Attributes { configuredKeys[strings.ToLower(key)] = regexp.MustCompile(value) } conds.Add(&AttributeMatcher{configuredKeys}) } if conds.Len() == 0 { return nil, errors.New("this rule has no effective fields").AtWarning() } return conds, nil } // Build builds the balancing rule func (br *BalancingRule) Build(ohm outbound.Manager, dispatcher routing.Dispatcher) (*Balancer, error) { switch strings.ToLower(br.Strategy) { case "leastping": return &Balancer{ selectors: br.OutboundSelector, strategy: &LeastPingStrategy{}, fallbackTag: br.FallbackTag, ohm: ohm, }, nil case "roundrobin": return &Balancer{ selectors: br.OutboundSelector, strategy: &RoundRobinStrategy{FallbackTag: br.FallbackTag}, fallbackTag: br.FallbackTag, ohm: ohm, }, nil case "leastload": i, err := br.StrategySettings.GetInstance() if err != nil { return nil, err } s, ok := i.(*StrategyLeastLoadConfig) if !ok { return nil, errors.New("not a StrategyLeastLoadConfig").AtError() } leastLoadStrategy := NewLeastLoadStrategy(s) return &Balancer{ selectors: br.OutboundSelector, ohm: ohm, fallbackTag: br.FallbackTag, strategy: leastLoadStrategy, }, nil case "random": fallthrough case "": return &Balancer{ selectors: br.OutboundSelector, ohm: ohm, fallbackTag: br.FallbackTag, strategy: &RandomStrategy{FallbackTag: br.FallbackTag}, }, nil default: return nil, errors.New("unrecognized balancer type") } }