package rules import ( "fmt" "strings" C "github.com/Dreamacro/clash/constant" "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" ) type GEOIP struct { country string adapter string noResolveIP bool ruleExtra *C.RuleExtra geoIPMatcher *router.GeoIPMatcher } func (g *GEOIP) RuleType() C.RuleType { return C.GEOIP } func (g *GEOIP) Match(metadata *C.Metadata) bool { ip := metadata.DstIP if ip == nil { return false } if strings.EqualFold(g.country, "LAN") { return ip.IsPrivate() } return g.geoIPMatcher.Match(ip) } func (g *GEOIP) Adapter() string { return g.adapter } func (g *GEOIP) Payload() string { return g.country } func (g *GEOIP) ShouldResolveIP() bool { return !g.noResolveIP } func (g *GEOIP) RuleExtra() *C.RuleExtra { return g.ruleExtra } func NewGEOIP(country string, adapter string, noResolveIP bool, ruleExtra *C.RuleExtra) (*GEOIP, error) { geoLoaderName := "standard" //geoLoaderName := "memconservative" geoLoader, err := geodata.GetGeoDataLoader(geoLoaderName) if err != nil { return nil, fmt.Errorf("[GeoIP] %s", err.Error()) } records, err := geoLoader.LoadGeoIP(strings.ReplaceAll(country, "!", "")) if err != nil { return nil, fmt.Errorf("[GeoIP] %s", err.Error()) } geoIP := &router.GeoIP{ CountryCode: country, Cidr: records, ReverseMatch: strings.Contains(country, "!"), } geoIPMatcher, err := router.NewGeoIPMatcher(geoIP) if err != nil { return nil, fmt.Errorf("[GeoIP] %s", err.Error()) } log.Infoln("Start initial GeoIP rule %s => %s, records: %d", country, adapter, len(records)) geoip := &GEOIP{ country: country, adapter: adapter, noResolveIP: noResolveIP, ruleExtra: ruleExtra, geoIPMatcher: geoIPMatcher, } return geoip, nil }