Clash.Meta/dns/filters.go

94 lines
2.0 KiB
Go
Raw Normal View History

2019-09-15 13:36:45 +08:00
package dns
2020-01-11 21:07:01 +08:00
import (
"net"
"github.com/Dreamacro/clash/component/trie"
2021-07-01 22:49:29 +08:00
"github.com/Dreamacro/clash/log"
"github.com/Dreamacro/clash/rule/geodata"
"github.com/Dreamacro/clash/rule/geodata/router"
_ "github.com/Dreamacro/clash/rule/geodata/standard"
2020-01-11 21:07:01 +08:00
)
2019-09-15 13:36:45 +08:00
var multiGeoIPMatcher *router.MultiGeoIPMatcher
2021-07-01 22:49:29 +08:00
type fallbackIPFilter interface {
2019-09-15 13:36:45 +08:00
Match(net.IP) bool
}
type geoipFilter struct{}
func (gf *geoipFilter) Match(ip net.IP) bool {
if multiGeoIPMatcher == nil {
countryCodeCN := "cn"
countryCodePrivate := "private"
2021-07-01 22:49:29 +08:00
geoLoader, err := geodata.GetGeoDataLoader("standard")
if err != nil {
log.Errorln("[GeoIPFilter] GetGeoDataLoader error: %s", err.Error())
return false
}
recordsCN, err := geoLoader.LoadGeoIP(countryCodeCN)
2021-07-01 22:49:29 +08:00
if err != nil {
log.Errorln("[GeoIPFilter] LoadGeoIP error: %s", err.Error())
return false
}
recordsPrivate, err := geoLoader.LoadGeoIP(countryCodePrivate)
if err != nil {
log.Errorln("[GeoIPFilter] LoadGeoIP error: %s", err.Error())
return false
2021-07-01 22:49:29 +08:00
}
geoips := []*router.GeoIP{
{
CountryCode: countryCodeCN,
Cidr: recordsCN,
ReverseMatch: false,
},
{
CountryCode: countryCodePrivate,
Cidr: recordsPrivate,
ReverseMatch: false,
},
}
multiGeoIPMatcher, err = router.NewMultiGeoIPMatcher(geoips)
2021-07-01 22:49:29 +08:00
if err != nil {
log.Errorln("[GeoIPFilter] NewMultiGeoIPMatcher error: %s", err.Error())
2021-07-01 22:49:29 +08:00
return false
}
}
return !multiGeoIPMatcher.ApplyIp(ip)
2019-09-15 13:36:45 +08:00
}
type ipnetFilter struct {
ipnet *net.IPNet
}
func (inf *ipnetFilter) Match(ip net.IP) bool {
return inf.ipnet.Contains(ip)
}
type fallbackDomainFilter interface {
Match(domain string) bool
}
type domainFilter struct {
tree *trie.DomainTrie
}
func NewDomainFilter(domains []string) *domainFilter {
df := domainFilter{tree: trie.New()}
for _, domain := range domains {
df.tree.Insert(domain, "")
}
return &df
}
func (df *domainFilter) Match(domain string) bool {
return df.tree.Search(domain) != nil
}