Feature: log out country, region and city of IP
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
package updater
|
package constants
|
||||||
|
|
||||||
func getCountryCodes() map[string]string { //nolint:dupl
|
func CountryCodes() map[string]string {
|
||||||
return map[string]string{
|
return map[string]string{
|
||||||
"af": "Afghanistan",
|
"af": "Afghanistan",
|
||||||
"ax": "Aland Islands",
|
"ax": "Aland Islands",
|
||||||
54
internal/publicip/info.go
Normal file
54
internal/publicip/info.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package publicip
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/qdm12/gluetun/internal/constants"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ipInfoData struct {
|
||||||
|
Region string `json:"region"`
|
||||||
|
Country string `json:"country"`
|
||||||
|
City string `json:"city"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var ErrBadHTTPStatus = errors.New("bad HTTP status received")
|
||||||
|
|
||||||
|
func Info(ctx context.Context, client *http.Client, ip net.IP) ( //nolint:interfacer
|
||||||
|
country, region, city string, err error) {
|
||||||
|
const baseURL = "https://ipinfo.io/"
|
||||||
|
url := baseURL + ip.String()
|
||||||
|
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
response, err := client.Do(request)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", "", err
|
||||||
|
}
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
if response.StatusCode != http.StatusOK {
|
||||||
|
return "", "", "", fmt.Errorf("%w: %d", ErrBadHTTPStatus, response.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
decoder := json.NewDecoder(response.Body)
|
||||||
|
var data ipInfoData
|
||||||
|
if err := decoder.Decode(&data); err != nil {
|
||||||
|
return "", "", "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
countryCode := strings.ToLower(data.Country)
|
||||||
|
country, ok := constants.CountryCodes()[countryCode]
|
||||||
|
if !ok {
|
||||||
|
country = data.Country
|
||||||
|
}
|
||||||
|
return country, data.Region, data.City, nil
|
||||||
|
}
|
||||||
@@ -28,6 +28,7 @@ type looper struct {
|
|||||||
state state
|
state state
|
||||||
// Objects
|
// Objects
|
||||||
getter IPGetter
|
getter IPGetter
|
||||||
|
client *http.Client
|
||||||
logger logging.Logger
|
logger logging.Logger
|
||||||
os os.OS
|
os os.OS
|
||||||
// Fixed settings
|
// Fixed settings
|
||||||
@@ -57,6 +58,7 @@ func NewLooper(client *http.Client, logger logging.Logger,
|
|||||||
settings: settings,
|
settings: settings,
|
||||||
},
|
},
|
||||||
// Objects
|
// Objects
|
||||||
|
client: client,
|
||||||
getter: NewIPGetter(client),
|
getter: NewIPGetter(client),
|
||||||
logger: logger.WithPrefix("ip getter: "),
|
logger: logger.WithPrefix("ip getter: "),
|
||||||
os: os,
|
os: os,
|
||||||
@@ -150,8 +152,17 @@ func (l *looper) Run(ctx context.Context, wg *sync.WaitGroup) {
|
|||||||
case ip := <-ipCh:
|
case ip := <-ipCh:
|
||||||
getCancel()
|
getCancel()
|
||||||
l.state.setPublicIP(ip)
|
l.state.setPublicIP(ip)
|
||||||
l.logger.Info("Public IP address is %s", ip)
|
|
||||||
err := persistPublicIP(l.os.OpenFile, l.state.settings.IPFilepath,
|
message := "Public IP address is " + ip.String()
|
||||||
|
country, region, city, err := Info(ctx, l.client, ip)
|
||||||
|
if err != nil {
|
||||||
|
l.logger.Warn(err)
|
||||||
|
} else {
|
||||||
|
message += " (" + country + ", " + region + ", " + city + ")"
|
||||||
|
}
|
||||||
|
l.logger.Info(message)
|
||||||
|
|
||||||
|
err = persistPublicIP(l.os.OpenFile, l.state.settings.IPFilepath,
|
||||||
ip.String(), l.puid, l.pgid)
|
ip.String(), l.puid, l.pgid)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
l.logger.Error(err)
|
l.logger.Error(err)
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
|
"github.com/qdm12/gluetun/internal/constants"
|
||||||
"github.com/qdm12/gluetun/internal/models"
|
"github.com/qdm12/gluetun/internal/models"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -23,7 +24,7 @@ func (u *updater) updateCyberghost(ctx context.Context) (err error) {
|
|||||||
|
|
||||||
func findCyberghostServers(ctx context.Context, lookupIP lookupIPFunc) (servers []models.CyberghostServer, err error) {
|
func findCyberghostServers(ctx context.Context, lookupIP lookupIPFunc) (servers []models.CyberghostServer, err error) {
|
||||||
groups := getCyberghostGroups()
|
groups := getCyberghostGroups()
|
||||||
allCountryCodes := getCountryCodes()
|
allCountryCodes := constants.CountryCodes()
|
||||||
cyberghostCountryCodes := getCyberghostSubdomainToRegion()
|
cyberghostCountryCodes := getCyberghostSubdomainToRegion()
|
||||||
possibleCountryCodes := mergeCountryCodes(cyberghostCountryCodes, allCountryCodes)
|
possibleCountryCodes := mergeCountryCodes(cyberghostCountryCodes, allCountryCodes)
|
||||||
|
|
||||||
@@ -115,7 +116,7 @@ func mergeCountryCodes(base, extend map[string]string) (merged map[string]string
|
|||||||
return merged
|
return merged
|
||||||
}
|
}
|
||||||
|
|
||||||
func getCyberghostSubdomainToRegion() map[string]string { //nolint:dupl
|
func getCyberghostSubdomainToRegion() map[string]string {
|
||||||
return map[string]string{
|
return map[string]string{
|
||||||
"af": "Afghanistan",
|
"af": "Afghanistan",
|
||||||
"ax": "Aland Islands",
|
"ax": "Aland Islands",
|
||||||
|
|||||||
@@ -2,16 +2,8 @@ package updater
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/qdm12/golibs/network"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func uniqueSortedIPs(ips []net.IP) []net.IP {
|
func uniqueSortedIPs(ips []net.IP) []net.IP {
|
||||||
@@ -33,35 +25,3 @@ func uniqueSortedIPs(ips []net.IP) []net.IP {
|
|||||||
})
|
})
|
||||||
return ips
|
return ips
|
||||||
}
|
}
|
||||||
|
|
||||||
var errBadHTTPStatus = errors.New("bad HTTP status received")
|
|
||||||
|
|
||||||
type ipInfoData struct {
|
|
||||||
Region string `json:"region"`
|
|
||||||
Country string `json:"country"`
|
|
||||||
City string `json:"city"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func getIPInfo(ctx context.Context, client network.Client, ip net.IP) (country, region, city string, err error) {
|
|
||||||
const baseURL = "https://ipinfo.io/"
|
|
||||||
url := baseURL + ip.String()
|
|
||||||
request, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
||||||
if err != nil {
|
|
||||||
return "", "", "", err
|
|
||||||
}
|
|
||||||
b, status, err := client.Do(request)
|
|
||||||
if err != nil {
|
|
||||||
return "", "", "", err
|
|
||||||
} else if status != http.StatusOK {
|
|
||||||
return "", "", "", fmt.Errorf("%w: %d", errBadHTTPStatus, status)
|
|
||||||
}
|
|
||||||
var data ipInfoData
|
|
||||||
if err := json.Unmarshal(b, &data); err != nil {
|
|
||||||
return "", "", "", err
|
|
||||||
}
|
|
||||||
country, ok := getCountryCodes()[strings.ToLower(data.Country)]
|
|
||||||
if !ok {
|
|
||||||
country = data.Country
|
|
||||||
}
|
|
||||||
return country, data.Region, data.City, nil
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -3,10 +3,13 @@ package updater
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"net/http"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/qdm12/gluetun/internal/models"
|
"github.com/qdm12/gluetun/internal/models"
|
||||||
|
"github.com/qdm12/gluetun/internal/publicip"
|
||||||
"github.com/qdm12/golibs/network"
|
"github.com/qdm12/golibs/network"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -60,7 +63,11 @@ func findPurevpnServers(ctx context.Context, client network.Client, lookupIP loo
|
|||||||
warnings = append(warnings, warning)
|
warnings = append(warnings, warning)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
country, region, city, err := getIPInfo(ctx, client, IPs[0])
|
|
||||||
|
// TODO remove once we move away from network.Client
|
||||||
|
const httpTimeout = 3 * time.Second
|
||||||
|
httpClient := &http.Client{Timeout: httpTimeout}
|
||||||
|
country, region, city, err := publicip.Info(ctx, httpClient, IPs[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, warnings, err
|
return nil, warnings, err
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user