2022-03-26 18:34:15 +08:00
|
|
|
package provider
|
|
|
|
|
|
|
|
import (
|
2024-07-26 22:30:42 +08:00
|
|
|
"errors"
|
|
|
|
"io"
|
2024-07-27 23:54:28 +08:00
|
|
|
"strings"
|
2024-07-26 22:30:42 +08:00
|
|
|
|
2023-11-03 21:01:45 +08:00
|
|
|
"github.com/metacubex/mihomo/component/trie"
|
|
|
|
C "github.com/metacubex/mihomo/constant"
|
2024-07-27 10:36:11 +08:00
|
|
|
P "github.com/metacubex/mihomo/constant/provider"
|
2023-11-03 21:01:45 +08:00
|
|
|
"github.com/metacubex/mihomo/log"
|
2024-07-28 10:07:37 +08:00
|
|
|
|
|
|
|
"golang.org/x/exp/slices"
|
2022-03-26 18:34:15 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type domainStrategy struct {
|
2023-04-01 14:11:09 +08:00
|
|
|
count int
|
|
|
|
domainTrie *trie.DomainTrie[struct{}]
|
|
|
|
domainSet *trie.DomainSet
|
2022-03-26 18:34:15 +08:00
|
|
|
}
|
|
|
|
|
2024-07-27 10:36:11 +08:00
|
|
|
func (d *domainStrategy) Behavior() P.RuleBehavior {
|
|
|
|
return P.Domain
|
|
|
|
}
|
|
|
|
|
2023-01-20 16:29:08 +08:00
|
|
|
func (d *domainStrategy) ShouldFindProcess() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-03-26 18:34:15 +08:00
|
|
|
func (d *domainStrategy) Match(metadata *C.Metadata) bool {
|
2023-04-01 14:11:09 +08:00
|
|
|
return d.domainSet != nil && d.domainSet.Has(metadata.RuleHost())
|
2022-03-26 18:34:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *domainStrategy) Count() int {
|
|
|
|
return d.count
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *domainStrategy) ShouldResolveIP() bool {
|
2022-05-18 18:43:44 +08:00
|
|
|
return false
|
2022-03-26 18:34:15 +08:00
|
|
|
}
|
|
|
|
|
2023-04-01 14:11:09 +08:00
|
|
|
func (d *domainStrategy) Reset() {
|
|
|
|
d.domainTrie = trie.New[struct{}]()
|
|
|
|
d.domainSet = nil
|
|
|
|
d.count = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *domainStrategy) Insert(rule string) {
|
2024-07-27 23:54:28 +08:00
|
|
|
if strings.ContainsRune(rule, '/') {
|
|
|
|
log.Warnln("invalid domain:[%s]", rule)
|
|
|
|
return
|
|
|
|
}
|
2023-04-01 14:11:09 +08:00
|
|
|
err := d.domainTrie.Insert(rule, struct{}{})
|
|
|
|
if err != nil {
|
|
|
|
log.Warnln("invalid domain:[%s]", rule)
|
|
|
|
} else {
|
|
|
|
d.count++
|
2023-04-01 12:15:03 +08:00
|
|
|
}
|
2023-04-01 14:11:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *domainStrategy) FinishInsert() {
|
|
|
|
d.domainSet = d.domainTrie.NewDomainSet()
|
|
|
|
d.domainTrie = nil
|
2022-03-26 18:34:15 +08:00
|
|
|
}
|
|
|
|
|
2024-07-27 10:36:11 +08:00
|
|
|
func (d *domainStrategy) FromMrs(r io.Reader, count int) error {
|
|
|
|
domainSet, err := trie.ReadDomainSetBin(r)
|
2024-07-26 22:30:42 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-07-27 10:36:11 +08:00
|
|
|
d.count = count
|
2024-07-26 22:30:42 +08:00
|
|
|
d.domainSet = domainSet
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *domainStrategy) WriteMrs(w io.Writer) error {
|
|
|
|
if d.domainSet == nil {
|
|
|
|
return errors.New("nil domainSet")
|
|
|
|
}
|
2024-07-27 10:36:11 +08:00
|
|
|
return d.domainSet.WriteBin(w)
|
2024-07-26 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2024-07-28 10:07:37 +08:00
|
|
|
func (d *domainStrategy) DumpMrs(f func(key string) bool) {
|
|
|
|
if d.domainSet != nil {
|
|
|
|
var keys []string
|
|
|
|
d.domainSet.Foreach(func(key string) bool {
|
|
|
|
keys = append(keys, key)
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
slices.Sort(keys)
|
|
|
|
|
|
|
|
for _, key := range keys {
|
|
|
|
if _, ok := slices.BinarySearch(keys, "+."+key); ok {
|
|
|
|
continue // ignore the rules added by trie internal processing
|
|
|
|
}
|
|
|
|
if !f(key) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-26 22:30:42 +08:00
|
|
|
var _ mrsRuleStrategy = (*domainStrategy)(nil)
|
|
|
|
|
2022-03-26 18:34:15 +08:00
|
|
|
func NewDomainStrategy() *domainStrategy {
|
2022-05-18 18:43:44 +08:00
|
|
|
return &domainStrategy{}
|
2022-03-26 18:34:15 +08:00
|
|
|
}
|