wttr.in/internal/geo/ip/ip.go

245 lines
5.4 KiB
Go
Raw Normal View History

2022-12-04 15:48:04 +00:00
package ip
import (
2022-12-06 17:52:26 +00:00
"fmt"
"log"
"net/http"
2022-12-04 15:48:04 +00:00
"os"
"path"
"regexp"
2022-12-04 15:48:04 +00:00
"strconv"
"strings"
"github.com/chubin/wttr.in/internal/config"
"github.com/chubin/wttr.in/internal/routing"
2022-12-06 17:52:26 +00:00
"github.com/chubin/wttr.in/internal/types"
"github.com/chubin/wttr.in/internal/util"
2022-12-06 17:52:26 +00:00
"github.com/samonzeweb/godb"
"github.com/samonzeweb/godb/adapters/sqlite"
2022-12-04 15:48:04 +00:00
)
2022-12-18 15:26:30 +00:00
// Address information.
type Address struct {
2022-12-04 20:16:23 +00:00
IP string `db:"ip,key"`
CountryCode string `db:"countryCode"`
Country string `db:"country"`
Region string `db:"region"`
City string `db:"city"`
Latitude float64 `db:"latitude"`
Longitude float64 `db:"longitude"`
2022-12-04 15:48:04 +00:00
}
2022-12-18 15:26:30 +00:00
func (l *Address) String() string {
2022-12-06 17:52:26 +00:00
if l.Latitude == -1000 {
return fmt.Sprintf(
"%s;%s;%s;%s",
2022-12-22 17:07:01 +00:00
l.CountryCode, l.Country, l.Region, l.City)
2022-12-06 17:52:26 +00:00
}
2022-12-11 13:28:34 +00:00
2022-12-06 17:52:26 +00:00
return fmt.Sprintf(
"%s;%s;%s;%s;%v;%v",
2022-12-22 17:07:01 +00:00
l.CountryCode, l.Country, l.Region, l.City, l.Latitude, l.Longitude)
2022-12-06 17:52:26 +00:00
}
2022-12-04 15:48:04 +00:00
// Cache provides access to the IP Geodata cache.
type Cache struct {
config *config.Config
2022-12-06 17:52:26 +00:00
db *godb.DB
2022-12-04 15:48:04 +00:00
}
// NewCache returns new cache reader for the specified config.
2022-12-06 17:52:26 +00:00
func NewCache(config *config.Config) (*Cache, error) {
db, err := godb.Open(sqlite.Adapter, config.Geo.IPCacheDB)
if err != nil {
return nil, err
}
2022-12-06 18:37:57 +00:00
// Needed for "upsert" implementation in Put()
db.UseErrorParser()
2022-12-04 15:48:04 +00:00
return &Cache{
config: config,
2022-12-06 17:52:26 +00:00
db: db,
}, nil
2022-12-04 15:48:04 +00:00
}
// Read returns location information from the cache, if found,
2022-12-11 08:36:11 +00:00
// or types.ErrNotFound if not found. If the entry is found, but its format
// is invalid, types.ErrInvalidCacheEntry is returned.
2022-12-04 15:48:04 +00:00
//
// Format:
//
2022-12-11 13:28:34 +00:00
// [CountryCode];Country;Region;City;[Latitude];[Longitude]
2022-12-04 15:48:04 +00:00
//
// Example:
//
2022-12-11 13:28:34 +00:00
// DE;Germany;Free and Hanseatic City of Hamburg;Hamburg;53.5736;9.9782
2022-12-04 15:48:04 +00:00
//
2022-12-11 13:28:34 +00:00
2022-12-18 15:26:30 +00:00
func (c *Cache) Read(addr string) (*Address, error) {
2022-12-11 08:36:11 +00:00
if c.config.Geo.IPCacheType == types.CacheTypeDB {
2022-12-06 17:52:26 +00:00
return c.readFromCacheDB(addr)
}
2022-12-11 13:28:34 +00:00
2022-12-06 17:52:26 +00:00
return c.readFromCacheFile(addr)
}
2022-12-18 15:26:30 +00:00
func (c *Cache) readFromCacheFile(addr string) (*Address, error) {
2022-12-04 15:48:04 +00:00
bytes, err := os.ReadFile(c.cacheFile(addr))
if err != nil {
2022-12-11 08:36:11 +00:00
return nil, types.ErrNotFound
2022-12-04 15:48:04 +00:00
}
2022-12-11 13:28:34 +00:00
2022-12-18 15:26:30 +00:00
return NewAddressFromString(addr, string(bytes))
2022-12-04 15:48:04 +00:00
}
2022-12-18 15:26:30 +00:00
func (c *Cache) readFromCacheDB(addr string) (*Address, error) {
result := Address{}
2022-12-06 17:52:26 +00:00
err := c.db.Select(&result).
Where("IP = ?", addr).
Do()
if err != nil {
return nil, err
}
2022-12-11 13:28:34 +00:00
2022-12-06 17:52:26 +00:00
return &result, nil
}
2022-12-18 15:26:30 +00:00
func (c *Cache) Put(addr string, loc *Address) error {
2022-12-11 08:36:11 +00:00
if c.config.Geo.IPCacheType == types.CacheTypeDB {
2022-12-11 13:28:34 +00:00
return c.putToCacheDB(loc)
2022-12-06 18:37:57 +00:00
}
2022-12-11 13:28:34 +00:00
2022-12-06 18:37:57 +00:00
return c.putToCacheFile(addr, loc)
}
2022-12-18 15:26:30 +00:00
func (c *Cache) putToCacheDB(loc *Address) error {
2022-12-06 18:37:57 +00:00
err := c.db.Insert(loc).Do()
// it should work like this:
//
// target := dberror.UniqueConstraint{}
// if errors.As(err, &target) {
//
// See: https://github.com/samonzeweb/godb/pull/23
//
// But for some reason it does not work,
// so the dirty hack is used:
if strings.Contains(fmt.Sprint(err), "UNIQUE constraint failed") {
return c.db.Update(loc).Do()
}
2022-12-11 13:28:34 +00:00
2022-12-06 18:37:57 +00:00
return err
}
2022-12-11 13:28:34 +00:00
func (c *Cache) putToCacheFile(addr string, loc fmt.Stringer) error {
return os.WriteFile(c.cacheFile(addr), []byte(loc.String()), 0o600)
2022-12-06 18:37:57 +00:00
}
2022-12-11 13:28:34 +00:00
// cacheFile returns path to the cache entry for addr.
2022-12-04 15:48:04 +00:00
func (c *Cache) cacheFile(addr string) string {
return path.Join(c.config.Geo.IPCache, addr)
}
2022-12-18 15:26:30 +00:00
// NewAddressFromString parses the location cache entry s,
2022-12-04 15:48:04 +00:00
// and return location, or error, if the cache entry is invalid.
2022-12-18 15:26:30 +00:00
func NewAddressFromString(addr, s string) (*Address, error) {
2022-12-04 15:48:04 +00:00
var (
lat float64 = -1000
long float64 = -1000
err error
)
parts := strings.Split(s, ";")
if len(parts) < 4 {
2022-12-11 08:36:11 +00:00
return nil, types.ErrInvalidCacheEntry
2022-12-04 15:48:04 +00:00
}
if len(parts) >= 6 {
lat, err = strconv.ParseFloat(parts[4], 64)
if err != nil {
2022-12-11 08:36:11 +00:00
return nil, types.ErrInvalidCacheEntry
2022-12-04 15:48:04 +00:00
}
long, err = strconv.ParseFloat(parts[5], 64)
if err != nil {
2022-12-11 08:36:11 +00:00
return nil, types.ErrInvalidCacheEntry
2022-12-04 15:48:04 +00:00
}
}
2022-12-18 15:26:30 +00:00
return &Address{
2022-12-04 20:16:23 +00:00
IP: addr,
2022-12-04 15:48:04 +00:00
CountryCode: parts[0],
Country: parts[1],
Region: parts[2],
City: parts[3],
Latitude: lat,
Longitude: long,
}, nil
}
2022-12-11 13:28:34 +00:00
// Response provides routing interface to the geo cache.
//
// Temporary workaround to switch IP addresses handling to the Go server.
// Handles two queries:
//
2022-12-11 13:28:34 +00:00
// - /:geo-ip-put?ip=IP&value=VALUE
// - /:geo-ip-get?ip=IP
//
2022-12-11 13:28:34 +00:00
//nolint:cyclop
func (c *Cache) Response(r *http.Request) *routing.Cadre {
var (
respERR = &routing.Cadre{Body: []byte("ERR")}
respOK = &routing.Cadre{Body: []byte("OK")}
)
if ip := util.ReadUserIP(r); ip != "127.0.0.1" {
log.Printf("geoIP access from %s rejected\n", ip)
2022-12-11 13:28:34 +00:00
return nil
}
if r.URL.Path == "/:geo-ip-put" {
ip := r.URL.Query().Get("ip")
value := r.URL.Query().Get("value")
if !validIP4(ip) || value == "" {
log.Printf("invalid geoIP put query: ip='%s' value='%s'\n", ip, value)
2022-12-11 13:28:34 +00:00
return respERR
}
2022-12-18 15:26:30 +00:00
location, err := NewAddressFromString(ip, value)
2022-12-06 18:37:57 +00:00
if err != nil {
return respERR
}
err = c.Put(ip, location)
if err != nil {
return respERR
}
2022-12-11 13:28:34 +00:00
return respOK
}
if r.URL.Path == "/:geo-ip-get" {
ip := r.URL.Query().Get("ip")
if !validIP4(ip) {
return respERR
}
2022-12-06 17:52:26 +00:00
result, err := c.Read(ip)
if result == nil || err != nil {
return respERR
}
2022-12-11 13:28:34 +00:00
2022-12-06 17:52:26 +00:00
return &routing.Cadre{Body: []byte(result.String())}
}
2022-12-11 13:28:34 +00:00
return nil
}
func validIP4(ipAddress string) bool {
2022-12-11 13:28:34 +00:00
re := regexp.MustCompile(
`^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$`)
return re.MatchString(strings.Trim(ipAddress, " "))
}