123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354 |
- /*
- 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 deployment
- import (
- "math"
- "testing"
- "time"
- apps "k8s.io/api/apps/v1"
- "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/client-go/kubernetes/fake"
- "k8s.io/client-go/util/workqueue"
- "k8s.io/kubernetes/pkg/controller/deployment/util"
- )
- func newDeploymentStatus(replicas, updatedReplicas, availableReplicas int32) apps.DeploymentStatus {
- return apps.DeploymentStatus{
- Replicas: replicas,
- UpdatedReplicas: updatedReplicas,
- AvailableReplicas: availableReplicas,
- }
- }
- // assumes the retuned deployment is always observed - not needed to be tested here.
- func currentDeployment(pds *int32, replicas, statusReplicas, updatedReplicas, availableReplicas int32, conditions []apps.DeploymentCondition) *apps.Deployment {
- d := &apps.Deployment{
- ObjectMeta: metav1.ObjectMeta{
- Name: "progress-test",
- },
- Spec: apps.DeploymentSpec{
- ProgressDeadlineSeconds: pds,
- Replicas: &replicas,
- Strategy: apps.DeploymentStrategy{
- Type: apps.RecreateDeploymentStrategyType,
- },
- },
- Status: newDeploymentStatus(statusReplicas, updatedReplicas, availableReplicas),
- }
- d.Status.Conditions = conditions
- return d
- }
- // helper to create RS with given availableReplicas
- func newRSWithAvailable(name string, specReplicas, statusReplicas, availableReplicas int) *apps.ReplicaSet {
- rs := rs(name, specReplicas, nil, metav1.Time{})
- rs.Status = apps.ReplicaSetStatus{
- Replicas: int32(statusReplicas),
- AvailableReplicas: int32(availableReplicas),
- }
- return rs
- }
- func TestRequeueStuckDeployment(t *testing.T) {
- pds := int32(60)
- infinite := int32(math.MaxInt32)
- failed := []apps.DeploymentCondition{
- {
- Type: apps.DeploymentProgressing,
- Status: v1.ConditionFalse,
- Reason: util.TimedOutReason,
- },
- }
- stuck := []apps.DeploymentCondition{
- {
- Type: apps.DeploymentProgressing,
- Status: v1.ConditionTrue,
- LastUpdateTime: metav1.Date(2017, 2, 15, 18, 49, 00, 00, time.UTC),
- },
- }
- tests := []struct {
- name string
- d *apps.Deployment
- status apps.DeploymentStatus
- nowFn func() time.Time
- expected time.Duration
- }{
- {
- name: "nil progressDeadlineSeconds specified",
- d: currentDeployment(nil, 4, 3, 3, 2, nil),
- status: newDeploymentStatus(3, 3, 2),
- expected: time.Duration(-1),
- },
- {
- name: "infinite progressDeadlineSeconds specified",
- d: currentDeployment(&infinite, 4, 3, 3, 2, nil),
- status: newDeploymentStatus(3, 3, 2),
- expected: time.Duration(-1),
- },
- {
- name: "no progressing condition found",
- d: currentDeployment(&pds, 4, 3, 3, 2, nil),
- status: newDeploymentStatus(3, 3, 2),
- expected: time.Duration(-1),
- },
- {
- name: "complete deployment does not need to be requeued",
- d: currentDeployment(&pds, 3, 3, 3, 3, nil),
- status: newDeploymentStatus(3, 3, 3),
- expected: time.Duration(-1),
- },
- {
- name: "already failed deployment does not need to be requeued",
- d: currentDeployment(&pds, 3, 3, 3, 0, failed),
- status: newDeploymentStatus(3, 3, 0),
- expected: time.Duration(-1),
- },
- {
- name: "stuck deployment - 30s",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 49, 30, 00, time.UTC).Time },
- expected: 30 * time.Second,
- },
- {
- name: "stuck deployment - 1s",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 49, 59, 00, time.UTC).Time },
- expected: 1 * time.Second,
- },
- {
- name: "failed deployment - less than a second => now",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 49, 59, 1, time.UTC).Time },
- expected: time.Duration(0),
- },
- {
- name: "failed deployment - now",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 50, 00, 00, time.UTC).Time },
- expected: time.Duration(0),
- },
- {
- name: "failed deployment - 1s after deadline",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 50, 01, 00, time.UTC).Time },
- expected: time.Duration(0),
- },
- {
- name: "failed deployment - 60s after deadline",
- d: currentDeployment(&pds, 3, 3, 3, 1, stuck),
- status: newDeploymentStatus(3, 3, 1),
- nowFn: func() time.Time { return metav1.Date(2017, 2, 15, 18, 51, 00, 00, time.UTC).Time },
- expected: time.Duration(0),
- },
- }
- dc := &DeploymentController{
- queue: workqueue.NewNamedRateLimitingQueue(workqueue.DefaultControllerRateLimiter(), "doesnt_matter"),
- }
- dc.enqueueDeployment = dc.enqueue
- for _, test := range tests {
- t.Run(test.name, func(t *testing.T) {
- if test.nowFn != nil {
- nowFn = test.nowFn
- }
- got := dc.requeueStuckDeployment(test.d, test.status)
- if got != test.expected {
- t.Errorf("%s: got duration: %v, expected duration: %v", test.name, got, test.expected)
- }
- })
- }
- }
- func TestSyncRolloutStatus(t *testing.T) {
- pds := int32(60)
- testTime := metav1.Date(2017, 2, 15, 18, 49, 00, 00, time.UTC)
- failedTimedOut := apps.DeploymentCondition{
- Type: apps.DeploymentProgressing,
- Status: v1.ConditionFalse,
- Reason: util.TimedOutReason,
- }
- newRSAvailable := apps.DeploymentCondition{
- Type: apps.DeploymentProgressing,
- Status: v1.ConditionTrue,
- Reason: util.NewRSAvailableReason,
- LastUpdateTime: testTime,
- LastTransitionTime: testTime,
- }
- replicaSetUpdated := apps.DeploymentCondition{
- Type: apps.DeploymentProgressing,
- Status: v1.ConditionTrue,
- Reason: util.ReplicaSetUpdatedReason,
- LastUpdateTime: testTime,
- LastTransitionTime: testTime,
- }
- tests := []struct {
- name string
- d *apps.Deployment
- allRSs []*apps.ReplicaSet
- newRS *apps.ReplicaSet
- conditionType apps.DeploymentConditionType
- conditionStatus v1.ConditionStatus
- conditionReason string
- lastUpdate metav1.Time
- lastTransition metav1.Time
- }{
- {
- name: "General: remove Progressing condition and do not estimate progress if deployment has no Progress Deadline",
- d: currentDeployment(nil, 3, 2, 2, 2, []apps.DeploymentCondition{replicaSetUpdated}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("bar", 0, 1, 1)},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- },
- {
- name: "General: do not estimate progress of deployment with only one active ReplicaSet",
- d: currentDeployment(&pds, 3, 3, 3, 3, []apps.DeploymentCondition{newRSAvailable}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("bar", 3, 3, 3)},
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.NewRSAvailableReason,
- lastUpdate: testTime,
- lastTransition: testTime,
- },
- {
- name: "DeploymentProgressing: dont update lastTransitionTime if deployment already has Progressing=True",
- d: currentDeployment(&pds, 3, 2, 2, 2, []apps.DeploymentCondition{replicaSetUpdated}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("bar", 0, 1, 1)},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.ReplicaSetUpdatedReason,
- lastTransition: testTime,
- },
- {
- name: "DeploymentProgressing: update everything if deployment has Progressing=False",
- d: currentDeployment(&pds, 3, 2, 2, 2, []apps.DeploymentCondition{failedTimedOut}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("bar", 0, 1, 1)},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.ReplicaSetUpdatedReason,
- },
- {
- name: "DeploymentProgressing: create Progressing condition if it does not exist",
- d: currentDeployment(&pds, 3, 2, 2, 2, []apps.DeploymentCondition{}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("bar", 0, 1, 1)},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.ReplicaSetUpdatedReason,
- },
- {
- name: "DeploymentComplete: dont update lastTransitionTime if deployment already has Progressing=True",
- d: currentDeployment(&pds, 3, 3, 3, 3, []apps.DeploymentCondition{replicaSetUpdated}),
- allRSs: []*apps.ReplicaSet{},
- newRS: newRSWithAvailable("foo", 3, 3, 3),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.NewRSAvailableReason,
- lastTransition: testTime,
- },
- {
- name: "DeploymentComplete: update everything if deployment has Progressing=False",
- d: currentDeployment(&pds, 3, 3, 3, 3, []apps.DeploymentCondition{failedTimedOut}),
- allRSs: []*apps.ReplicaSet{},
- newRS: newRSWithAvailable("foo", 3, 3, 3),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.NewRSAvailableReason,
- },
- {
- name: "DeploymentComplete: create Progressing condition if it does not exist",
- d: currentDeployment(&pds, 3, 3, 3, 3, []apps.DeploymentCondition{}),
- allRSs: []*apps.ReplicaSet{},
- newRS: newRSWithAvailable("foo", 3, 3, 3),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.NewRSAvailableReason,
- },
- {
- name: "DeploymentComplete: defend against NPE when newRS=nil",
- d: currentDeployment(&pds, 0, 3, 3, 3, []apps.DeploymentCondition{replicaSetUpdated}),
- allRSs: []*apps.ReplicaSet{newRSWithAvailable("foo", 0, 0, 0)},
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionTrue,
- conditionReason: util.NewRSAvailableReason,
- },
- {
- name: "DeploymentTimedOut: update status if rollout exceeds Progress Deadline",
- d: currentDeployment(&pds, 3, 2, 2, 2, []apps.DeploymentCondition{replicaSetUpdated}),
- allRSs: []*apps.ReplicaSet{},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionFalse,
- conditionReason: util.TimedOutReason,
- },
- {
- name: "DeploymentTimedOut: do not update status if deployment has existing timedOut condition",
- d: currentDeployment(&pds, 3, 2, 2, 2, []apps.DeploymentCondition{failedTimedOut}),
- allRSs: []*apps.ReplicaSet{},
- newRS: newRSWithAvailable("foo", 3, 2, 2),
- conditionType: apps.DeploymentProgressing,
- conditionStatus: v1.ConditionFalse,
- conditionReason: util.TimedOutReason,
- lastUpdate: testTime,
- lastTransition: testTime,
- },
- }
- for _, test := range tests {
- t.Run(test.name, func(t *testing.T) {
- fake := fake.Clientset{}
- dc := &DeploymentController{
- client: &fake,
- }
- if test.newRS != nil {
- test.allRSs = append(test.allRSs, test.newRS)
- }
- err := dc.syncRolloutStatus(test.allRSs, test.newRS, test.d)
- if err != nil {
- t.Error(err)
- }
- newCond := util.GetDeploymentCondition(test.d.Status, test.conditionType)
- switch {
- case newCond == nil:
- if test.d.Spec.ProgressDeadlineSeconds != nil && *test.d.Spec.ProgressDeadlineSeconds != math.MaxInt32 {
- t.Errorf("%s: expected deployment condition: %s", test.name, test.conditionType)
- }
- case newCond.Status != test.conditionStatus || newCond.Reason != test.conditionReason:
- t.Errorf("%s: DeploymentProgressing has status %s with reason %s. Expected %s with %s.", test.name, newCond.Status, newCond.Reason, test.conditionStatus, test.conditionReason)
- case !test.lastUpdate.IsZero() && test.lastUpdate != testTime:
- t.Errorf("%s: LastUpdateTime was changed to %s but expected %s;", test.name, test.lastUpdate, testTime)
- case !test.lastTransition.IsZero() && test.lastTransition != testTime:
- t.Errorf("%s: LastTransitionTime was changed to %s but expected %s;", test.name, test.lastTransition, testTime)
- }
- })
- }
- }
|