external-dns/provider/cached_provider_test.go
Ivan Ka eb40149b99
test: improve code coverage (#6321)
* test: improve code coverage

Signed-off-by: ivan katliarchuk <ivan.katliarchuk@gmail.com>

* test: improve code coverage

Signed-off-by: ivan katliarchuk <ivan.katliarchuk@gmail.com>

---------

Signed-off-by: ivan katliarchuk <ivan.katliarchuk@gmail.com>
2026-03-29 23:46:12 +05:30

229 lines
7.5 KiB
Go

/*
Copyright 2017 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package provider
import (
"context"
"errors"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"sigs.k8s.io/external-dns/endpoint"
"sigs.k8s.io/external-dns/plan"
)
type testProviderFunc struct {
records func(ctx context.Context) ([]*endpoint.Endpoint, error)
applyChanges func(ctx context.Context, changes *plan.Changes) error
propertyValuesEqual func(name string, previous string, current string) bool
adjustEndpoints func(endpoints []*endpoint.Endpoint) ([]*endpoint.Endpoint, error)
getDomainFilter func() endpoint.DomainFilterInterface
}
func (p *testProviderFunc) Records(ctx context.Context) ([]*endpoint.Endpoint, error) {
return p.records(ctx)
}
func (p *testProviderFunc) ApplyChanges(ctx context.Context, changes *plan.Changes) error {
return p.applyChanges(ctx, changes)
}
func (p *testProviderFunc) PropertyValuesEqual(name string, previous string, current string) bool {
return p.propertyValuesEqual(name, previous, current)
}
func (p *testProviderFunc) AdjustEndpoints(endpoints []*endpoint.Endpoint) ([]*endpoint.Endpoint, error) {
return p.adjustEndpoints(endpoints)
}
func (p *testProviderFunc) GetDomainFilter() endpoint.DomainFilterInterface {
return p.getDomainFilter()
}
func recordsNotCalled(t *testing.T) func(ctx context.Context) ([]*endpoint.Endpoint, error) {
return func(_ context.Context) ([]*endpoint.Endpoint, error) {
t.Errorf("unexpected call to Records")
return nil, nil
}
}
func applyChangesNotCalled(t *testing.T) func(_ context.Context, _ *plan.Changes) error {
return func(_ context.Context, _ *plan.Changes) error {
t.Errorf("unexpected call to ApplyChanges")
return nil
}
}
func propertyValuesEqualNotCalled(t *testing.T) func(name string, previous string, current string) bool {
return func(_ string, _ string, _ string) bool {
t.Errorf("unexpected call to PropertyValuesEqual")
return false
}
}
func adjustEndpointsNotCalled(t *testing.T) func(endpoints []*endpoint.Endpoint) ([]*endpoint.Endpoint, error) {
return func(endpoints []*endpoint.Endpoint) ([]*endpoint.Endpoint, error) {
t.Errorf("unexpected call to AdjustEndpoints")
return endpoints, errors.New("unexpected call to AdjustEndpoints")
}
}
func newTestProviderFunc(t *testing.T) *testProviderFunc {
return &testProviderFunc{
records: recordsNotCalled(t),
applyChanges: applyChangesNotCalled(t),
propertyValuesEqual: propertyValuesEqualNotCalled(t),
adjustEndpoints: adjustEndpointsNotCalled(t),
}
}
func TestNewCachedProvider(t *testing.T) {
inner := newTestProviderFunc(t)
delay := 5 * time.Minute
cp := NewCachedProvider(inner, delay)
assert.Equal(t, inner, cp.Provider)
assert.Equal(t, delay, cp.RefreshDelay)
assert.Nil(t, cp.cache)
assert.True(t, cp.lastRead.IsZero())
}
func TestCachedProviderRecordsError(t *testing.T) {
testProvider := newTestProviderFunc(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return nil, assert.AnError
}
cp := NewCachedProvider(testProvider, 0)
_, err := cp.Records(t.Context())
require.ErrorIs(t, err, assert.AnError)
assert.Nil(t, cp.cache)
}
func TestCachedProviderCallsProviderOnFirstCall(t *testing.T) {
testProvider := newTestProviderFunc(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "domain.fqdn"}}, nil
}
provider := CachedProvider{
Provider: testProvider,
}
endpoints, err := provider.Records(t.Context())
assert.NoError(t, err)
require.NotNil(t, endpoints)
require.Len(t, endpoints, 1)
require.NotNil(t, endpoints[0])
assert.Equal(t, "domain.fqdn", endpoints[0].DNSName)
}
func TestCachedProviderUsesCacheWhileValid(t *testing.T) {
testProvider := newTestProviderFunc(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "domain.fqdn"}}, nil
}
provider := CachedProvider{
RefreshDelay: 30 * time.Second,
Provider: testProvider,
}
_, err := provider.Records(t.Context())
require.NoError(t, err)
t.Run("With consecutive calls within the caching time frame", func(t *testing.T) {
testProvider.records = recordsNotCalled(t)
endpoints, err := provider.Records(t.Context())
assert.NoError(t, err)
require.NotNil(t, endpoints)
require.Len(t, endpoints, 1)
require.NotNil(t, endpoints[0])
assert.Equal(t, "domain.fqdn", endpoints[0].DNSName)
})
t.Run("When the caching time frame is exceeded", func(t *testing.T) {
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "new.domain.fqdn"}}, nil
}
provider.lastRead = time.Now().Add(-20 * time.Minute)
endpoints, err := provider.Records(t.Context())
assert.NoError(t, err)
require.NotNil(t, endpoints)
require.Len(t, endpoints, 1)
require.NotNil(t, endpoints[0])
assert.Equal(t, "new.domain.fqdn", endpoints[0].DNSName)
})
}
func TestCachedProviderForcesCacheRefreshOnUpdate(t *testing.T) {
testProvider := newTestProviderFunc(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "domain.fqdn"}}, nil
}
provider := CachedProvider{
RefreshDelay: 30 * time.Second,
Provider: testProvider,
}
_, err := provider.Records(t.Context())
require.NoError(t, err)
t.Run("When empty changes are applied", func(t *testing.T) {
testProvider.records = recordsNotCalled(t)
testProvider.applyChanges = func(_ context.Context, _ *plan.Changes) error {
return nil
}
err := provider.ApplyChanges(t.Context(), &plan.Changes{})
assert.NoError(t, err)
t.Run("Next call to Records is cached", func(t *testing.T) {
testProvider.applyChanges = applyChangesNotCalled(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "new.domain.fqdn"}}, nil
}
endpoints, err := provider.Records(t.Context())
assert.NoError(t, err)
require.NotNil(t, endpoints)
require.Len(t, endpoints, 1)
require.NotNil(t, endpoints[0])
assert.Equal(t, "domain.fqdn", endpoints[0].DNSName)
})
})
t.Run("When changes are applied", func(t *testing.T) {
testProvider.records = recordsNotCalled(t)
testProvider.applyChanges = func(_ context.Context, _ *plan.Changes) error {
return nil
}
err := provider.ApplyChanges(t.Context(), &plan.Changes{
Create: []*endpoint.Endpoint{
{DNSName: "hello.world"},
},
})
assert.NoError(t, err)
t.Run("Next call to Records is not cached", func(t *testing.T) {
testProvider.applyChanges = applyChangesNotCalled(t)
testProvider.records = func(_ context.Context) ([]*endpoint.Endpoint, error) {
return []*endpoint.Endpoint{{DNSName: "new.domain.fqdn"}}, nil
}
endpoints, err := provider.Records(t.Context())
assert.NoError(t, err)
require.NotNil(t, endpoints)
require.Len(t, endpoints, 1)
require.NotNil(t, endpoints[0])
assert.Equal(t, "new.domain.fqdn", endpoints[0].DNSName)
})
})
}