apiserver/pkg/storage/interfaces_test.go

197 lines
4.9 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 storage
import (
"errors"
"strings"
"testing"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
func TestPreconditionsCheckWithNilObject(t *testing.T) {
p := &Preconditions{}
err := p.Check("foo", nil)
if err == nil {
t.Fatalf("expected an error")
}
var internalErr InternalError
if !errors.As(err, &internalErr) {
t.Fatalf("expected error to be of type: %T, but got: %#v", InternalError{}, err)
}
if want := "can't enforce preconditions"; !strings.Contains(internalErr.Error(), want) {
t.Errorf("expected error to contain %q", want)
}
}
func TestValidateListOptions(t *testing.T) {
testCases := []struct {
name string
opts ListOptions
expectedError string
expectedRev int64
expectedContinueKey string
}{
{
name: "specifying resource version when using continue",
opts: ListOptions{
Recursive: true,
ResourceVersion: "200",
Predicate: SelectionPredicate{
Continue: encodeContinueOrDie("meta.k8s.io/v1", 100, "continue"),
},
},
expectedError: "specifying resource version is not allowed when using continue",
},
{
name: "invalid resource version",
opts: ListOptions{
ResourceVersion: "invalid",
},
expectedError: "invalid resource version",
},
{
name: "unknown ResourceVersionMatch value",
opts: ListOptions{
ResourceVersion: "200",
ResourceVersionMatch: "unknown",
},
expectedError: "unknown ResourceVersionMatch value",
},
{
name: "use continueRV",
opts: ListOptions{
ResourceVersion: "0",
Recursive: true,
Predicate: SelectionPredicate{
Continue: encodeContinueOrDie("meta.k8s.io/v1", 100, "continue"),
},
},
expectedRev: 100,
expectedContinueKey: "continue",
},
{
name: "use continueRV with empty rv",
opts: ListOptions{
ResourceVersion: "",
Recursive: true,
Predicate: SelectionPredicate{
Continue: encodeContinueOrDie("meta.k8s.io/v1", 100, "continue"),
},
},
expectedRev: 100,
expectedContinueKey: "continue",
},
{
name: "continueRV = 0",
opts: ListOptions{
ResourceVersion: "",
Recursive: true,
Predicate: SelectionPredicate{
Continue: encodeContinueOrDie("meta.k8s.io/v1", 0, "continue"),
},
},
expectedError: "invalid continue token",
},
{
name: "continueRV < 0",
opts: ListOptions{
ResourceVersion: "",
Recursive: true,
Predicate: SelectionPredicate{
Continue: encodeContinueOrDie("meta.k8s.io/v1", -1, "continue"),
},
},
expectedRev: 0,
expectedContinueKey: "continue",
},
{
name: "default",
opts: ListOptions{},
expectedRev: 0,
},
{
name: "rev resolve to 0 if ResourceVersionMatchNotOlderThan",
opts: ListOptions{
ResourceVersion: "200",
ResourceVersionMatch: metav1.ResourceVersionMatchNotOlderThan,
},
expectedRev: 0,
},
{
name: "specified rev if ResourceVersionMatchExact",
opts: ListOptions{
ResourceVersion: "200",
ResourceVersionMatch: metav1.ResourceVersionMatchExact,
},
expectedRev: 200,
},
{
name: "rev resolve to 0 if not recursive",
opts: ListOptions{
ResourceVersion: "200",
Predicate: SelectionPredicate{
Limit: 1,
},
},
expectedRev: 0,
},
{
name: "rev resolve to 0 if limit unspecified",
opts: ListOptions{
ResourceVersion: "200",
Recursive: true,
},
expectedRev: 0,
},
{
name: "specified rev if recursive with limit",
opts: ListOptions{
ResourceVersion: "200",
Recursive: true,
Predicate: SelectionPredicate{
Limit: 1,
},
},
expectedRev: 200,
},
}
for _, tt := range testCases {
tt := tt
t.Run(tt.name, func(t *testing.T) {
withRev, continueKey, err := ValidateListOptions("", APIObjectVersioner{}, tt.opts)
if len(tt.expectedError) > 0 {
if err == nil || !strings.Contains(err.Error(), tt.expectedError) {
t.Fatalf("expected error: %s, but got: %v", tt.expectedError, err)
}
return
}
if err != nil {
t.Fatalf("resolveRevForGetList failed: %v", err)
}
if withRev != tt.expectedRev {
t.Errorf("%s: expecting rev = %d, but get %d", tt.name, tt.expectedRev, withRev)
}
if continueKey != tt.expectedContinueKey {
t.Errorf("%s: expecting continueKey = %q, but get %q", tt.name, tt.expectedContinueKey, continueKey)
}
})
}
}