213 lines
4.9 KiB
Go
213 lines
4.9 KiB
Go
|
package utils
|
||
|
|
||
|
import (
|
||
|
"archive/tar"
|
||
|
"compress/gzip"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"io/ioutil"
|
||
|
"math/rand"
|
||
|
"net"
|
||
|
"net/http"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"time"
|
||
|
|
||
|
"github.com/oschwald/geoip2-golang"
|
||
|
"github.com/pkg/errors"
|
||
|
)
|
||
|
|
||
|
// LocationInfo contains location information
|
||
|
type LocationInfo struct {
|
||
|
IP string
|
||
|
ASN uint
|
||
|
NetworkName string
|
||
|
CountryCode string
|
||
|
}
|
||
|
|
||
|
// XXX consider integration with: https://updates.maxmind.com/app/update_getfilename?product_id=GeoLite2-ASN
|
||
|
var geoipFiles = map[string]string{
|
||
|
"GeoLite2-ASN.mmdb": "http://geolite.maxmind.com/download/geoip/database/GeoLite2-ASN.tar.gz",
|
||
|
"GeoLite2-Country.mmdb": "http://geolite.maxmind.com/download/geoip/database/GeoLite2-Country.tar.gz",
|
||
|
}
|
||
|
|
||
|
// DownloadGeoIPDatabaseFiles into the target directory
|
||
|
func DownloadGeoIPDatabaseFiles(dir string) error {
|
||
|
for filename, url := range geoipFiles {
|
||
|
dstPath := filepath.Join(dir, filename)
|
||
|
|
||
|
// Download the file to a temporary location
|
||
|
out, err := ioutil.TempFile(os.TempDir(), "maxmind")
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "failed to create temporary directory")
|
||
|
}
|
||
|
resp, err := http.Get(url)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "failed to fetch URL")
|
||
|
}
|
||
|
|
||
|
_, err = io.Copy(out, resp.Body)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "failed to copy response body")
|
||
|
}
|
||
|
out.Close()
|
||
|
resp.Body.Close()
|
||
|
|
||
|
// Extract the tar.gz file
|
||
|
|
||
|
f, err := os.Open(out.Name())
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "failed to read file")
|
||
|
}
|
||
|
|
||
|
gzf, err := gzip.NewReader(f)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "failed to create gzip reader")
|
||
|
}
|
||
|
tarReader := tar.NewReader(gzf)
|
||
|
|
||
|
// Look inside of the tar for the file we need
|
||
|
for {
|
||
|
header, err := tarReader.Next()
|
||
|
if err == io.EOF {
|
||
|
break
|
||
|
}
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "error extracting tar.gz")
|
||
|
}
|
||
|
name := header.Name
|
||
|
if filepath.Base(name) == filename {
|
||
|
outFile, err := os.Create(dstPath)
|
||
|
if err != nil {
|
||
|
return errors.Wrap(err, "error creating file")
|
||
|
}
|
||
|
if _, err := io.Copy(outFile, tarReader); err != nil {
|
||
|
return errors.Wrap(err, "error reading file from tar")
|
||
|
}
|
||
|
outFile.Close()
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
f.Close()
|
||
|
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// LookupLocation resolves an IP to a location according to the Maxmind DB
|
||
|
func LookupLocation(dbPath string, ipStr string) (LocationInfo, error) {
|
||
|
loc := LocationInfo{}
|
||
|
fmt.Printf("Opening %s", filepath.Join(dbPath, "GeoLite2-ASN.mmdb"))
|
||
|
|
||
|
asnDB, err := geoip2.Open(filepath.Join(dbPath, "GeoLite2-ASN.mmdb"))
|
||
|
if err != nil {
|
||
|
return loc, errors.Wrap(err, "failed to open ASN db")
|
||
|
}
|
||
|
defer asnDB.Close()
|
||
|
|
||
|
countryDB, err := geoip2.Open(filepath.Join(dbPath, "GeoLite2-Country.mmdb"))
|
||
|
if err != nil {
|
||
|
return loc, errors.Wrap(err, "failed to open country db")
|
||
|
}
|
||
|
defer countryDB.Close()
|
||
|
|
||
|
ip := net.ParseIP(ipStr)
|
||
|
|
||
|
asn, err := asnDB.ASN(ip)
|
||
|
if err != nil {
|
||
|
return loc, err
|
||
|
}
|
||
|
country, err := countryDB.Country(ip)
|
||
|
if err != nil {
|
||
|
return loc, err
|
||
|
}
|
||
|
loc.ASN = asn.AutonomousSystemNumber
|
||
|
loc.NetworkName = asn.AutonomousSystemOrganization
|
||
|
loc.CountryCode = country.Country.IsoCode
|
||
|
|
||
|
return loc, nil
|
||
|
}
|
||
|
|
||
|
type avastResponse struct {
|
||
|
IP string `json:"ip"`
|
||
|
}
|
||
|
|
||
|
func avastLookup() (string, error) {
|
||
|
var parsed = new(avastResponse)
|
||
|
|
||
|
resp, err := http.Get("https://ip-info.ff.avast.com/v1/info")
|
||
|
if err != nil {
|
||
|
return "", errors.Wrap(err, "failed to perform request")
|
||
|
}
|
||
|
|
||
|
body, err := ioutil.ReadAll(resp.Body)
|
||
|
if err != nil {
|
||
|
return "", errors.Wrap(err, "failed to read response body")
|
||
|
}
|
||
|
err = json.Unmarshal([]byte(body), &parsed)
|
||
|
if err != nil {
|
||
|
return "", errors.Wrap(err, "failed to parse json")
|
||
|
}
|
||
|
|
||
|
return parsed.IP, nil
|
||
|
}
|
||
|
|
||
|
func akamaiLookup() (string, error) {
|
||
|
// This is a domain fronted request to akamai
|
||
|
client := &http.Client{}
|
||
|
req, err := http.NewRequest("GET", "https://a248.e.akamai.net/", nil)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
req.Host = "whatismyip.akamai.com"
|
||
|
resp, err := client.Do(req)
|
||
|
if err != nil {
|
||
|
return "", errors.Wrap(err, "failed to perform request")
|
||
|
}
|
||
|
|
||
|
body, err := ioutil.ReadAll(resp.Body)
|
||
|
if err != nil {
|
||
|
return "", errors.Wrap(err, "failed to read response body")
|
||
|
}
|
||
|
return string(body), nil
|
||
|
}
|
||
|
|
||
|
type lookupFunc func() (string, error)
|
||
|
|
||
|
var lookupServices = []lookupFunc{
|
||
|
avastLookup,
|
||
|
akamaiLookup,
|
||
|
}
|
||
|
|
||
|
// IPLookup gets the users IP address from a IP lookup service
|
||
|
func IPLookup() (string, error) {
|
||
|
rand.Seed(time.Now().Unix())
|
||
|
|
||
|
retries := 3
|
||
|
for retries > 0 {
|
||
|
lookup := lookupServices[rand.Intn(len(lookupServices))]
|
||
|
ipStr, err := lookup()
|
||
|
if err == nil {
|
||
|
return ipStr, nil
|
||
|
}
|
||
|
retries--
|
||
|
}
|
||
|
return "", errors.New("exceeded maximum retries")
|
||
|
}
|
||
|
|
||
|
// GeoIPLookup does a geoip lookup and returns location information
|
||
|
func GeoIPLookup(dbPath string) (*LocationInfo, error) {
|
||
|
ipStr, err := IPLookup()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
location, err := LookupLocation(dbPath, ipStr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &location, nil
|
||
|
}
|