mirror of
https://github.com/therootcompany/golib.git
synced 2026-04-24 20:58:00 +00:00
httpcache: write <path>.meta JSON sidecar after each successful download; load it on first Fetch so conditional GETs work after process restarts. Tests verify: download, sidecar written, same-cacher 304, fresh-cacher 304 (the last being the key case — no in-memory state, sidecar drives ETag). MaxMind integration test reads GeoIP.conf, downloads City+ASN, verifies fresh-cacher conditional GET skips re-download via sidecar ETag.
112 lines
2.6 KiB
Go
112 lines
2.6 KiB
Go
//go:build integration
|
|
|
|
package geoip_test
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/therootcompany/golib/net/geoip"
|
|
)
|
|
|
|
func testdataDir(t *testing.T) string {
|
|
t.Helper()
|
|
dir, _ := filepath.Abs(".")
|
|
for {
|
|
if _, err := os.Stat(filepath.Join(dir, "go.mod")); err == nil {
|
|
return filepath.Join(dir, "testdata")
|
|
}
|
|
parent := filepath.Dir(dir)
|
|
if parent == dir {
|
|
t.Fatal("could not find module root (go.mod)")
|
|
}
|
|
dir = parent
|
|
}
|
|
}
|
|
|
|
func geoipConf(t *testing.T) *geoip.Conf {
|
|
t.Helper()
|
|
// Look for GeoIP.conf relative to the module root.
|
|
dir, _ := filepath.Abs(".")
|
|
for {
|
|
p := filepath.Join(dir, "GeoIP.conf")
|
|
if _, err := os.Stat(p); err == nil {
|
|
cfg, err := geoip.ParseConf(p)
|
|
if err != nil {
|
|
t.Fatalf("GeoIP.conf: %v", err)
|
|
}
|
|
return cfg
|
|
}
|
|
parent := filepath.Dir(dir)
|
|
if parent == dir {
|
|
break
|
|
}
|
|
dir = parent
|
|
}
|
|
t.Skip("GeoIP.conf not found; skipping MaxMind integration test")
|
|
return nil
|
|
}
|
|
|
|
func TestDownloader_CityAndASN(t *testing.T) {
|
|
cfg := geoipConf(t)
|
|
td := testdataDir(t)
|
|
|
|
d := geoip.New(cfg.AccountID, cfg.LicenseKey)
|
|
|
|
for _, edition := range []string{geoip.CityEdition, geoip.ASNEdition} {
|
|
path := filepath.Join(td, edition+".mmdb")
|
|
os.Remove(path)
|
|
os.Remove(path + ".meta")
|
|
|
|
cacher := d.NewCacher(edition, path)
|
|
updated, err := cacher.Fetch()
|
|
if err != nil {
|
|
t.Fatalf("%s Fetch: %v", edition, err)
|
|
}
|
|
if !updated {
|
|
t.Errorf("%s: expected updated=true on first fetch", edition)
|
|
}
|
|
|
|
info, err := os.Stat(path)
|
|
if err != nil {
|
|
t.Fatalf("%s: file not created: %v", edition, err)
|
|
}
|
|
if info.Size() == 0 {
|
|
t.Errorf("%s: downloaded file is empty", edition)
|
|
}
|
|
t.Logf("%s: %d bytes", edition, info.Size())
|
|
|
|
if _, err := os.Stat(path + ".meta"); err != nil {
|
|
t.Errorf("%s: sidecar not written: %v", edition, err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDownloader_ConditionalGet_FreshCacher(t *testing.T) {
|
|
cfg := geoipConf(t)
|
|
td := testdataDir(t)
|
|
|
|
d := geoip.New(cfg.AccountID, cfg.LicenseKey)
|
|
|
|
for _, edition := range []string{geoip.CityEdition, geoip.ASNEdition} {
|
|
path := filepath.Join(td, edition+".mmdb")
|
|
|
|
// Ensure downloaded.
|
|
if _, err := d.NewCacher(edition, path).Fetch(); err != nil {
|
|
t.Fatalf("%s initial Fetch: %v", edition, err)
|
|
}
|
|
|
|
// Fresh cacher — no in-memory ETag, must use sidecar.
|
|
fresh := d.NewCacher(edition, path)
|
|
updated, err := fresh.Fetch()
|
|
if err != nil {
|
|
t.Fatalf("%s fresh Fetch: %v", edition, err)
|
|
}
|
|
if updated {
|
|
t.Errorf("%s: fresh cacher expected updated=false (sidecar ETag should have been used)", edition)
|
|
}
|
|
t.Logf("%s: fresh-cacher conditional GET correctly skipped re-download", edition)
|
|
}
|
|
}
|