mirror of
https://github.com/daeuniverse/dae.git
synced 2025-01-08 14:34:08 +07:00
243 lines
5.8 KiB
Go
243 lines
5.8 KiB
Go
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
* Copyright (c) 2023, v2rayA Organization <team@v2raya.org>
|
|
*/
|
|
|
|
package domain_matcher
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/sirupsen/logrus"
|
|
"github.com/v2rayA/dae/common/consts"
|
|
"github.com/v2rayA/dae/component/routing"
|
|
"github.com/v2rayA/dae/config"
|
|
"github.com/v2rayA/dae/pkg/config_parser"
|
|
"hash/fnv"
|
|
"math/rand"
|
|
"reflect"
|
|
"testing"
|
|
)
|
|
|
|
var TestSample = []string{
|
|
"9game.cn",
|
|
"aliapp.org",
|
|
"alibaba-inc.com",
|
|
"alibaba.com",
|
|
"alibabacapital.com",
|
|
"alibabacorp.com",
|
|
"alibabadoctor.com",
|
|
"alibabafuturehotel.com",
|
|
"alibabagroup.com",
|
|
"alibabaplanet.com",
|
|
"alibabaued.com",
|
|
"alibabausercontent.com",
|
|
"alifanyi.com",
|
|
"alihealth.com.cn",
|
|
"alihealth.hk",
|
|
"aliimg.com",
|
|
"51y5.net",
|
|
"a.adtng.com",
|
|
"aaxads.com",
|
|
"addthisedge.com",
|
|
"adtrue.com",
|
|
"ad-srv.net",
|
|
"ad.api.moji.com",
|
|
"ad.wang502.com",
|
|
"adbutter.net",
|
|
"ads.trafficjunky.net",
|
|
"adtechus.com",
|
|
"adxprtz.com",
|
|
"cdn.advertserve.com",
|
|
"cdn.banclip.com",
|
|
"cfts1tifqr.com",
|
|
"contentabc.com",
|
|
"cretgate.com",
|
|
"ero-advertising.com",
|
|
"eroadvertising.com",
|
|
"exoclick.com",
|
|
"exosrv.com",
|
|
"go2.global",
|
|
"img-bss.csdn.net",
|
|
"imglnkc.com",
|
|
"imglnkd.com",
|
|
"innovid.com",
|
|
"ja2.gamersky.com",
|
|
"jl3.yjaxa.top",
|
|
"juicyads.com",
|
|
"kepler-37b.com",
|
|
"lqc006.com",
|
|
"moat.com",
|
|
"moatads.com",
|
|
"realsrv.com",
|
|
"s4yxaqyq95.com",
|
|
"shhs-ydd8x2.yjrmss.cn",
|
|
"static.javhd.com",
|
|
"tm-banners.gamingadult.com",
|
|
"trafficfactory.biz",
|
|
"tsyndicate.com",
|
|
"abchina.com",
|
|
"bankcomm.com",
|
|
"bankofbeijing.com.cn",
|
|
"bosc.cn",
|
|
"bsb.com.cn",
|
|
"ccb.com",
|
|
"cgbchina.com.cn",
|
|
"cib.com.cn",
|
|
"citibank.com.cn",
|
|
"cmbc.com.cn",
|
|
"hsbc.com.cn",
|
|
"hxb.com.cn",
|
|
"njcb.com.cn",
|
|
"psbc.com",
|
|
"spdb.com.cn",
|
|
"whccb.com",
|
|
}
|
|
|
|
type RoutingMatcherBuilder struct {
|
|
*routing.DefaultMatcherBuilder
|
|
outboundName2Id map[string]uint8
|
|
simulatedDomainSet []routing.DomainSet
|
|
Fallback string
|
|
|
|
err error
|
|
}
|
|
|
|
func (b *RoutingMatcherBuilder) OutboundToId(outbound string) uint8 {
|
|
h := fnv.New64()
|
|
h.Write([]byte(outbound))
|
|
return uint8(h.Sum64() & 0xFF)
|
|
}
|
|
|
|
func (b *RoutingMatcherBuilder) AddDomain(f *config_parser.Function, key string, values []string, outbound *routing.Outbound) {
|
|
if b.err != nil {
|
|
return
|
|
}
|
|
switch consts.RoutingDomainKey(key) {
|
|
case consts.RoutingDomainKey_Regex,
|
|
consts.RoutingDomainKey_Full,
|
|
consts.RoutingDomainKey_Keyword,
|
|
consts.RoutingDomainKey_Suffix:
|
|
default:
|
|
b.err = fmt.Errorf("AddDomain: unsupported key: %v", key)
|
|
return
|
|
}
|
|
b.simulatedDomainSet = append(b.simulatedDomainSet, routing.DomainSet{
|
|
Key: consts.RoutingDomainKey(key),
|
|
RuleIndex: len(b.simulatedDomainSet),
|
|
Domains: values,
|
|
})
|
|
}
|
|
|
|
func getDomain() (simulatedDomainSet []routing.DomainSet, err error) {
|
|
var rules []*config_parser.RoutingRule
|
|
sections, err := config_parser.Parse(`
|
|
routing {
|
|
pname(NetworkManager, dnsmasq, systemd-resolved) -> must_direct # Traffic of DNS in local must be direct to avoid loop when binding to WAN.
|
|
pname(sogou-qimpanel, sogou-qimpanel-watchdog) -> block
|
|
ip(geoip:private, 224.0.0.0/3, 'ff00::/8') -> direct # Put it in front unless you know what you're doing.
|
|
domain(geosite:bing)->us
|
|
domain(full:dns.google) && port(53) -> direct
|
|
domain(geosite:category-ads-all) -> block
|
|
ip(geoip:private) -> direct
|
|
ip(geoip:cn) -> direct
|
|
domain(geosite:cn) -> direct
|
|
fallback: my_group
|
|
}`)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var r config.Routing
|
|
if err = config.RoutingRuleAndParamParser(reflect.ValueOf(&r), sections[0]); err != nil {
|
|
return nil, err
|
|
}
|
|
if rules, err = routing.ApplyRulesOptimizers(r.Rules,
|
|
&routing.RefineFunctionParamKeyOptimizer{},
|
|
&routing.DatReaderOptimizer{Logger: logrus.StandardLogger()},
|
|
&routing.MergeAndSortRulesOptimizer{},
|
|
&routing.DeduplicateParamsOptimizer{},
|
|
); err != nil {
|
|
return nil, fmt.Errorf("ApplyRulesOptimizers error:\n%w", err)
|
|
}
|
|
builder := RoutingMatcherBuilder{}
|
|
if err = routing.ApplyMatcherBuilder(logrus.StandardLogger(), &builder, rules, r.Fallback); err != nil {
|
|
return nil, fmt.Errorf("ApplyMatcherBuilder: %w", err)
|
|
}
|
|
return builder.simulatedDomainSet, nil
|
|
}
|
|
|
|
func BenchmarkBruteforce(b *testing.B) {
|
|
b.StopTimer()
|
|
logrus.SetLevel(logrus.WarnLevel)
|
|
simulatedDomainSet, err := getDomain()
|
|
if err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
bf := NewBruteforce(consts.MaxMatchSetLen)
|
|
for _, domains := range simulatedDomainSet {
|
|
bf.AddSet(domains.RuleIndex, domains.Domains, domains.Key)
|
|
}
|
|
if err = bf.Build(); err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
b.StartTimer()
|
|
runBenchmark(b, bf)
|
|
}
|
|
|
|
func BenchmarkGoRegexpNfa(b *testing.B) {
|
|
b.StopTimer()
|
|
logrus.SetLevel(logrus.WarnLevel)
|
|
simulatedDomainSet, err := getDomain()
|
|
if err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
nfa := NewGoRegexpNfa(consts.MaxMatchSetLen)
|
|
for _, domains := range simulatedDomainSet {
|
|
nfa.AddSet(domains.RuleIndex, domains.Domains, domains.Key)
|
|
}
|
|
if err = nfa.Build(); err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
b.StartTimer()
|
|
runBenchmark(b, nfa)
|
|
}
|
|
|
|
func BenchmarkAhocorasickSlimtrie(b *testing.B) {
|
|
b.StopTimer()
|
|
logrus.SetLevel(logrus.WarnLevel)
|
|
simulatedDomainSet, err := getDomain()
|
|
if err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
ahocorasick := NewAhocorasickSlimtrie(consts.MaxMatchSetLen)
|
|
for _, domains := range simulatedDomainSet {
|
|
ahocorasick.AddSet(domains.RuleIndex, domains.Domains, domains.Key)
|
|
}
|
|
if err = ahocorasick.Build(); err != nil {
|
|
b.Fatal(err)
|
|
}
|
|
b.StartTimer()
|
|
runBenchmark(b, ahocorasick)
|
|
}
|
|
|
|
func runBenchmark(b *testing.B, matcher routing.DomainMatcher) {
|
|
rand.Seed(100)
|
|
for i := 0; i < b.N; i++ {
|
|
sample := TestSample[rand.Intn(len(TestSample))]
|
|
choice := rand.Intn(10)
|
|
switch {
|
|
case choice < 4:
|
|
addN := rand.Intn(5)
|
|
buf := make([]byte, addN)
|
|
for i := range buf {
|
|
buf[i] = 'a' + byte(rand.Intn('z'-'a'))
|
|
}
|
|
sample = string(buf) + "." + sample
|
|
case choice >= 4 && choice < 6:
|
|
k := rand.Intn(len(sample))
|
|
sample = sample[k:]
|
|
default:
|
|
}
|
|
matcher.MatchDomainBitmap(sample)
|
|
}
|
|
}
|