ip/cache.go: skip storing a NetInterfaceIPResolver in struct, Define a CacheDefaultCallback type and have it passed to the new GetWithDefault function instead.
This commit is contained in:
parent
0f21902ffd
commit
a6c98a3209
3 changed files with 74 additions and 16 deletions
26
app/app.go
26
app/app.go
|
|
@ -1,8 +1,10 @@
|
||||||
package app
|
package app
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net"
|
"net"
|
||||||
|
"time"
|
||||||
|
|
||||||
"dnsupdater/provider/manager"
|
"dnsupdater/provider/manager"
|
||||||
|
|
||||||
|
|
@ -11,12 +13,25 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type App struct {
|
type App struct {
|
||||||
iplookup ip.NetInterfaceIPResolver
|
cache *ip.Cache
|
||||||
|
|
||||||
|
cacheDefaultCallback ip.CacheDefaultCallback
|
||||||
|
|
||||||
// Updater manager
|
// Updater manager
|
||||||
ProviderManager *manager.Manager
|
ProviderManager *manager.Manager
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func makeCacheCallback(service resolver.Service) ip.CacheDefaultCallback {
|
||||||
|
return func(name string) (net.IP, error) {
|
||||||
|
if name == resolver.WAN_IFACE {
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second*10)
|
||||||
|
defer cancel()
|
||||||
|
return service.Lookup(ctx)
|
||||||
|
}
|
||||||
|
return ip.GetInterfaceIP(name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func NewApp(config *Config) (*App, error) {
|
func NewApp(config *Config) (*App, error) {
|
||||||
providerMgr := manager.New()
|
providerMgr := manager.New()
|
||||||
// providerMgr.Register("digitalocean", digitalocean.New(config.Services.DigitalOcean.Token))
|
// providerMgr.Register("digitalocean", digitalocean.New(config.Services.DigitalOcean.Token))
|
||||||
|
|
@ -25,18 +40,19 @@ func NewApp(config *Config) (*App, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
l := resolver.Get(config.Services.IPLookup)
|
service := resolver.Get(config.Services.IPLookup)
|
||||||
|
|
||||||
if l == nil {
|
if service == nil {
|
||||||
return nil, fmt.Errorf("Failed to load lookup service: %s", config.Services.IPLookup)
|
return nil, fmt.Errorf("Failed to load lookup service: %s", config.Services.IPLookup)
|
||||||
}
|
}
|
||||||
|
|
||||||
return &App{
|
return &App{
|
||||||
ProviderManager: providerMgr,
|
ProviderManager: providerMgr,
|
||||||
iplookup: ip.NewCache(ip.LookupWrapper(l)).Get,
|
cache: ip.NewCache(),
|
||||||
|
cacheDefaultCallback: makeCacheCallback(service),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a App) GetIP(iface_name string) (net.IP, error) {
|
func (a App) GetIP(iface_name string) (net.IP, error) {
|
||||||
return a.iplookup(iface_name)
|
return a.cache.GetWithDefault(iface_name, a.cacheDefaultCallback)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
17
ip/cache.go
17
ip/cache.go
|
|
@ -1,17 +1,18 @@
|
||||||
package ip
|
package ip
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"net"
|
"net"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type CacheDefaultCallback func(name string) (net.IP, error)
|
||||||
|
|
||||||
type Cache struct {
|
type Cache struct {
|
||||||
resolver NetInterfaceIPResolver
|
|
||||||
items map[string]net.IP
|
items map[string]net.IP
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewCache(resolver NetInterfaceIPResolver) *Cache {
|
func NewCache() *Cache {
|
||||||
return &Cache{
|
return &Cache{
|
||||||
resolver: resolver,
|
|
||||||
items: make(map[string]net.IP),
|
items: make(map[string]net.IP),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -21,8 +22,16 @@ func (c Cache) Get(name string) (net.IP, error) {
|
||||||
if cached, ok := c.items[name]; ok {
|
if cached, ok := c.items[name]; ok {
|
||||||
return cached, nil
|
return cached, nil
|
||||||
}
|
}
|
||||||
|
return nil, errors.New("key did not exist")
|
||||||
|
}
|
||||||
|
|
||||||
ip, err := c.resolver(name)
|
func (c Cache) GetWithDefault(name string, callback CacheDefaultCallback) (net.IP, error) {
|
||||||
|
// Return cached entry.
|
||||||
|
if cached, ok := c.items[name]; ok {
|
||||||
|
return cached, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
ip, err := callback(name)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
c.Set(name, ip)
|
c.Set(name, ip)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -9,13 +9,20 @@ import (
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func mockResolver(t *testing.T, expected_name string, ip net.IP, err error) NetInterfaceIPResolver {
|
func defaultCallback(t *testing.T, expected_name string, ip net.IP, err error) CacheDefaultCallback {
|
||||||
return func(name string) (net.IP, error) {
|
return func(name string) (net.IP, error) {
|
||||||
assert.Equal(t, expected_name, name)
|
assert.Equal(t, expected_name, name)
|
||||||
return ip, err
|
return ip, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func dontCallDefaultCallback(t *testing.T) CacheDefaultCallback {
|
||||||
|
return func(name string) (net.IP, error) {
|
||||||
|
t.Error("Should not have been called")
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestCache_Get(t *testing.T) {
|
func TestCache_Get(t *testing.T) {
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
name string
|
name string
|
||||||
|
|
@ -24,9 +31,8 @@ func TestCache_Get(t *testing.T) {
|
||||||
want net.IP
|
want net.IP
|
||||||
wantErr bool
|
wantErr bool
|
||||||
}{
|
}{
|
||||||
{"FromCache", &Cache{resolver: nil, items: map[string]net.IP{"eth0": net.IPv4(10, 4, 0, 1)}}, "eth0", net.IPv4(10, 4, 0, 1), false},
|
{"Exists in cache", &Cache{items: map[string]net.IP{"eth0": net.IPv4(10, 4, 0, 1)}}, "eth0", net.IPv4(10, 4, 0, 1), false},
|
||||||
{"FromResolver", NewCache(mockResolver(t, "eth1", net.IPv4(192, 172, 44, 25), nil)), "eth1", net.IPv4(192, 172, 44, 25), false},
|
{"Did not exist in cache", &Cache{items: map[string]net.IP{}}, "eth0", nil, true},
|
||||||
{"NoInterface", NewCache(mockResolver(t, "eth2", nil, errors.New("Invalid interface"))), "eth2", nil, true},
|
|
||||||
}
|
}
|
||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
|
@ -41,3 +47,30 @@ func TestCache_Get(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCache_GetWithDefault(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
c *Cache
|
||||||
|
def CacheDefaultCallback
|
||||||
|
iface string
|
||||||
|
want net.IP
|
||||||
|
wantErr bool
|
||||||
|
}{
|
||||||
|
{"Exists in cache", &Cache{items: map[string]net.IP{"eth0": net.IPv4(10, 4, 0, 1)}}, dontCallDefaultCallback(t), "eth0", net.IPv4(10, 4, 0, 1), false},
|
||||||
|
{"Did not exists in cache", NewCache(), defaultCallback(t, "eth1", net.IPv4(192, 172, 44, 25), nil), "eth1", net.IPv4(192, 172, 44, 25), false},
|
||||||
|
{"Callback returns error", NewCache(), defaultCallback(t, "eth1", nil, errors.New("some error")), "eth1", nil, true},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
got, err := tt.c.GetWithDefault(tt.iface, tt.def)
|
||||||
|
if (err != nil) != tt.wantErr {
|
||||||
|
t.Errorf("Cache.Get() error = %v, wantErr %v", err, tt.wantErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !reflect.DeepEqual(got, tt.want) {
|
||||||
|
t.Errorf("Cache.Get() = %v, want %v", got, tt.want)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue