mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-05 21:07:41 +03:00
domain: Add a new label type for domain suffix
This commit is contained in:
parent
2bf9cc7253
commit
332e470075
3 changed files with 48 additions and 11 deletions
|
@ -12,7 +12,7 @@ type Matcher struct {
|
||||||
set *succinctSet
|
set *succinctSet
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewMatcher(domains []string, domainSuffix []string) *Matcher {
|
func NewMatcher(domains []string, domainSuffix []string, generateLegacy bool) *Matcher {
|
||||||
domainList := make([]string, 0, len(domains)+2*len(domainSuffix))
|
domainList := make([]string, 0, len(domains)+2*len(domainSuffix))
|
||||||
seen := make(map[string]bool, len(domainList))
|
seen := make(map[string]bool, len(domainList))
|
||||||
for _, domain := range domainSuffix {
|
for _, domain := range domainSuffix {
|
||||||
|
@ -22,9 +22,15 @@ func NewMatcher(domains []string, domainSuffix []string) *Matcher {
|
||||||
seen[domain] = true
|
seen[domain] = true
|
||||||
if domain[0] == '.' {
|
if domain[0] == '.' {
|
||||||
domainList = append(domainList, reverseDomainSuffix(domain))
|
domainList = append(domainList, reverseDomainSuffix(domain))
|
||||||
} else {
|
} else if generateLegacy {
|
||||||
domainList = append(domainList, reverseDomain(domain))
|
domainList = append(domainList, reverseDomain(domain))
|
||||||
domainList = append(domainList, reverseRootDomainSuffix(domain))
|
suffixDomain := "." + domain
|
||||||
|
if !seen[suffixDomain] {
|
||||||
|
seen[suffixDomain] = true
|
||||||
|
domainList = append(domainList, reverseDomainSuffix(suffixDomain))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
domainList = append(domainList, reverseDomainRoot(domain))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, domain := range domains {
|
for _, domain := range domains {
|
||||||
|
@ -79,6 +85,8 @@ func (m *Matcher) Dump() (domainList []string, prefixList []string) {
|
||||||
key = reverseDomain(key)
|
key = reverseDomain(key)
|
||||||
if key[0] == prefixLabel {
|
if key[0] == prefixLabel {
|
||||||
prefixMap[key[1:]] = true
|
prefixMap[key[1:]] = true
|
||||||
|
} else if key[0] == rootLabel {
|
||||||
|
prefixList = append(prefixList, key[1:])
|
||||||
} else {
|
} else {
|
||||||
domainMap[key] = true
|
domainMap[key] = true
|
||||||
}
|
}
|
||||||
|
@ -124,15 +132,14 @@ func reverseDomainSuffix(domain string) string {
|
||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
func reverseRootDomainSuffix(domain string) string {
|
func reverseDomainRoot(domain string) string {
|
||||||
l := len(domain)
|
l := len(domain)
|
||||||
b := make([]byte, l+2)
|
b := make([]byte, l+1)
|
||||||
for i := 0; i < l; {
|
for i := 0; i < l; {
|
||||||
r, n := utf8.DecodeRuneInString(domain[i:])
|
r, n := utf8.DecodeRuneInString(domain[i:])
|
||||||
i += n
|
i += n
|
||||||
utf8.EncodeRune(b[l-i:], r)
|
utf8.EncodeRune(b[l-i:], r)
|
||||||
}
|
}
|
||||||
b[l] = '.'
|
b[l] = rootLabel
|
||||||
b[l+1] = prefixLabel
|
|
||||||
return string(b)
|
return string(b)
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,7 +14,26 @@ import (
|
||||||
func TestMatcher(t *testing.T) {
|
func TestMatcher(t *testing.T) {
|
||||||
testDomain := []string{"example.com", "example.org"}
|
testDomain := []string{"example.com", "example.org"}
|
||||||
testDomainSuffix := []string{".com.cn", ".org.cn", "sagernet.org"}
|
testDomainSuffix := []string{".com.cn", ".org.cn", "sagernet.org"}
|
||||||
matcher := domain.NewMatcher(testDomain, testDomainSuffix)
|
matcher := domain.NewMatcher(testDomain, testDomainSuffix, false)
|
||||||
|
require.NotNil(t, matcher)
|
||||||
|
require.True(t, matcher.Match("example.com"))
|
||||||
|
require.True(t, matcher.Match("example.org"))
|
||||||
|
require.False(t, matcher.Match("example.cn"))
|
||||||
|
require.True(t, matcher.Match("example.com.cn"))
|
||||||
|
require.True(t, matcher.Match("example.org.cn"))
|
||||||
|
require.False(t, matcher.Match("com.cn"))
|
||||||
|
require.False(t, matcher.Match("org.cn"))
|
||||||
|
require.True(t, matcher.Match("sagernet.org"))
|
||||||
|
require.True(t, matcher.Match("sing-box.sagernet.org"))
|
||||||
|
dDomain, dDomainSuffix := matcher.Dump()
|
||||||
|
require.Equal(t, testDomain, dDomain)
|
||||||
|
require.Equal(t, testDomainSuffix, dDomainSuffix)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMatcherLegacy(t *testing.T) {
|
||||||
|
testDomain := []string{"example.com", "example.org"}
|
||||||
|
testDomainSuffix := []string{".com.cn", ".org.cn", "sagernet.org"}
|
||||||
|
matcher := domain.NewMatcher(testDomain, testDomainSuffix, true)
|
||||||
require.NotNil(t, matcher)
|
require.NotNil(t, matcher)
|
||||||
require.True(t, matcher.Match("example.com"))
|
require.True(t, matcher.Match("example.com"))
|
||||||
require.True(t, matcher.Match("example.org"))
|
require.True(t, matcher.Match("example.org"))
|
||||||
|
@ -50,7 +69,7 @@ func TestDumpLarge(t *testing.T) {
|
||||||
require.True(t, len(domainList)+len(domainSuffixList) > 0)
|
require.True(t, len(domainList)+len(domainSuffixList) > 0)
|
||||||
sort.Strings(domainList)
|
sort.Strings(domainList)
|
||||||
sort.Strings(domainSuffixList)
|
sort.Strings(domainSuffixList)
|
||||||
matcher := domain.NewMatcher(domainList, domainSuffixList)
|
matcher := domain.NewMatcher(domainList, domainSuffixList, false)
|
||||||
require.NotNil(t, matcher)
|
require.NotNil(t, matcher)
|
||||||
dDomain, dDomainSuffix := matcher.Dump()
|
dDomain, dDomainSuffix := matcher.Dump()
|
||||||
require.Equal(t, domainList, dDomain)
|
require.Equal(t, domainList, dDomain)
|
||||||
|
|
|
@ -4,7 +4,10 @@ import (
|
||||||
"math/bits"
|
"math/bits"
|
||||||
)
|
)
|
||||||
|
|
||||||
const prefixLabel = '\r'
|
const (
|
||||||
|
prefixLabel = '\r'
|
||||||
|
rootLabel = '\n'
|
||||||
|
)
|
||||||
|
|
||||||
// mod from https://github.com/openacid/succinct
|
// mod from https://github.com/openacid/succinct
|
||||||
|
|
||||||
|
@ -54,6 +57,13 @@ func (ss *succinctSet) Has(key string) bool {
|
||||||
if nextLabel == prefixLabel {
|
if nextLabel == prefixLabel {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
if nextLabel == rootLabel {
|
||||||
|
nextNodeId := countZeros(ss.labelBitmap, ss.ranks, bmIdx+1)
|
||||||
|
hasNext := getBit(ss.leaves, nextNodeId) != 0
|
||||||
|
if currentChar == '.' && hasNext {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
if nextLabel == currentChar {
|
if nextLabel == currentChar {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
@ -68,7 +78,8 @@ func (ss *succinctSet) Has(key string) bool {
|
||||||
if getBit(ss.labelBitmap, bmIdx) != 0 {
|
if getBit(ss.labelBitmap, bmIdx) != 0 {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if ss.labels[bmIdx-nodeId] == prefixLabel {
|
nextLabel := ss.labels[bmIdx-nodeId]
|
||||||
|
if nextLabel == prefixLabel || nextLabel == rootLabel {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue