mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2025-01-03 16:13:30 +08:00
205 lines
4.7 KiB
Go
205 lines
4.7 KiB
Go
package geodata
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/metacubex/mihomo/common/atomic"
|
|
mihomoHttp "github.com/metacubex/mihomo/component/http"
|
|
"github.com/metacubex/mihomo/component/mmdb"
|
|
C "github.com/metacubex/mihomo/constant"
|
|
"github.com/metacubex/mihomo/log"
|
|
)
|
|
|
|
var (
|
|
initGeoSite bool
|
|
initGeoIP int
|
|
initASN bool
|
|
|
|
initGeoSiteMutex sync.Mutex
|
|
initGeoIPMutex sync.Mutex
|
|
initASNMutex sync.Mutex
|
|
|
|
geoIpEnable atomic.Bool
|
|
geoSiteEnable atomic.Bool
|
|
asnEnable atomic.Bool
|
|
|
|
geoIpUrl string
|
|
mmdbUrl string
|
|
geoSiteUrl string
|
|
asnUrl string
|
|
)
|
|
|
|
func GeoIpUrl() string {
|
|
return geoIpUrl
|
|
}
|
|
|
|
func SetGeoIpUrl(url string) {
|
|
geoIpUrl = url
|
|
}
|
|
|
|
func MmdbUrl() string {
|
|
return mmdbUrl
|
|
}
|
|
|
|
func SetMmdbUrl(url string) {
|
|
mmdbUrl = url
|
|
}
|
|
|
|
func GeoSiteUrl() string {
|
|
return geoSiteUrl
|
|
}
|
|
|
|
func SetGeoSiteUrl(url string) {
|
|
geoSiteUrl = url
|
|
}
|
|
|
|
func ASNUrl() string {
|
|
return asnUrl
|
|
}
|
|
|
|
func SetASNUrl(url string) {
|
|
asnUrl = url
|
|
}
|
|
|
|
func downloadToPath(url string, path string) (err error) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second*90)
|
|
defer cancel()
|
|
resp, err := mihomoHttp.HttpRequest(ctx, url, http.MethodGet, nil, nil)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
f, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY, 0o644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
_, err = io.Copy(f, resp.Body)
|
|
|
|
return err
|
|
}
|
|
|
|
func InitGeoSite() error {
|
|
geoSiteEnable.Store(true)
|
|
initGeoSiteMutex.Lock()
|
|
defer initGeoSiteMutex.Unlock()
|
|
if _, err := os.Stat(C.Path.GeoSite()); os.IsNotExist(err) {
|
|
log.Infoln("Can't find GeoSite.dat, start download")
|
|
if err := downloadToPath(GeoSiteUrl(), C.Path.GeoSite()); err != nil {
|
|
return fmt.Errorf("can't download GeoSite.dat: %s", err.Error())
|
|
}
|
|
log.Infoln("Download GeoSite.dat finish")
|
|
initGeoSite = false
|
|
}
|
|
if !initGeoSite {
|
|
if err := Verify(C.GeositeName); err != nil {
|
|
log.Warnln("GeoSite.dat invalid, remove and download: %s", err)
|
|
if err := os.Remove(C.Path.GeoSite()); err != nil {
|
|
return fmt.Errorf("can't remove invalid GeoSite.dat: %s", err.Error())
|
|
}
|
|
if err := downloadToPath(GeoSiteUrl(), C.Path.GeoSite()); err != nil {
|
|
return fmt.Errorf("can't download GeoSite.dat: %s", err.Error())
|
|
}
|
|
}
|
|
initGeoSite = true
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func InitGeoIP() error {
|
|
geoIpEnable.Store(true)
|
|
initGeoIPMutex.Lock()
|
|
defer initGeoIPMutex.Unlock()
|
|
if GeodataMode() {
|
|
if _, err := os.Stat(C.Path.GeoIP()); os.IsNotExist(err) {
|
|
log.Infoln("Can't find GeoIP.dat, start download")
|
|
if err := downloadToPath(GeoIpUrl(), C.Path.GeoIP()); err != nil {
|
|
return fmt.Errorf("can't download GeoIP.dat: %s", err.Error())
|
|
}
|
|
log.Infoln("Download GeoIP.dat finish")
|
|
initGeoIP = 0
|
|
}
|
|
|
|
if initGeoIP != 1 {
|
|
if err := Verify(C.GeoipName); err != nil {
|
|
log.Warnln("GeoIP.dat invalid, remove and download: %s", err)
|
|
if err := os.Remove(C.Path.GeoIP()); err != nil {
|
|
return fmt.Errorf("can't remove invalid GeoIP.dat: %s", err.Error())
|
|
}
|
|
if err := downloadToPath(GeoIpUrl(), C.Path.GeoIP()); err != nil {
|
|
return fmt.Errorf("can't download GeoIP.dat: %s", err.Error())
|
|
}
|
|
}
|
|
initGeoIP = 1
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if _, err := os.Stat(C.Path.MMDB()); os.IsNotExist(err) {
|
|
log.Infoln("Can't find MMDB, start download")
|
|
if err := downloadToPath(MmdbUrl(), C.Path.MMDB()); err != nil {
|
|
return fmt.Errorf("can't download MMDB: %s", err.Error())
|
|
}
|
|
}
|
|
|
|
if initGeoIP != 2 {
|
|
if !mmdb.Verify(C.Path.MMDB()) {
|
|
log.Warnln("MMDB invalid, remove and download")
|
|
if err := os.Remove(C.Path.MMDB()); err != nil {
|
|
return fmt.Errorf("can't remove invalid MMDB: %s", err.Error())
|
|
}
|
|
if err := downloadToPath(MmdbUrl(), C.Path.MMDB()); err != nil {
|
|
return fmt.Errorf("can't download MMDB: %s", err.Error())
|
|
}
|
|
}
|
|
initGeoIP = 2
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func InitASN() error {
|
|
asnEnable.Store(true)
|
|
initASNMutex.Lock()
|
|
defer initASNMutex.Unlock()
|
|
if _, err := os.Stat(C.Path.ASN()); os.IsNotExist(err) {
|
|
log.Infoln("Can't find ASN.mmdb, start download")
|
|
if err := downloadToPath(ASNUrl(), C.Path.ASN()); err != nil {
|
|
return fmt.Errorf("can't download ASN.mmdb: %s", err.Error())
|
|
}
|
|
log.Infoln("Download ASN.mmdb finish")
|
|
initASN = false
|
|
}
|
|
if !initASN {
|
|
if !mmdb.Verify(C.Path.ASN()) {
|
|
log.Warnln("ASN invalid, remove and download")
|
|
if err := os.Remove(C.Path.ASN()); err != nil {
|
|
return fmt.Errorf("can't remove invalid ASN: %s", err.Error())
|
|
}
|
|
if err := downloadToPath(ASNUrl(), C.Path.ASN()); err != nil {
|
|
return fmt.Errorf("can't download ASN: %s", err.Error())
|
|
}
|
|
}
|
|
initASN = true
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func GeoIpEnable() bool {
|
|
return geoIpEnable.Load()
|
|
}
|
|
|
|
func GeoSiteEnable() bool {
|
|
return geoSiteEnable.Load()
|
|
}
|
|
|
|
func ASNEnable() bool {
|
|
return asnEnable.Load()
|
|
}
|