clash/component/domain-trie/tire.go

120 lines
2 KiB
Go
Raw Normal View History

2019-07-14 11:29:58 +00:00
package trie
import (
"errors"
"strings"
)
const (
wildcard = "*"
dotWildcard = ""
domainStep = "."
2019-07-14 11:29:58 +00:00
)
var (
// ErrInvalidDomain means insert domain is invalid
ErrInvalidDomain = errors.New("invalid domain")
)
// Trie contains the main logic for adding and searching nodes for domain segments.
// support wildcard domain (e.g *.google.com)
type Trie struct {
root *Node
}
func validAndSplitDomain(domain string) ([]string, bool) {
if domain != "" && domain[len(domain)-1] == '.' {
return nil, false
}
parts := strings.Split(domain, domainStep)
if len(parts) == 1 {
return nil, false
}
for _, part := range parts[1:] {
if part == "" {
return nil, false
}
}
return parts, true
}
2019-07-14 11:29:58 +00:00
// Insert adds a node to the trie.
// Support
// 1. www.example.com
// 2. *.example.com
// 3. subdomain.*.example.com
// 4. .example.com
2019-07-14 11:29:58 +00:00
func (t *Trie) Insert(domain string, data interface{}) error {
parts, valid := validAndSplitDomain(domain)
if !valid {
2019-07-14 11:29:58 +00:00
return ErrInvalidDomain
}
node := t.root
// reverse storage domain part to save space
for i := len(parts) - 1; i >= 0; i-- {
part := parts[i]
if !node.hasChild(part) {
node.addChild(part, newNode(nil))
}
node = node.getChild(part)
}
node.Data = data
return nil
}
// Search is the most important part of the Trie.
// Priority as:
// 1. static part
// 2. wildcard domain
// 2. dot wildcard domain
2019-07-14 11:29:58 +00:00
func (t *Trie) Search(domain string) *Node {
parts, valid := validAndSplitDomain(domain)
if !valid || parts[0] == "" {
2019-07-14 11:29:58 +00:00
return nil
}
n := t.root
var dotWildcardNode *Node
2019-07-14 11:29:58 +00:00
for i := len(parts) - 1; i >= 0; i-- {
part := parts[i]
if node := n.getChild(dotWildcard); node != nil {
dotWildcardNode = node
}
2019-07-14 11:29:58 +00:00
if n.hasChild(part) {
n = n.getChild(part)
2019-07-14 11:29:58 +00:00
} else {
n = n.getChild(wildcard)
}
if n == nil {
break
2019-07-14 11:29:58 +00:00
}
}
2019-07-14 11:29:58 +00:00
if n == nil {
if dotWildcardNode != nil {
return dotWildcardNode
}
return nil
2019-07-14 11:29:58 +00:00
}
2019-07-15 10:00:51 +00:00
if n.Data == nil {
return nil
}
2019-07-14 11:29:58 +00:00
return n
}
// New returns a new, empty Trie.
func New() *Trie {
return &Trie{root: newNode(nil)}
}