2018-06-10 22:50:03 +08:00
|
|
|
package rules
|
|
|
|
|
|
|
|
import (
|
2018-08-01 00:18:29 +08:00
|
|
|
"sync"
|
|
|
|
|
2018-06-10 22:50:03 +08:00
|
|
|
C "github.com/Dreamacro/clash/constant"
|
|
|
|
|
|
|
|
"github.com/oschwald/geoip2-golang"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
2018-08-01 00:18:29 +08:00
|
|
|
var (
|
|
|
|
mmdb *geoip2.Reader
|
|
|
|
once sync.Once
|
|
|
|
)
|
2018-06-10 22:50:03 +08:00
|
|
|
|
|
|
|
type GEOIP struct {
|
2019-10-28 00:02:23 +08:00
|
|
|
country string
|
|
|
|
adapter string
|
|
|
|
noResolveIP bool
|
2018-06-10 22:50:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (g *GEOIP) RuleType() C.RuleType {
|
|
|
|
return C.GEOIP
|
|
|
|
}
|
|
|
|
|
2019-10-28 00:02:23 +08:00
|
|
|
func (g *GEOIP) Match(metadata *C.Metadata) bool {
|
|
|
|
ip := metadata.DstIP
|
|
|
|
if ip == nil {
|
2018-06-10 22:50:03 +08:00
|
|
|
return false
|
|
|
|
}
|
2019-10-28 00:02:23 +08:00
|
|
|
record, _ := mmdb.Country(ip)
|
2018-06-10 22:50:03 +08:00
|
|
|
return record.Country.IsoCode == g.country
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *GEOIP) Adapter() string {
|
|
|
|
return g.adapter
|
|
|
|
}
|
|
|
|
|
2018-06-20 22:41:02 +08:00
|
|
|
func (g *GEOIP) Payload() string {
|
|
|
|
return g.country
|
|
|
|
}
|
|
|
|
|
2019-10-28 00:02:23 +08:00
|
|
|
func (g *GEOIP) NoResolveIP() bool {
|
|
|
|
return g.noResolveIP
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewGEOIP(country string, adapter string, noResolveIP bool) *GEOIP {
|
2018-08-01 00:18:29 +08:00
|
|
|
once.Do(func() {
|
|
|
|
var err error
|
2018-10-14 21:22:58 +08:00
|
|
|
mmdb, err = geoip2.Open(C.Path.MMDB())
|
2018-08-01 00:18:29 +08:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Can't load mmdb: %s", err.Error())
|
|
|
|
}
|
|
|
|
})
|
2019-10-28 00:02:23 +08:00
|
|
|
|
|
|
|
geoip := &GEOIP{
|
|
|
|
country: country,
|
|
|
|
adapter: adapter,
|
|
|
|
noResolveIP: noResolveIP,
|
2018-06-10 22:50:03 +08:00
|
|
|
}
|
2019-10-28 00:02:23 +08:00
|
|
|
|
|
|
|
return geoip
|
2018-06-10 22:50:03 +08:00
|
|
|
}
|