
When the maxsurge daemonset gate is disabled, the registry and validation must properly handle stripping the field. In the special case where that would leave the MaxUnavailable field set to 0, we must set it to 1 which is the default value.
401 lines
14 KiB
Go
401 lines
14 KiB
Go
/*
|
|
Copyright 2015 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 daemonset
|
|
|
|
import (
|
|
"reflect"
|
|
"testing"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/util/diff"
|
|
"k8s.io/apimachinery/pkg/util/intstr"
|
|
"k8s.io/apimachinery/pkg/util/validation/field"
|
|
genericapirequest "k8s.io/apiserver/pkg/endpoints/request"
|
|
"k8s.io/apiserver/pkg/registry/rest"
|
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
|
featuregatetesting "k8s.io/component-base/featuregate/testing"
|
|
"k8s.io/kubernetes/pkg/apis/apps"
|
|
api "k8s.io/kubernetes/pkg/apis/core"
|
|
"k8s.io/kubernetes/pkg/features"
|
|
)
|
|
|
|
const (
|
|
fakeImageName = "fake-name"
|
|
fakeImage = "fakeimage"
|
|
daemonsetName = "test-daemonset"
|
|
namespace = "test-namespace"
|
|
)
|
|
|
|
func TestDaemonsetDefaultGarbageCollectionPolicy(t *testing.T) {
|
|
// Make sure we correctly implement the interface.
|
|
// Otherwise a typo could silently change the default.
|
|
var gcds rest.GarbageCollectionDeleteStrategy = Strategy
|
|
tests := []struct {
|
|
requestInfo genericapirequest.RequestInfo
|
|
expectedGCPolicy rest.GarbageCollectionPolicy
|
|
isNilRequestInfo bool
|
|
}{
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "extensions",
|
|
APIVersion: "v1beta1",
|
|
Resource: "daemonsets",
|
|
},
|
|
rest.OrphanDependents,
|
|
false,
|
|
},
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "apps",
|
|
APIVersion: "v1beta2",
|
|
Resource: "daemonsets",
|
|
},
|
|
rest.OrphanDependents,
|
|
false,
|
|
},
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "apps",
|
|
APIVersion: "v1",
|
|
Resource: "daemonsets",
|
|
},
|
|
rest.DeleteDependents,
|
|
false,
|
|
},
|
|
{
|
|
expectedGCPolicy: rest.DeleteDependents,
|
|
isNilRequestInfo: true,
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
context := genericapirequest.NewContext()
|
|
if !test.isNilRequestInfo {
|
|
context = genericapirequest.WithRequestInfo(context, &test.requestInfo)
|
|
}
|
|
if got, want := gcds.DefaultGarbageCollectionPolicy(context), test.expectedGCPolicy; got != want {
|
|
t.Errorf("%s/%s: DefaultGarbageCollectionPolicy() = %#v, want %#v", test.requestInfo.APIGroup,
|
|
test.requestInfo.APIVersion, got, want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSelectorImmutability(t *testing.T) {
|
|
tests := []struct {
|
|
requestInfo genericapirequest.RequestInfo
|
|
oldSelectorLabels map[string]string
|
|
newSelectorLabels map[string]string
|
|
expectedErrorList field.ErrorList
|
|
}{
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "apps",
|
|
APIVersion: "v1",
|
|
Resource: "daemonsets",
|
|
},
|
|
map[string]string{"a": "b"},
|
|
map[string]string{"c": "d"},
|
|
field.ErrorList{
|
|
&field.Error{
|
|
Type: field.ErrorTypeInvalid,
|
|
Field: field.NewPath("spec").Child("selector").String(),
|
|
BadValue: &metav1.LabelSelector{
|
|
MatchLabels: map[string]string{"c": "d"},
|
|
MatchExpressions: []metav1.LabelSelectorRequirement{},
|
|
},
|
|
Detail: "field is immutable",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "apps",
|
|
APIVersion: "v1beta2",
|
|
Resource: "daemonsets",
|
|
},
|
|
map[string]string{"a": "b"},
|
|
map[string]string{"c": "d"},
|
|
field.ErrorList{
|
|
&field.Error{
|
|
Type: field.ErrorTypeInvalid,
|
|
Field: field.NewPath("spec").Child("selector").String(),
|
|
BadValue: &metav1.LabelSelector{
|
|
MatchLabels: map[string]string{"c": "d"},
|
|
MatchExpressions: []metav1.LabelSelectorRequirement{},
|
|
},
|
|
Detail: "field is immutable",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
genericapirequest.RequestInfo{
|
|
APIGroup: "extensions",
|
|
APIVersion: "v1beta1",
|
|
Resource: "daemonsets",
|
|
},
|
|
map[string]string{"a": "b"},
|
|
map[string]string{"c": "d"},
|
|
field.ErrorList{},
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
oldDaemonSet := newDaemonSetWithSelectorLabels(test.oldSelectorLabels, 1)
|
|
newDaemonSet := newDaemonSetWithSelectorLabels(test.newSelectorLabels, 2)
|
|
context := genericapirequest.NewContext()
|
|
context = genericapirequest.WithRequestInfo(context, &test.requestInfo)
|
|
errorList := daemonSetStrategy{}.ValidateUpdate(context, newDaemonSet, oldDaemonSet)
|
|
if !reflect.DeepEqual(test.expectedErrorList, errorList) {
|
|
t.Errorf("Unexpected error list, expected: %v, actual: %v", test.expectedErrorList, errorList)
|
|
}
|
|
}
|
|
}
|
|
|
|
func newDaemonSetWithSelectorLabels(selectorLabels map[string]string, templateGeneration int64) *apps.DaemonSet {
|
|
return &apps.DaemonSet{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: daemonsetName,
|
|
Namespace: namespace,
|
|
ResourceVersion: "1",
|
|
},
|
|
Spec: apps.DaemonSetSpec{
|
|
Selector: &metav1.LabelSelector{
|
|
MatchLabels: selectorLabels,
|
|
MatchExpressions: []metav1.LabelSelectorRequirement{},
|
|
},
|
|
UpdateStrategy: apps.DaemonSetUpdateStrategy{
|
|
Type: apps.OnDeleteDaemonSetStrategyType,
|
|
},
|
|
TemplateGeneration: templateGeneration,
|
|
Template: api.PodTemplateSpec{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Labels: selectorLabels,
|
|
},
|
|
Spec: api.PodSpec{
|
|
RestartPolicy: api.RestartPolicyAlways,
|
|
DNSPolicy: api.DNSClusterFirst,
|
|
Containers: []api.Container{{Name: fakeImageName, Image: fakeImage, ImagePullPolicy: "IfNotPresent", TerminationMessagePolicy: api.TerminationMessageReadFile}},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
func makeDaemonSetWithSurge(unavailable intstr.IntOrString, surge intstr.IntOrString) *apps.DaemonSet {
|
|
return &apps.DaemonSet{
|
|
Spec: apps.DaemonSetSpec{
|
|
UpdateStrategy: apps.DaemonSetUpdateStrategy{
|
|
Type: apps.RollingUpdateDaemonSetStrategyType,
|
|
RollingUpdate: &apps.RollingUpdateDaemonSet{
|
|
MaxUnavailable: unavailable,
|
|
MaxSurge: surge,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
func TestDropDisabledField(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
enableSurge bool
|
|
ds *apps.DaemonSet
|
|
old *apps.DaemonSet
|
|
expect *apps.DaemonSet
|
|
}{
|
|
{
|
|
name: "not surge, no update",
|
|
enableSurge: false,
|
|
ds: &apps.DaemonSet{},
|
|
old: nil,
|
|
expect: &apps.DaemonSet{},
|
|
},
|
|
{
|
|
name: "not surge, field not used",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
old: nil,
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, field not used in old and new",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, field used",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
},
|
|
{
|
|
name: "not surge, field used, percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
},
|
|
{
|
|
name: "not surge, field used and cleared",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, field used and cleared, percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "surge, field not used",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
old: nil,
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "surge, field not used in old and new",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "surge, field used",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
old: nil,
|
|
expect: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
},
|
|
{
|
|
name: "surge, field used, percent",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromString("1%")),
|
|
},
|
|
{
|
|
name: "surge, field used in old and new",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
old: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
},
|
|
{
|
|
name: "surge, allows both fields (validation must catch)",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.FromInt(1)),
|
|
},
|
|
{
|
|
name: "surge, allows change from unavailable to surge",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "surge, allows change from surge to unvailable",
|
|
enableSurge: true,
|
|
ds: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
expect: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
},
|
|
{
|
|
name: "not surge, allows change from unavailable to surge",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, allows change from surge to unvailable",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromInt(1)),
|
|
old: makeDaemonSetWithSurge(intstr.FromInt(2), intstr.IntOrString{}),
|
|
expect: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, allows change from unavailable to surge, percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromString("2%"), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromString("2%"), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, allows change from surge to unvailable, percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.FromString("1%")),
|
|
old: makeDaemonSetWithSurge(intstr.FromString("2%"), intstr.IntOrString{}),
|
|
expect: makeDaemonSetWithSurge(intstr.IntOrString{}, intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, resets zero percent, one percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.FromString("1%")),
|
|
old: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.FromString("1%")),
|
|
},
|
|
{
|
|
name: "not surge, resets and clears when zero percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.IntOrString{}),
|
|
old: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.FromString("1%")),
|
|
expect: makeDaemonSetWithSurge(intstr.FromInt(1), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, sets zero percent, one percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.FromString("1%")),
|
|
old: nil,
|
|
expect: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.IntOrString{}),
|
|
},
|
|
{
|
|
name: "not surge, sets and clears zero percent",
|
|
enableSurge: false,
|
|
ds: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.IntOrString{}),
|
|
old: nil,
|
|
expect: makeDaemonSetWithSurge(intstr.FromString("0%"), intstr.IntOrString{}),
|
|
},
|
|
}
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.DaemonSetUpdateSurge, tc.enableSurge)()
|
|
old := tc.old.DeepCopy()
|
|
|
|
dropDaemonSetDisabledFields(tc.ds, tc.old)
|
|
|
|
// old obj should never be changed
|
|
if !reflect.DeepEqual(tc.old, old) {
|
|
t.Fatalf("old ds changed: %v", diff.ObjectReflectDiff(tc.old, old))
|
|
}
|
|
|
|
if !reflect.DeepEqual(tc.ds, tc.expect) {
|
|
t.Fatalf("unexpected ds spec: %v", diff.ObjectReflectDiff(tc.expect, tc.ds))
|
|
}
|
|
})
|
|
}
|
|
|
|
}
|