mirror of https://github.com/v2ray/v2ray-core
AnyCondition
parent
ee72984f78
commit
4f5743604a
|
@ -5,21 +5,8 @@ package rules
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"github.com/v2ray/v2ray-core/common/log"
|
"github.com/v2ray/v2ray-core/common/log"
|
||||||
v2net "github.com/v2ray/v2ray-core/common/net"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ChinaSitesCondition struct {
|
|
||||||
}
|
|
||||||
|
|
||||||
func (this *ChinaSitesCondition) Apply(dest v2net.Destination) bool {
|
|
||||||
for _, cond := range chinaSitesConds {
|
|
||||||
if cond.Apply(dest) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseChinaSitesRule(data []byte) (*Rule, error) {
|
func parseChinaSitesRule(data []byte) (*Rule, error) {
|
||||||
rawRule := new(JsonRule)
|
rawRule := new(JsonRule)
|
||||||
err := json.Unmarshal(data, rawRule)
|
err := json.Unmarshal(data, rawRule)
|
||||||
|
@ -29,7 +16,7 @@ func parseChinaSitesRule(data []byte) (*Rule, error) {
|
||||||
}
|
}
|
||||||
return &Rule{
|
return &Rule{
|
||||||
Tag: rawRule.OutboundTag,
|
Tag: rawRule.OutboundTag,
|
||||||
Condition: &ChinaSitesCondition{},
|
Condition: ChinaSitesConds,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,7 +35,7 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
chinaSitesConds []Condition
|
ChinaSitesConds Condition
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -360,12 +347,15 @@ func init() {
|
||||||
anySubDomain + "zhubajie" + dotCom,
|
anySubDomain + "zhubajie" + dotCom,
|
||||||
}
|
}
|
||||||
|
|
||||||
chinaSitesConds = make([]Condition, len(regexpDomains))
|
conds := make([]Condition, len(regexpDomains))
|
||||||
for idx, pattern := range regexpDomains {
|
for idx, pattern := range regexpDomains {
|
||||||
matcher, err := NewRegexpDomainMatcher(pattern)
|
matcher, err := NewRegexpDomainMatcher(pattern)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
chinaSitesConds[idx] = matcher
|
conds[idx] = matcher
|
||||||
}
|
}
|
||||||
|
|
||||||
|
anyConds := AnyCondition(conds)
|
||||||
|
ChinaSitesConds = &anyConds
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,7 @@ func makeDomainDestination(domain string) v2net.Destination {
|
||||||
func TestChinaSites(t *testing.T) {
|
func TestChinaSites(t *testing.T) {
|
||||||
v2testing.Current(t)
|
v2testing.Current(t)
|
||||||
|
|
||||||
rule := new(ChinaSitesCondition)
|
rule := ChinaSitesConds
|
||||||
assert.Bool(rule.Apply(makeDomainDestination("v.qq.com"))).IsTrue()
|
assert.Bool(rule.Apply(makeDomainDestination("v.qq.com"))).IsTrue()
|
||||||
assert.Bool(rule.Apply(makeDomainDestination("www.163.com"))).IsTrue()
|
assert.Bool(rule.Apply(makeDomainDestination("www.163.com"))).IsTrue()
|
||||||
assert.Bool(rule.Apply(makeDomainDestination("ngacn.cc"))).IsTrue()
|
assert.Bool(rule.Apply(makeDomainDestination("ngacn.cc"))).IsTrue()
|
||||||
|
|
|
@ -37,6 +37,31 @@ func (this *ConditionChan) Len() int {
|
||||||
return len(*this)
|
return len(*this)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AnyCondition []Condition
|
||||||
|
|
||||||
|
func NewAnyCondition() *AnyCondition {
|
||||||
|
var anyCond AnyCondition = make([]Condition, 0, 8)
|
||||||
|
return &anyCond
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *AnyCondition) Add(cond Condition) *AnyCondition {
|
||||||
|
*this = append(*this, cond)
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *AnyCondition) Apply(dest v2net.Destination) bool {
|
||||||
|
for _, cond := range *this {
|
||||||
|
if cond.Apply(dest) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *AnyCondition) Len() int {
|
||||||
|
return len(*this)
|
||||||
|
}
|
||||||
|
|
||||||
type PlainDomainMatcher struct {
|
type PlainDomainMatcher struct {
|
||||||
pattern serial.StringLiteral
|
pattern serial.StringLiteral
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,6 +34,7 @@ func parseFieldRule(msg json.RawMessage) (*Rule, error) {
|
||||||
conds := NewConditionChan()
|
conds := NewConditionChan()
|
||||||
|
|
||||||
if rawFieldRule.Domain != nil && rawFieldRule.Domain.Len() > 0 {
|
if rawFieldRule.Domain != nil && rawFieldRule.Domain.Len() > 0 {
|
||||||
|
anyCond := NewAnyCondition()
|
||||||
for _, rawDomain := range *(rawFieldRule.Domain) {
|
for _, rawDomain := range *(rawFieldRule.Domain) {
|
||||||
var matcher Condition
|
var matcher Condition
|
||||||
if strings.HasPrefix(rawDomain.String(), "regexp:") {
|
if strings.HasPrefix(rawDomain.String(), "regexp:") {
|
||||||
|
@ -45,19 +46,22 @@ func parseFieldRule(msg json.RawMessage) (*Rule, error) {
|
||||||
} else {
|
} else {
|
||||||
matcher = NewPlainDomainMatcher(rawDomain.String())
|
matcher = NewPlainDomainMatcher(rawDomain.String())
|
||||||
}
|
}
|
||||||
conds.Add(matcher)
|
anyCond.Add(matcher)
|
||||||
}
|
}
|
||||||
|
conds.Add(anyCond)
|
||||||
}
|
}
|
||||||
|
|
||||||
if rawFieldRule.IP != nil && rawFieldRule.IP.Len() > 0 {
|
if rawFieldRule.IP != nil && rawFieldRule.IP.Len() > 0 {
|
||||||
|
anyCond := NewAnyCondition()
|
||||||
for _, ipStr := range *(rawFieldRule.IP) {
|
for _, ipStr := range *(rawFieldRule.IP) {
|
||||||
cidrMatcher, err := NewCIDRMatcher(ipStr.String())
|
cidrMatcher, err := NewCIDRMatcher(ipStr.String())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("Router: Invalid IP range in router rule: ", err)
|
log.Error("Router: Invalid IP range in router rule: ", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
conds.Add(cidrMatcher)
|
anyCond.Add(cidrMatcher)
|
||||||
}
|
}
|
||||||
|
conds.Add(anyCond)
|
||||||
}
|
}
|
||||||
if rawFieldRule.Port != nil {
|
if rawFieldRule.Port != nil {
|
||||||
conds.Add(NewPortMatcher(*rawFieldRule.Port))
|
conds.Add(NewPortMatcher(*rawFieldRule.Port))
|
||||||
|
@ -74,7 +78,7 @@ func parseFieldRule(msg json.RawMessage) (*Rule, error) {
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseRule(msg json.RawMessage) *Rule {
|
func ParseRule(msg json.RawMessage) *Rule {
|
||||||
rawRule := new(JsonRule)
|
rawRule := new(JsonRule)
|
||||||
err := json.Unmarshal(msg, rawRule)
|
err := json.Unmarshal(msg, rawRule)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -121,7 +125,7 @@ func init() {
|
||||||
}
|
}
|
||||||
config := NewRouterRuleConfig()
|
config := NewRouterRuleConfig()
|
||||||
for _, rawRule := range jsonConfig.RuleList {
|
for _, rawRule := range jsonConfig.RuleList {
|
||||||
rule := parseRule(rawRule)
|
rule := ParseRule(rawRule)
|
||||||
config.Add(rule)
|
config.Add(rule)
|
||||||
}
|
}
|
||||||
return config, nil
|
return config, nil
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
// +build json
|
||||||
|
|
||||||
|
package rules_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
. "github.com/v2ray/v2ray-core/app/router/rules"
|
||||||
|
v2net "github.com/v2ray/v2ray-core/common/net"
|
||||||
|
v2testing "github.com/v2ray/v2ray-core/testing"
|
||||||
|
"github.com/v2ray/v2ray-core/testing/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestFieldRule(t *testing.T) {
|
||||||
|
v2testing.Current(t)
|
||||||
|
|
||||||
|
rule := ParseRule([]byte(`{
|
||||||
|
"type": "field",
|
||||||
|
"domain": [
|
||||||
|
"ooxx.com",
|
||||||
|
"oxox.com"
|
||||||
|
],
|
||||||
|
"outboundTag": "direct"
|
||||||
|
}`))
|
||||||
|
assert.Pointer(rule).IsNotNil()
|
||||||
|
assert.Bool(rule.Apply(v2net.TCPDestination(v2net.DomainAddress("www.ooxx.com"), 80))).IsTrue()
|
||||||
|
}
|
Loading…
Reference in New Issue