2023-02-18 17:27:28 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2024-01-04 16:28:16 +07:00
|
|
|
* Copyright (c) 2022-2024, daeuniverse Organization <dae@v2raya.org>
|
2023-02-18 17:27:28 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package domain_matcher
|
|
|
|
|
|
|
|
import (
|
2023-02-19 13:08:13 +07:00
|
|
|
"fmt"
|
2023-03-14 14:01:55 +07:00
|
|
|
"github.com/daeuniverse/dae/common/consts"
|
|
|
|
"github.com/daeuniverse/dae/component/routing"
|
2023-02-18 17:27:28 +07:00
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Bruteforce struct {
|
|
|
|
simulatedDomainSet []routing.DomainSet
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
|
2023-02-19 13:08:13 +07:00
|
|
|
func NewBruteforce(bitLength int) *Bruteforce {
|
2023-02-18 17:27:28 +07:00
|
|
|
return &Bruteforce{
|
2023-02-19 13:08:13 +07:00
|
|
|
simulatedDomainSet: make([]routing.DomainSet, bitLength),
|
2023-02-18 17:27:28 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
func (n *Bruteforce) AddSet(bitIndex int, patterns []string, typ consts.RoutingDomainKey) {
|
2023-02-19 13:08:13 +07:00
|
|
|
if n.err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if len(n.simulatedDomainSet[bitIndex].Domains) != 0 {
|
|
|
|
n.err = fmt.Errorf("duplicated RuleIndex: %v", bitIndex)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
n.simulatedDomainSet[bitIndex] = routing.DomainSet{
|
|
|
|
Key: typ,
|
|
|
|
RuleIndex: bitIndex,
|
|
|
|
Domains: patterns,
|
|
|
|
}
|
2023-02-18 17:27:28 +07:00
|
|
|
}
|
|
|
|
func (n *Bruteforce) MatchDomainBitmap(domain string) (bitmap []uint32) {
|
|
|
|
N := len(n.simulatedDomainSet) / 32
|
|
|
|
if len(n.simulatedDomainSet)%32 != 0 {
|
|
|
|
N++
|
|
|
|
}
|
2023-02-19 13:08:13 +07:00
|
|
|
domain = strings.ToLower(strings.TrimSuffix(domain, "."))
|
2023-02-18 17:27:28 +07:00
|
|
|
bitmap = make([]uint32, N)
|
|
|
|
for _, s := range n.simulatedDomainSet {
|
|
|
|
for _, d := range s.Domains {
|
|
|
|
var hit bool
|
|
|
|
switch s.Key {
|
|
|
|
case consts.RoutingDomainKey_Suffix:
|
|
|
|
if domain == d || strings.HasSuffix(domain, "."+strings.TrimPrefix(d, ".")) {
|
|
|
|
hit = true
|
|
|
|
}
|
|
|
|
case consts.RoutingDomainKey_Full:
|
|
|
|
if strings.EqualFold(domain, d) {
|
|
|
|
hit = true
|
|
|
|
}
|
|
|
|
case consts.RoutingDomainKey_Keyword:
|
|
|
|
if strings.Contains(strings.ToLower(domain), strings.ToLower(d)) {
|
|
|
|
hit = true
|
|
|
|
}
|
|
|
|
case consts.RoutingDomainKey_Regex:
|
|
|
|
if regexp.MustCompile(d).MatchString(strings.ToLower(domain)) {
|
|
|
|
hit = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if hit {
|
2023-02-19 13:08:13 +07:00
|
|
|
//logrus.Traceln(d, s.Key, "matched given", domain)
|
2023-02-18 17:27:28 +07:00
|
|
|
bitmap[s.RuleIndex/32] |= 1 << (s.RuleIndex % 32)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return bitmap
|
|
|
|
}
|
|
|
|
func (n *Bruteforce) Build() error {
|
|
|
|
if n.err != nil {
|
|
|
|
return n.err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|