apiserver/pkg/storage/feature/feature_support_checker_tes...

270 lines
8.3 KiB
Go

/*
Copyright 2024 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 feature
import (
"context"
"fmt"
"testing"
"github.com/stretchr/testify/assert"
clientv3 "go.etcd.io/etcd/client/v3"
"k8s.io/apiserver/pkg/storage"
)
type mockEndpointVersion struct {
Endpoint string
Version string
Error error
}
// MockEtcdClient is a mock implementation of the EtcdClientInterface interface.
type MockEtcdClient struct {
EndpointVersion []mockEndpointVersion
}
func (m MockEtcdClient) getEndpoints() []string {
var endpoints []string
for _, ev := range m.EndpointVersion {
endpoints = append(endpoints, ev.Endpoint)
}
return endpoints
}
func (m MockEtcdClient) getVersion(endpoint string) (string, error) {
for _, ev := range m.EndpointVersion {
if ev.Endpoint == endpoint {
return ev.Version, ev.Error
}
}
// Never should happen, unless tests having a problem.
return "", fmt.Errorf("No version found")
}
func (m *MockEtcdClient) Endpoints() []string {
return m.getEndpoints()
}
// Status returns a mock status response.
func (m *MockEtcdClient) Status(ctx context.Context, endpoint string) (*clientv3.StatusResponse, error) {
version, err := m.getVersion(endpoint)
if err != nil {
return nil, err
}
// Return a mock status response
return &clientv3.StatusResponse{
Version: version,
}, nil
}
func TestSupports(t *testing.T) {
tests := []struct {
testName string
featureName string
expectedResult bool
}{
{
testName: "Disabled - with unknown feature",
featureName: "some unknown feature",
},
{
testName: "Disabled - with empty feature",
featureName: "",
},
{
testName: "Disabled - default",
featureName: storage.RequestWatchProgress,
expectedResult: false,
},
}
for _, tt := range tests {
t.Run(tt.testName, func(t *testing.T) {
var testFeatureSupportChecker FeatureSupportChecker = newDefaultFeatureSupportChecker()
supported := testFeatureSupportChecker.Supports(tt.featureName)
assert.Equal(t, tt.expectedResult, supported)
})
}
}
func TestSupportsRequestWatchProgress(t *testing.T) {
type testCase struct {
endpointsVersion []mockEndpointVersion
expectedResult bool
expectedError error
}
tests := []struct {
testName string
rounds []testCase
}{
{
testName: "Disabled - default disabled",
rounds: []testCase{{expectedResult: false}},
},
{
testName: "Enabled - supported versions bound",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.31", Endpoint: "localhost:2390"}},
expectedResult: true,
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.13", Endpoint: "localhost:2391"}},
expectedResult: true,
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2392"}},
expectedResult: true}},
},
{
testName: "Disabled - supported versions bound, 3.4.30",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.30", Endpoint: "localhost:2390"}},
expectedResult: false}},
},
{
testName: "Disabled - supported versions bound, 3.5.0",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.0", Endpoint: "localhost:2390"}},
expectedResult: false}},
},
{
testName: "Disabled - supported versions bound, 3.5.12",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.12", Endpoint: "localhost:2390"}},
expectedResult: false}},
},
{
testName: "Disabled - disables if called with one client doesn't support it",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.13", Endpoint: "localhost:2390"},
{Version: "3.5.10", Endpoint: "localhost:2391"}},
expectedResult: false}},
},
{
testName: "Disabled - disables if called with all client doesn't support it",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.9", Endpoint: "localhost:2390"},
{Version: "3.5.10", Endpoint: "localhost:2391"}},
expectedResult: false}},
},
{
testName: "Enabled - if provided client has at least one endpoint that supports it and no client that doesn't",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.31", Endpoint: "localhost:2390"},
{Version: "3.5.13", Endpoint: "localhost:2391"},
{Version: "3.5.14", Endpoint: "localhost:2392"},
{Version: "3.6.0", Endpoint: "localhost:2393"}},
expectedResult: true}},
},
{
testName: "Disabled - cannot be re-enabled",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.0", Endpoint: "localhost:2390"},
{Version: "3.4.1", Endpoint: "localhost:2391"}},
expectedResult: false},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2392"}},
expectedResult: false}},
},
{
testName: "Enabled - one client supports it and later disabled it with second client",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2390"},
{Version: "3.5.14", Endpoint: "localhost:2391"}},
expectedResult: true},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.0", Endpoint: "localhost:2392"}},
expectedResult: false}},
},
{
testName: "Disabled - malformed version would disable the supported cluster and can not be re-enabled again",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2390"}},
expectedResult: true,
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.4.--aaa", Endpoint: "localhost:2392"}},
expectedResult: false},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.13", Endpoint: "localhost:2393"}},
expectedResult: false}},
},
{
testName: "Enabled - error on first client, enabled success on second client",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2390", Error: fmt.Errorf("some error")}},
expectedResult: false,
expectedError: fmt.Errorf("failed checking etcd version, endpoint: %q: %w", "localhost:2390", fmt.Errorf("some error")),
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.14", Endpoint: "localhost:2391"}},
expectedResult: true}},
},
{
testName: "Disabled - enabled success on first client, error on second client, disabled success on third client",
rounds: []testCase{
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2390"}},
expectedResult: true,
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.6.0", Endpoint: "localhost:2391", Error: fmt.Errorf("some error")}},
expectedResult: true,
expectedError: fmt.Errorf("failed checking etcd version, endpoint: %q: %w", "localhost:2391", fmt.Errorf("some error")),
},
{endpointsVersion: []mockEndpointVersion{
{Version: "3.5.10", Endpoint: "localhost:2392"}},
expectedResult: false}},
},
{
testName: "Disabled - client doesn't have any endpoints",
rounds: []testCase{{endpointsVersion: []mockEndpointVersion{}, expectedResult: false}},
},
}
for _, tt := range tests {
t.Run(tt.testName, func(t *testing.T) {
var testFeatureSupportChecker = newDefaultFeatureSupportChecker()
for _, round := range tt.rounds {
// Mock Etcd client
mockClient := &MockEtcdClient{EndpointVersion: round.endpointsVersion}
ctx := context.Background()
for _, ep := range mockClient.Endpoints() {
err := testFeatureSupportChecker.clientSupportsRequestWatchProgress(ctx, mockClient, ep)
assert.Equal(t, round.expectedError, err)
}
supported := testFeatureSupportChecker.Supports(storage.RequestWatchProgress)
assert.Equal(t, round.expectedResult, supported)
}
})
}
}