package common import ( "errors" "fmt" "strings" "github.com/metacubex/mihomo/component/geodata" "github.com/metacubex/mihomo/component/geodata/router" "github.com/metacubex/mihomo/component/mmdb" "github.com/metacubex/mihomo/component/resolver" C "github.com/metacubex/mihomo/constant" "github.com/metacubex/mihomo/log" ) type GEOIP struct { *Base country string adapter string noResolveIP bool isSourceIP bool geodata bool } var _ C.Rule = (*GEOIP)(nil) func (g *GEOIP) RuleType() C.RuleType { if g.isSourceIP { return C.SrcGEOIP } return C.GEOIP } func (g *GEOIP) Match(metadata *C.Metadata) (bool, string) { ip := metadata.DstIP if g.isSourceIP { ip = metadata.SrcIP } if !ip.IsValid() { return false, "" } if g.country == "lan" { return ip.IsPrivate() || ip.IsUnspecified() || ip.IsLoopback() || ip.IsMulticast() || ip.IsLinkLocalUnicast() || resolver.IsFakeBroadcastIP(ip), g.adapter } for _, code := range metadata.DstGeoIP { if g.country == code { return true, g.adapter } } if !C.GeodataMode { if g.isSourceIP { codes := mmdb.IPInstance().LookupCode(ip.AsSlice()) for _, code := range codes { if g.country == code { return true, g.adapter } } return false, g.adapter } if metadata.DstGeoIP != nil { return false, g.adapter } metadata.DstGeoIP = mmdb.IPInstance().LookupCode(ip.AsSlice()) for _, code := range metadata.DstGeoIP { if g.country == code { return true, g.adapter } } return false, g.adapter } matcher, err := g.GetIPMatcher() if err != nil { return false, "" } match := matcher.Match(ip) if match && !g.isSourceIP { metadata.DstGeoIP = append(metadata.DstGeoIP, g.country) } return match, g.adapter } 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) GetCountry() string { return g.country } func (g *GEOIP) GetIPMatcher() (router.IPMatcher, error) { if g.geodata { geoIPMatcher, err := geodata.LoadGeoIPMatcher(g.country) if err != nil { return nil, fmt.Errorf("[GeoIP] %w", err) } return geoIPMatcher, nil } return nil, errors.New("geoip country not set") } func (g *GEOIP) GetRecodeSize() int { if matcher, err := g.GetIPMatcher(); err == nil { return matcher.Count() } return 0 } func NewGEOIP(country string, adapter string, isSrc, noResolveIP bool) (*GEOIP, error) { if err := geodata.InitGeoIP(); err != nil { log.Errorln("can't initial GeoIP: %s", err) return nil, err } country = strings.ToLower(country) geoip := &GEOIP{ Base: &Base{}, country: country, adapter: adapter, noResolveIP: noResolveIP, isSourceIP: isSrc, } if !C.GeodataMode || country == "lan" { return geoip, nil } geoip.geodata = true geoIPMatcher, err := geoip.GetIPMatcher() // test load if err != nil { return nil, err } log.Infoln("Finished initial GeoIP rule %s => %s, records: %d", country, adapter, geoIPMatcher.Count()) return geoip, nil }