2024-02-20 09:59:56 +01:00
|
|
|
package geolocation
|
|
|
|
|
|
|
|
import (
|
2024-07-03 11:33:02 +02:00
|
|
|
"context"
|
2024-02-20 09:59:56 +01:00
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"path"
|
2024-09-09 18:27:42 +02:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2024-02-20 09:59:56 +01:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/oschwald/maxminddb-golang"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
2025-01-02 13:51:01 +01:00
|
|
|
type Geolocation interface {
|
|
|
|
Lookup(ip net.IP) (*Record, error)
|
|
|
|
GetAllCountries() ([]Country, error)
|
|
|
|
GetCitiesByCountry(countryISOCode string) ([]City, error)
|
|
|
|
Stop() error
|
|
|
|
}
|
|
|
|
|
|
|
|
type geolocationImpl struct {
|
2024-09-09 18:27:42 +02:00
|
|
|
mmdbPath string
|
|
|
|
mux sync.RWMutex
|
|
|
|
db *maxminddb.Reader
|
|
|
|
locationDB *SqliteStore
|
|
|
|
stopCh chan struct{}
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type Record struct {
|
|
|
|
City struct {
|
|
|
|
GeonameID uint `maxminddb:"geoname_id"`
|
|
|
|
Names struct {
|
|
|
|
En string `maxminddb:"en"`
|
|
|
|
} `maxminddb:"names"`
|
|
|
|
} `maxminddb:"city"`
|
|
|
|
Continent struct {
|
|
|
|
GeonameID uint `maxminddb:"geoname_id"`
|
|
|
|
Code string `maxminddb:"code"`
|
|
|
|
} `maxminddb:"continent"`
|
|
|
|
Country struct {
|
|
|
|
GeonameID uint `maxminddb:"geoname_id"`
|
|
|
|
ISOCode string `maxminddb:"iso_code"`
|
|
|
|
} `maxminddb:"country"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type City struct {
|
|
|
|
GeoNameID int `gorm:"column:geoname_id"`
|
|
|
|
CityName string
|
|
|
|
}
|
|
|
|
|
|
|
|
type Country struct {
|
|
|
|
CountryISOCode string `gorm:"column:country_iso_code"`
|
|
|
|
CountryName string
|
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
const (
|
|
|
|
mmdbPattern = "GeoLite2-City_*.mmdb"
|
|
|
|
geonamesdbPattern = "geonames_*.db"
|
|
|
|
)
|
|
|
|
|
2025-01-02 13:51:01 +01:00
|
|
|
func NewGeolocation(ctx context.Context, dataDir string, autoUpdate bool) (Geolocation, error) {
|
2024-09-09 18:27:42 +02:00
|
|
|
mmdbGlobPattern := filepath.Join(dataDir, mmdbPattern)
|
|
|
|
mmdbFile, err := getDatabaseFilename(ctx, geoLiteCityTarGZURL, mmdbGlobPattern, autoUpdate)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get database filename: %v", err)
|
2024-02-26 22:49:28 +01:00
|
|
|
}
|
2024-02-20 09:59:56 +01:00
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
geonamesDbGlobPattern := filepath.Join(dataDir, geonamesdbPattern)
|
|
|
|
geonamesDbFile, err := getDatabaseFilename(ctx, geoLiteCityZipURL, geonamesDbGlobPattern, autoUpdate)
|
2024-02-20 09:59:56 +01:00
|
|
|
if err != nil {
|
2024-09-09 18:27:42 +02:00
|
|
|
return nil, fmt.Errorf("failed to get database filename: %v", err)
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
if err := loadGeolocationDatabases(ctx, dataDir, mmdbFile, geonamesDbFile); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to load MaxMind databases: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := cleanupMaxMindDatabases(ctx, dataDir, mmdbFile, geonamesDbFile); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to remove old MaxMind databases: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
mmdbPath := path.Join(dataDir, mmdbFile)
|
|
|
|
db, err := openDB(mmdbPath)
|
2024-02-20 09:59:56 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
locationDB, err := NewSqliteStore(ctx, dataDir, geonamesDbFile)
|
2024-02-20 09:59:56 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-01-02 13:51:01 +01:00
|
|
|
geo := &geolocationImpl{
|
2024-09-09 18:27:42 +02:00
|
|
|
mmdbPath: mmdbPath,
|
|
|
|
mux: sync.RWMutex{},
|
|
|
|
db: db,
|
|
|
|
locationDB: locationDB,
|
|
|
|
stopCh: make(chan struct{}),
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return geo, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func openDB(mmdbPath string) (*maxminddb.Reader, error) {
|
|
|
|
_, err := os.Stat(mmdbPath)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return nil, fmt.Errorf("%v does not exist", mmdbPath)
|
|
|
|
} else if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
db, err := maxminddb.Open(mmdbPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%v could not be opened: %w", mmdbPath, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return db, nil
|
|
|
|
}
|
|
|
|
|
2025-01-02 13:51:01 +01:00
|
|
|
func (gl *geolocationImpl) Lookup(ip net.IP) (*Record, error) {
|
2024-02-20 09:59:56 +01:00
|
|
|
gl.mux.RLock()
|
|
|
|
defer gl.mux.RUnlock()
|
|
|
|
|
|
|
|
var record Record
|
|
|
|
err := gl.db.Lookup(ip, &record)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &record, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAllCountries retrieves a list of all countries.
|
2025-01-02 13:51:01 +01:00
|
|
|
func (gl *geolocationImpl) GetAllCountries() ([]Country, error) {
|
2024-02-20 09:59:56 +01:00
|
|
|
allCountries, err := gl.locationDB.GetAllCountries()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
countries := make([]Country, 0)
|
|
|
|
for _, country := range allCountries {
|
|
|
|
if country.CountryName != "" {
|
|
|
|
countries = append(countries, country)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return countries, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCitiesByCountry retrieves a list of cities in a specific country based on the country's ISO code.
|
2025-01-02 13:51:01 +01:00
|
|
|
func (gl *geolocationImpl) GetCitiesByCountry(countryISOCode string) ([]City, error) {
|
2024-02-20 09:59:56 +01:00
|
|
|
allCities, err := gl.locationDB.GetCitiesByCountry(countryISOCode)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
cities := make([]City, 0)
|
|
|
|
for _, city := range allCities {
|
|
|
|
if city.CityName != "" {
|
|
|
|
cities = append(cities, city)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return cities, nil
|
|
|
|
}
|
|
|
|
|
2025-01-02 13:51:01 +01:00
|
|
|
func (gl *geolocationImpl) Stop() error {
|
2024-02-20 09:59:56 +01:00
|
|
|
close(gl.stopCh)
|
|
|
|
if gl.db != nil {
|
|
|
|
if err := gl.db.Close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if gl.locationDB != nil {
|
|
|
|
if err := gl.locationDB.close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
func fileExists(filePath string) (bool, error) {
|
|
|
|
_, err := os.Stat(filePath)
|
|
|
|
if err == nil {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return false, fmt.Errorf("%v does not exist", filePath)
|
|
|
|
}
|
|
|
|
return false, err
|
|
|
|
}
|
2024-02-20 09:59:56 +01:00
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
func getExistingDatabases(pattern string) []string {
|
|
|
|
files, _ := filepath.Glob(pattern)
|
|
|
|
return files
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDatabaseFilename(ctx context.Context, databaseURL string, filenamePattern string, autoUpdate bool) (string, error) {
|
|
|
|
var (
|
|
|
|
filename string
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if autoUpdate {
|
|
|
|
filename, err = getFilenameFromURL(databaseURL)
|
|
|
|
if err != nil {
|
|
|
|
log.WithContext(ctx).Debugf("Failed to update database from url: %s", databaseURL)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
files := getExistingDatabases(filenamePattern)
|
|
|
|
if len(files) < 1 {
|
|
|
|
filename, err = getFilenameFromURL(databaseURL)
|
2024-02-20 09:59:56 +01:00
|
|
|
if err != nil {
|
2024-09-09 18:27:42 +02:00
|
|
|
log.WithContext(ctx).Debugf("Failed to get database from url: %s", databaseURL)
|
|
|
|
return "", err
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
2024-09-09 18:27:42 +02:00
|
|
|
} else {
|
|
|
|
filename = filepath.Base(files[len(files)-1])
|
|
|
|
log.WithContext(ctx).Debugf("Using existing database, %s", filename)
|
|
|
|
return filename, nil
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
// strip suffixes that may be nested, such as .tar.gz
|
|
|
|
basename := strings.SplitN(filename, ".", 2)[0]
|
|
|
|
// get date version from basename
|
|
|
|
date := strings.SplitN(basename, "_", 2)[1]
|
|
|
|
// format db as "GeoLite2-Cities-{maxmind|geonames}_{DATE}.{mmdb|db}"
|
|
|
|
databaseFilename := filepath.Base(strings.Replace(filenamePattern, "*", date, 1))
|
2024-02-20 09:59:56 +01:00
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
return databaseFilename, nil
|
|
|
|
}
|
2024-02-20 09:59:56 +01:00
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
func cleanupOldDatabases(ctx context.Context, pattern string, currentFile string) error {
|
|
|
|
files := getExistingDatabases(pattern)
|
2024-02-20 09:59:56 +01:00
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
for _, db := range files {
|
|
|
|
if filepath.Base(db) == currentFile {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
log.WithContext(ctx).Debugf("Removing old database: %s", db)
|
|
|
|
err := os.Remove(db)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-09 18:27:42 +02:00
|
|
|
func cleanupMaxMindDatabases(ctx context.Context, dataDir string, mmdbFile string, geonamesdbFile string) error {
|
|
|
|
for _, file := range []string{mmdbFile, geonamesdbFile} {
|
|
|
|
switch file {
|
|
|
|
case mmdbFile:
|
|
|
|
pattern := filepath.Join(dataDir, mmdbPattern)
|
|
|
|
if err := cleanupOldDatabases(ctx, pattern, file); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case geonamesdbFile:
|
|
|
|
pattern := filepath.Join(dataDir, geonamesdbPattern)
|
|
|
|
if err := cleanupOldDatabases(ctx, pattern, file); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
2024-09-09 18:27:42 +02:00
|
|
|
return nil
|
2024-02-20 09:59:56 +01:00
|
|
|
}
|
2025-01-02 13:51:01 +01:00
|
|
|
|
|
|
|
type Mock struct{}
|
|
|
|
|
|
|
|
func (g *Mock) Lookup(ip net.IP) (*Record, error) {
|
|
|
|
return &Record{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Mock) GetAllCountries() ([]Country, error) {
|
|
|
|
return []Country{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Mock) GetCitiesByCountry(countryISOCode string) ([]City, error) {
|
|
|
|
return []City{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *Mock) Stop() error {
|
|
|
|
return nil
|
|
|
|
}
|