feat(publicip): resilient public ip fetcher (#2518)
- `PUBLICIP_API` accepts a comma separated list of ip data sources, where the first one is the base default one, and sources after it are backup sources used if we are rate limited. - `PUBLICIP_API` defaults to `ipinfo,ifconfigco,ip2location,cloudflare` such that it now has `ifconfigco,ip2location,cloudflare` as backup ip data sources. - `PUBLICIP_API_TOKEN` accepts a comma separated list of ip data source tokens, each corresponding by position to the APIs listed in `PUBLICIP_API`. - logs ip data source when logging public ip information - assume a rate limiting error is for 30 days (no persistence) - ready for future live settings updates - consider an ip data source no longer banned if the token changes - keeps track of ban times when updating the list of fetchers
This commit is contained in:
@@ -1,3 +1,30 @@
|
||||
package settings
|
||||
|
||||
func boolPtr(b bool) *bool { return &b }
|
||||
import gomock "github.com/golang/mock/gomock"
|
||||
|
||||
type sourceKeyValue struct {
|
||||
key string
|
||||
value string
|
||||
}
|
||||
|
||||
func newMockSource(ctrl *gomock.Controller, keyValues []sourceKeyValue) *MockSource {
|
||||
source := NewMockSource(ctrl)
|
||||
var previousCall *gomock.Call
|
||||
for _, keyValue := range keyValues {
|
||||
transformedKey := keyValue.key
|
||||
keyTransformCall := source.EXPECT().KeyTransform(keyValue.key).Return(transformedKey)
|
||||
if previousCall != nil {
|
||||
keyTransformCall.After(previousCall)
|
||||
}
|
||||
isSet := keyValue.value != ""
|
||||
previousCall = source.EXPECT().Get(transformedKey).
|
||||
Return(keyValue.value, isSet).After(keyTransformCall)
|
||||
if isSet {
|
||||
previousCall = source.EXPECT().KeyTransform(keyValue.key).
|
||||
Return(transformedKey).After(previousCall)
|
||||
previousCall = source.EXPECT().String().
|
||||
Return("mock source").After(previousCall)
|
||||
}
|
||||
}
|
||||
return source
|
||||
}
|
||||
|
||||
4
internal/configuration/settings/mocks_generate_test.go
Normal file
4
internal/configuration/settings/mocks_generate_test.go
Normal file
@@ -0,0 +1,4 @@
|
||||
package settings
|
||||
|
||||
//go:generate mockgen -destination=mocks_test.go -package=$GOPACKAGE . Warner
|
||||
//go:generate mockgen -destination=mocks_reader_test.go -package=$GOPACKAGE github.com/qdm12/gosettings/reader Source
|
||||
77
internal/configuration/settings/mocks_reader_test.go
Normal file
77
internal/configuration/settings/mocks_reader_test.go
Normal file
@@ -0,0 +1,77 @@
|
||||
// Code generated by MockGen. DO NOT EDIT.
|
||||
// Source: github.com/qdm12/gosettings/reader (interfaces: Source)
|
||||
|
||||
// Package settings is a generated GoMock package.
|
||||
package settings
|
||||
|
||||
import (
|
||||
reflect "reflect"
|
||||
|
||||
gomock "github.com/golang/mock/gomock"
|
||||
)
|
||||
|
||||
// MockSource is a mock of Source interface.
|
||||
type MockSource struct {
|
||||
ctrl *gomock.Controller
|
||||
recorder *MockSourceMockRecorder
|
||||
}
|
||||
|
||||
// MockSourceMockRecorder is the mock recorder for MockSource.
|
||||
type MockSourceMockRecorder struct {
|
||||
mock *MockSource
|
||||
}
|
||||
|
||||
// NewMockSource creates a new mock instance.
|
||||
func NewMockSource(ctrl *gomock.Controller) *MockSource {
|
||||
mock := &MockSource{ctrl: ctrl}
|
||||
mock.recorder = &MockSourceMockRecorder{mock}
|
||||
return mock
|
||||
}
|
||||
|
||||
// EXPECT returns an object that allows the caller to indicate expected use.
|
||||
func (m *MockSource) EXPECT() *MockSourceMockRecorder {
|
||||
return m.recorder
|
||||
}
|
||||
|
||||
// Get mocks base method.
|
||||
func (m *MockSource) Get(arg0 string) (string, bool) {
|
||||
m.ctrl.T.Helper()
|
||||
ret := m.ctrl.Call(m, "Get", arg0)
|
||||
ret0, _ := ret[0].(string)
|
||||
ret1, _ := ret[1].(bool)
|
||||
return ret0, ret1
|
||||
}
|
||||
|
||||
// Get indicates an expected call of Get.
|
||||
func (mr *MockSourceMockRecorder) Get(arg0 interface{}) *gomock.Call {
|
||||
mr.mock.ctrl.T.Helper()
|
||||
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Get", reflect.TypeOf((*MockSource)(nil).Get), arg0)
|
||||
}
|
||||
|
||||
// KeyTransform mocks base method.
|
||||
func (m *MockSource) KeyTransform(arg0 string) string {
|
||||
m.ctrl.T.Helper()
|
||||
ret := m.ctrl.Call(m, "KeyTransform", arg0)
|
||||
ret0, _ := ret[0].(string)
|
||||
return ret0
|
||||
}
|
||||
|
||||
// KeyTransform indicates an expected call of KeyTransform.
|
||||
func (mr *MockSourceMockRecorder) KeyTransform(arg0 interface{}) *gomock.Call {
|
||||
mr.mock.ctrl.T.Helper()
|
||||
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "KeyTransform", reflect.TypeOf((*MockSource)(nil).KeyTransform), arg0)
|
||||
}
|
||||
|
||||
// String mocks base method.
|
||||
func (m *MockSource) String() string {
|
||||
m.ctrl.T.Helper()
|
||||
ret := m.ctrl.Call(m, "String")
|
||||
ret0, _ := ret[0].(string)
|
||||
return ret0
|
||||
}
|
||||
|
||||
// String indicates an expected call of String.
|
||||
func (mr *MockSourceMockRecorder) String() *gomock.Call {
|
||||
mr.mock.ctrl.T.Helper()
|
||||
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "String", reflect.TypeOf((*MockSource)(nil).String))
|
||||
}
|
||||
46
internal/configuration/settings/mocks_test.go
Normal file
46
internal/configuration/settings/mocks_test.go
Normal file
@@ -0,0 +1,46 @@
|
||||
// Code generated by MockGen. DO NOT EDIT.
|
||||
// Source: github.com/qdm12/gluetun/internal/configuration/settings (interfaces: Warner)
|
||||
|
||||
// Package settings is a generated GoMock package.
|
||||
package settings
|
||||
|
||||
import (
|
||||
reflect "reflect"
|
||||
|
||||
gomock "github.com/golang/mock/gomock"
|
||||
)
|
||||
|
||||
// MockWarner is a mock of Warner interface.
|
||||
type MockWarner struct {
|
||||
ctrl *gomock.Controller
|
||||
recorder *MockWarnerMockRecorder
|
||||
}
|
||||
|
||||
// MockWarnerMockRecorder is the mock recorder for MockWarner.
|
||||
type MockWarnerMockRecorder struct {
|
||||
mock *MockWarner
|
||||
}
|
||||
|
||||
// NewMockWarner creates a new mock instance.
|
||||
func NewMockWarner(ctrl *gomock.Controller) *MockWarner {
|
||||
mock := &MockWarner{ctrl: ctrl}
|
||||
mock.recorder = &MockWarnerMockRecorder{mock}
|
||||
return mock
|
||||
}
|
||||
|
||||
// EXPECT returns an object that allows the caller to indicate expected use.
|
||||
func (m *MockWarner) EXPECT() *MockWarnerMockRecorder {
|
||||
return m.recorder
|
||||
}
|
||||
|
||||
// Warn mocks base method.
|
||||
func (m *MockWarner) Warn(arg0 string) {
|
||||
m.ctrl.T.Helper()
|
||||
m.ctrl.Call(m, "Warn", arg0)
|
||||
}
|
||||
|
||||
// Warn indicates an expected call of Warn.
|
||||
func (mr *MockWarnerMockRecorder) Warn(arg0 interface{}) *gomock.Call {
|
||||
mr.mock.ctrl.T.Helper()
|
||||
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Warn", reflect.TypeOf((*MockWarner)(nil).Warn), arg0)
|
||||
}
|
||||
@@ -10,7 +10,7 @@ func Test_PortForwarding_String(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
settings := PortForwarding{
|
||||
Enabled: boolPtr(false),
|
||||
Enabled: ptrTo(false),
|
||||
}
|
||||
|
||||
s := settings.String()
|
||||
|
||||
@@ -20,15 +20,20 @@ type PublicIP struct {
|
||||
// to write to a file. It cannot be nil for the
|
||||
// internal state
|
||||
IPFilepath *string
|
||||
// API is the API name to use to fetch public IP information.
|
||||
// It can be cloudflare, ifconfigco, ip2location or ipinfo.
|
||||
// It defaults to ipinfo.
|
||||
API string
|
||||
// APIToken is the token to use for the IP data service
|
||||
// such as ipinfo.io. It can be the empty string to
|
||||
// indicate not to use a token. It cannot be nil for the
|
||||
// internal state.
|
||||
APIToken *string
|
||||
// APIs is the list of public ip APIs to use to fetch public IP information.
|
||||
// If there is more than one API, the first one is used
|
||||
// by default and the others are used as fallbacks in case of
|
||||
// the service rate limiting us. It defaults to use all services,
|
||||
// with the first one being ipinfo.io for historical reasons.
|
||||
APIs []PublicIPAPI
|
||||
}
|
||||
|
||||
type PublicIPAPI struct {
|
||||
// Name is the name of the public ip API service.
|
||||
// It can be "cloudflare", "ifconfigco", "ip2location" or "ipinfo".
|
||||
Name string
|
||||
// Token is the token to use for the public ip API service.
|
||||
Token string
|
||||
}
|
||||
|
||||
// UpdateWith deep copies the receiving settings, overrides the copy with
|
||||
@@ -53,9 +58,11 @@ func (p PublicIP) validate() (err error) {
|
||||
}
|
||||
}
|
||||
|
||||
_, err = api.ParseProvider(p.API)
|
||||
if err != nil {
|
||||
return fmt.Errorf("API name: %w", err)
|
||||
for _, publicIPAPI := range p.APIs {
|
||||
_, err = api.ParseProvider(publicIPAPI.Name)
|
||||
if err != nil {
|
||||
return fmt.Errorf("API name: %w", err)
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
@@ -65,23 +72,25 @@ func (p *PublicIP) copy() (copied PublicIP) {
|
||||
return PublicIP{
|
||||
Enabled: gosettings.CopyPointer(p.Enabled),
|
||||
IPFilepath: gosettings.CopyPointer(p.IPFilepath),
|
||||
API: p.API,
|
||||
APIToken: gosettings.CopyPointer(p.APIToken),
|
||||
APIs: gosettings.CopySlice(p.APIs),
|
||||
}
|
||||
}
|
||||
|
||||
func (p *PublicIP) overrideWith(other PublicIP) {
|
||||
p.Enabled = gosettings.OverrideWithPointer(p.Enabled, other.Enabled)
|
||||
p.IPFilepath = gosettings.OverrideWithPointer(p.IPFilepath, other.IPFilepath)
|
||||
p.API = gosettings.OverrideWithComparable(p.API, other.API)
|
||||
p.APIToken = gosettings.OverrideWithPointer(p.APIToken, other.APIToken)
|
||||
p.APIs = gosettings.OverrideWithSlice(p.APIs, other.APIs)
|
||||
}
|
||||
|
||||
func (p *PublicIP) setDefaults() {
|
||||
p.Enabled = gosettings.DefaultPointer(p.Enabled, true)
|
||||
p.IPFilepath = gosettings.DefaultPointer(p.IPFilepath, "/tmp/gluetun/ip")
|
||||
p.API = gosettings.DefaultComparable(p.API, "ipinfo")
|
||||
p.APIToken = gosettings.DefaultPointer(p.APIToken, "")
|
||||
p.APIs = gosettings.DefaultSlice(p.APIs, []PublicIPAPI{
|
||||
{Name: string(api.IPInfo)},
|
||||
{Name: string(api.Cloudflare)},
|
||||
{Name: string(api.IfConfigCo)},
|
||||
{Name: string(api.IP2Location)},
|
||||
})
|
||||
}
|
||||
|
||||
func (p PublicIP) String() string {
|
||||
@@ -99,10 +108,20 @@ func (p PublicIP) toLinesNode() (node *gotree.Node) {
|
||||
node.Appendf("IP file path: %s", *p.IPFilepath)
|
||||
}
|
||||
|
||||
node.Appendf("Public IP data API: %s", p.API)
|
||||
|
||||
if *p.APIToken != "" {
|
||||
node.Appendf("API token: %s", gosettings.ObfuscateKey(*p.APIToken))
|
||||
baseAPIString := "Public IP data base API: " + p.APIs[0].Name
|
||||
if p.APIs[0].Token != "" {
|
||||
baseAPIString += " (token " + gosettings.ObfuscateKey(p.APIs[0].Token) + ")"
|
||||
}
|
||||
node.Append(baseAPIString)
|
||||
if len(p.APIs) > 1 {
|
||||
backupAPIsNode := node.Append("Public IP data backup APIs:")
|
||||
for i := 1; i < len(p.APIs); i++ {
|
||||
message := p.APIs[i].Name
|
||||
if p.APIs[i].Token != "" {
|
||||
message += " (token " + gosettings.ObfuscateKey(p.APIs[i].Token) + ")"
|
||||
}
|
||||
backupAPIsNode.Append(message)
|
||||
}
|
||||
}
|
||||
|
||||
return node
|
||||
@@ -116,8 +135,21 @@ func (p *PublicIP) read(r *reader.Reader, warner Warner) (err error) {
|
||||
|
||||
p.IPFilepath = r.Get("PUBLICIP_FILE",
|
||||
reader.ForceLowercase(false), reader.RetroKeys("IP_STATUS_FILE"))
|
||||
p.API = r.String("PUBLICIP_API")
|
||||
p.APIToken = r.Get("PUBLICIP_API_TOKEN")
|
||||
|
||||
apiNames := r.CSV("PUBLICIP_API")
|
||||
if len(apiNames) > 0 {
|
||||
apiTokens := r.CSV("PUBLICIP_API_TOKEN")
|
||||
p.APIs = make([]PublicIPAPI, len(apiNames))
|
||||
for i := range apiNames {
|
||||
p.APIs[i].Name = apiNames[i]
|
||||
var token string
|
||||
if i < len(apiTokens) { // only set token if it exists
|
||||
token = apiTokens[i]
|
||||
}
|
||||
p.APIs[i].Token = token
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
||||
161
internal/configuration/settings/publicip_test.go
Normal file
161
internal/configuration/settings/publicip_test.go
Normal file
@@ -0,0 +1,161 @@
|
||||
package settings
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/golang/mock/gomock"
|
||||
"github.com/qdm12/gosettings/reader"
|
||||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
func Test_PublicIP_read(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
testCases := map[string]struct {
|
||||
makeReader func(ctrl *gomock.Controller) *reader.Reader
|
||||
makeWarner func(ctrl *gomock.Controller) Warner
|
||||
settings PublicIP
|
||||
errWrapped error
|
||||
errMessage string
|
||||
}{
|
||||
"nothing_read": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
},
|
||||
"single_api_no_token": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API", value: "ipinfo"},
|
||||
{key: "PUBLICIP_API_TOKEN"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
settings: PublicIP{
|
||||
APIs: []PublicIPAPI{
|
||||
{Name: "ipinfo"},
|
||||
},
|
||||
},
|
||||
},
|
||||
"single_api_with_token": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API", value: "ipinfo"},
|
||||
{key: "PUBLICIP_API_TOKEN", value: "xyz"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
settings: PublicIP{
|
||||
APIs: []PublicIPAPI{
|
||||
{Name: "ipinfo", Token: "xyz"},
|
||||
},
|
||||
},
|
||||
},
|
||||
"multiple_apis_no_token": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API", value: "ipinfo,ip2location"},
|
||||
{key: "PUBLICIP_API_TOKEN"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
settings: PublicIP{
|
||||
APIs: []PublicIPAPI{
|
||||
{Name: "ipinfo"},
|
||||
{Name: "ip2location"},
|
||||
},
|
||||
},
|
||||
},
|
||||
"multiple_apis_with_token": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API", value: "ipinfo,ip2location"},
|
||||
{key: "PUBLICIP_API_TOKEN", value: "xyz,abc"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
settings: PublicIP{
|
||||
APIs: []PublicIPAPI{
|
||||
{Name: "ipinfo", Token: "xyz"},
|
||||
{Name: "ip2location", Token: "abc"},
|
||||
},
|
||||
},
|
||||
},
|
||||
"multiple_apis_with_and_without_token": {
|
||||
makeReader: func(ctrl *gomock.Controller) *reader.Reader {
|
||||
source := newMockSource(ctrl, []sourceKeyValue{
|
||||
{key: "PUBLICIP_PERIOD"},
|
||||
{key: "PUBLICIP_ENABLED"},
|
||||
{key: "IP_STATUS_FILE"},
|
||||
{key: "PUBLICIP_FILE"},
|
||||
{key: "PUBLICIP_API", value: "ipinfo,ip2location"},
|
||||
{key: "PUBLICIP_API_TOKEN", value: "xyz"},
|
||||
})
|
||||
return reader.New(reader.Settings{
|
||||
Sources: []reader.Source{source},
|
||||
})
|
||||
},
|
||||
settings: PublicIP{
|
||||
APIs: []PublicIPAPI{
|
||||
{Name: "ipinfo", Token: "xyz"},
|
||||
{Name: "ip2location"},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
for name, testCase := range testCases {
|
||||
t.Run(name, func(t *testing.T) {
|
||||
t.Parallel()
|
||||
ctrl := gomock.NewController(t)
|
||||
|
||||
reader := testCase.makeReader(ctrl)
|
||||
var warner Warner
|
||||
if testCase.makeWarner != nil {
|
||||
warner = testCase.makeWarner(ctrl)
|
||||
}
|
||||
|
||||
var settings PublicIP
|
||||
err := settings.read(reader, warner)
|
||||
|
||||
assert.Equal(t, testCase.settings, settings)
|
||||
assert.ErrorIs(t, err, testCase.errWrapped)
|
||||
if testCase.errWrapped != nil {
|
||||
assert.EqualError(t, err, testCase.errMessage)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -79,7 +79,11 @@ func Test_Settings_String(t *testing.T) {
|
||||
| └── Process GID: 1000
|
||||
├── Public IP settings:
|
||||
| ├── IP file path: /tmp/gluetun/ip
|
||||
| └── Public IP data API: ipinfo
|
||||
| ├── Public IP data base API: ipinfo
|
||||
| └── Public IP data backup APIs:
|
||||
| ├── cloudflare
|
||||
| ├── ifconfigco
|
||||
| └── ip2location
|
||||
└── Version settings:
|
||||
└── Enabled: yes`,
|
||||
},
|
||||
|
||||
Reference in New Issue
Block a user