| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 | 
							- // Copyright (C) 2024 The Syncthing Authors.
 
- //
 
- // This Source Code Form is subject to the terms of the Mozilla Public
 
- // License, v. 2.0. If a copy of the MPL was not distributed with this file,
 
- // You can obtain one at https://mozilla.org/MPL/2.0/.
 
- // Package geoip provides an automatically updating MaxMind GeoIP2 database
 
- // provider.
 
- package geoip
 
- import (
 
- 	"context"
 
- 	"errors"
 
- 	"fmt"
 
- 	"net"
 
- 	"os"
 
- 	"path/filepath"
 
- 	"sync"
 
- 	"time"
 
- 	"github.com/maxmind/geoipupdate/v6/pkg/geoipupdate"
 
- 	"github.com/oschwald/geoip2-golang"
 
- )
 
- type Provider struct {
 
- 	edition         string
 
- 	accountID       int
 
- 	licenseKey      string
 
- 	refreshInterval time.Duration
 
- 	directory       string
 
- 	mut          sync.Mutex
 
- 	currentDBDir string
 
- 	db           *geoip2.Reader
 
- }
 
- // NewGeoLite2CityProvider returns a new GeoIP2 database provider for the
 
- // GeoLite2-City database. The database will be stored in the given
 
- // directory (which should exist) and refreshed every 7 days.
 
- func NewGeoLite2CityProvider(ctx context.Context, accountID int, licenseKey string, directory string) (*Provider, error) {
 
- 	p := &Provider{
 
- 		edition:         "GeoLite2-City",
 
- 		accountID:       accountID,
 
- 		licenseKey:      licenseKey,
 
- 		refreshInterval: 7 * 24 * time.Hour,
 
- 		directory:       directory,
 
- 	}
 
- 	if err := p.download(ctx); err != nil {
 
- 		return nil, err
 
- 	}
 
- 	return p, nil
 
- }
 
- func (p *Provider) City(ip net.IP) (*geoip2.City, error) {
 
- 	p.mut.Lock()
 
- 	defer p.mut.Unlock()
 
- 	if p.db == nil {
 
- 		return nil, errors.New("database not open")
 
- 	}
 
- 	return p.db.City(ip)
 
- }
 
- // Serve downloads the GeoIP2 database and keeps it up to date. It will return
 
- // when the context is canceled.
 
- func (p *Provider) Serve(ctx context.Context) error {
 
- 	for {
 
- 		select {
 
- 		case <-ctx.Done():
 
- 			return ctx.Err()
 
- 		case <-time.After(p.refreshInterval):
 
- 			if err := p.download(ctx); err != nil {
 
- 				return err
 
- 			}
 
- 		}
 
- 	}
 
- }
 
- func (p *Provider) download(ctx context.Context) error {
 
- 	newSubdir, err := os.MkdirTemp(p.directory, "geoipupdate")
 
- 	if err != nil {
 
- 		return fmt.Errorf("download: %w", err)
 
- 	}
 
- 	cfg := &geoipupdate.Config{
 
- 		URL:               "https://updates.maxmind.com",
 
- 		DatabaseDirectory: newSubdir,
 
- 		LockFile:          filepath.Join(newSubdir, "geoipupdate.lock"),
 
- 		RetryFor:          5 * time.Minute,
 
- 		Parallelism:       1,
 
- 		AccountID:         p.accountID,
 
- 		LicenseKey:        p.licenseKey,
 
- 		EditionIDs:        []string{p.edition},
 
- 	}
 
- 	if err := geoipupdate.NewClient(cfg).Run(ctx); err != nil {
 
- 		return fmt.Errorf("download: %w", err)
 
- 	}
 
- 	dbPath := filepath.Join(newSubdir, p.edition+".mmdb")
 
- 	db, err := geoip2.Open(dbPath)
 
- 	if err != nil {
 
- 		return fmt.Errorf("open downloaded db: %w", err)
 
- 	}
 
- 	p.mut.Lock()
 
- 	prevDBDir := p.currentDBDir
 
- 	if p.db != nil {
 
- 		p.db.Close()
 
- 	}
 
- 	p.currentDBDir = newSubdir
 
- 	p.db = db
 
- 	p.mut.Unlock()
 
- 	if prevDBDir != "" {
 
- 		_ = os.RemoveAll(p.currentDBDir)
 
- 	}
 
- 	return nil
 
- }
 
 
  |