k3s/pkg/controller/petset/pet_set_test.go

265 lines
7.6 KiB
Go
Raw Normal View History

2016-04-25 19:24:40 +00:00
/*
Copyright 2016 The Kubernetes Authors All rights reserved.
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 petset
import (
"fmt"
"math/rand"
"reflect"
"testing"
"k8s.io/kubernetes/pkg/api"
"k8s.io/kubernetes/pkg/apis/apps"
"k8s.io/kubernetes/pkg/client/cache"
"k8s.io/kubernetes/pkg/controller"
)
func newFakePetSetController() (*PetSetController, *fakePetClient) {
fpc := newFakePetClient()
return &PetSetController{
kubeClient: nil,
blockingPetStore: newUnHealthyPetTracker(fpc),
podStoreSynced: func() bool { return true },
psStore: cache.StoreToPetSetLister{Store: cache.NewStore(controller.KeyFunc)},
podStore: cache.StoreToPodLister{Store: cache.NewStore(controller.KeyFunc)},
newSyncer: func(blockingPet *pcb) *petSyncer {
return &petSyncer{fpc, blockingPet}
},
}, fpc
}
func checkPets(ps *apps.PetSet, creates, deletes int, fc *fakePetClient, t *testing.T) {
if fc.petsCreated != creates || fc.petsDeleted != deletes {
t.Errorf("Found (creates: %d, deletes: %d), expected (creates: %d, deletes: %d)", fc.petsCreated, fc.petsDeleted, creates, deletes)
}
gotClaims := map[string]api.PersistentVolumeClaim{}
for _, pvc := range fc.claims {
gotClaims[pvc.Name] = pvc
}
for i := range fc.pets {
expectedPet, _ := newPCB(fmt.Sprintf("%v", i), ps)
if identityHash(ps, fc.pets[i].pod) != identityHash(ps, expectedPet.pod) {
t.Errorf("Unexpected pet at index %d", i)
}
for _, pvc := range expectedPet.pvcs {
gotPVC, ok := gotClaims[pvc.Name]
if !ok {
t.Errorf("PVC %v not created for pet %v", pvc.Name, expectedPet.pod.Name)
}
if !reflect.DeepEqual(gotPVC.Spec, pvc.Spec) {
t.Errorf("got PVC %v differs from created pvc", pvc.Name)
}
}
}
}
func scalePetSet(t *testing.T, ps *apps.PetSet, psc *PetSetController, fc *fakePetClient, scale int) []error {
errs := []error{}
for i := 0; i < scale; i++ {
pl := fc.getPodList()
if len(pl) != i {
t.Errorf("Unexpected number of pets, expected %d found %d", i, len(fc.pets))
}
_, syncErrs := psc.syncPetSet(ps, pl)
errs = append(errs, syncErrs...)
fc.setHealthy(i)
checkPets(ps, i+1, 0, fc, t)
}
return errs
}
func saturatePetSet(t *testing.T, ps *apps.PetSet, psc *PetSetController, fc *fakePetClient) {
errs := scalePetSet(t, ps, psc, fc, ps.Spec.Replicas)
if len(errs) != 0 {
t.Errorf("%v", errs)
}
}
func TestPetSetControllerCreates(t *testing.T) {
psc, fc := newFakePetSetController()
replicas := 3
ps := newPetSet(replicas)
saturatePetSet(t, ps, psc, fc)
podList := fc.getPodList()
// Deleted pet gets recreated
fc.pets = fc.pets[:replicas-1]
if _, errs := psc.syncPetSet(ps, podList); len(errs) != 0 {
t.Errorf("%v", errs)
}
checkPets(ps, replicas+1, 0, fc, t)
}
func TestPetSetControllerDeletes(t *testing.T) {
psc, fc := newFakePetSetController()
replicas := 4
ps := newPetSet(replicas)
saturatePetSet(t, ps, psc, fc)
// Drain
errs := []error{}
ps.Spec.Replicas = 0
knownPods := fc.getPodList()
for i := replicas - 1; i >= 0; i-- {
if len(fc.pets) != i+1 {
t.Errorf("Unexpected number of pets, expected %d found %d", i, len(fc.pets))
}
_, syncErrs := psc.syncPetSet(ps, knownPods)
errs = append(errs, syncErrs...)
}
if len(errs) != 0 {
t.Errorf("%v", errs)
}
checkPets(ps, replicas, replicas, fc, t)
}
func TestPetSetControllerRespectsTermination(t *testing.T) {
psc, fc := newFakePetSetController()
replicas := 4
ps := newPetSet(replicas)
saturatePetSet(t, ps, psc, fc)
fc.setDeletionTimestamp(replicas - 1)
ps.Spec.Replicas = 2
_, errs := psc.syncPetSet(ps, fc.getPodList())
if len(errs) != 0 {
t.Errorf("%v", errs)
}
// Finding a pod with the deletion timestamp will pause all deletions.
knownPods := fc.getPodList()
if len(knownPods) != 4 {
t.Errorf("Pods deleted prematurely before deletion timestamp expired, len %d", len(knownPods))
}
fc.pets = fc.pets[:replicas-1]
_, errs = psc.syncPetSet(ps, fc.getPodList())
if len(errs) != 0 {
t.Errorf("%v", errs)
}
checkPets(ps, replicas, 1, fc, t)
}
func TestPetSetControllerRespectsOrder(t *testing.T) {
psc, fc := newFakePetSetController()
replicas := 4
ps := newPetSet(replicas)
saturatePetSet(t, ps, psc, fc)
errs := []error{}
ps.Spec.Replicas = 0
// Shuffle known list and check that pets are deleted in reverse
knownPods := fc.getPodList()
for i := range knownPods {
j := rand.Intn(i + 1)
knownPods[i], knownPods[j] = knownPods[j], knownPods[i]
}
for i := 0; i < replicas; i++ {
if len(fc.pets) != replicas-i {
t.Errorf("Unexpected number of pets, expected %d found %d", i, len(fc.pets))
}
_, syncErrs := psc.syncPetSet(ps, knownPods)
errs = append(errs, syncErrs...)
checkPets(ps, replicas, i+1, fc, t)
}
if len(errs) != 0 {
t.Errorf("%v", errs)
}
}
func TestPetSetControllerBlocksScaling(t *testing.T) {
psc, fc := newFakePetSetController()
replicas := 5
ps := newPetSet(replicas)
scalePetSet(t, ps, psc, fc, 3)
// Create 4th pet, then before flipping it to healthy, kill the first pet.
// There should only be 1 not-healty pet at a time.
pl := fc.getPodList()
if _, errs := psc.syncPetSet(ps, pl); len(errs) != 0 {
t.Errorf("%v", errs)
}
deletedPod := pl[0]
fc.deletePetAtIndex(0)
pl = fc.getPodList()
if _, errs := psc.syncPetSet(ps, pl); len(errs) != 0 {
t.Errorf("%v", errs)
}
newPodList := fc.getPodList()
for _, p := range newPodList {
if p.Name == deletedPod.Name {
t.Errorf("Deleted pod was created while existing pod was unhealthy")
}
}
fc.setHealthy(len(newPodList) - 1)
if _, errs := psc.syncPetSet(ps, pl); len(errs) != 0 {
t.Errorf("%v", errs)
}
found := false
for _, p := range fc.getPodList() {
if p.Name == deletedPod.Name {
found = true
}
}
if !found {
t.Errorf("Deleted pod was not created after existing pods became healthy")
}
}
func TestPetSetBlockingPetIsCleared(t *testing.T) {
psc, fc := newFakePetSetController()
ps := newPetSet(3)
scalePetSet(t, ps, psc, fc, 1)
if blocking, err := psc.blockingPetStore.Get(ps, fc.getPodList()); err != nil || blocking != nil {
t.Errorf("Unexpected blocking pet %v, err %v", blocking, err)
}
// 1 not yet healthy pet
psc.syncPetSet(ps, fc.getPodList())
if blocking, err := psc.blockingPetStore.Get(ps, fc.getPodList()); err != nil || blocking == nil {
t.Errorf("Expected blocking pet %v, err %v", blocking, err)
}
// Deleting the petset should clear the blocking pet
if err := psc.psStore.Store.Delete(ps); err != nil {
t.Fatalf("Unable to delete pet %v from petset controller store.", ps.Name)
}
if errs := psc.Sync(fmt.Sprintf("%v/%v", ps.Namespace, ps.Name)); len(errs) != 0 {
t.Errorf("Error during sync of deleted petset %v", errs)
}
fc.pets = []*pcb{}
fc.petsCreated = 0
if blocking, err := psc.blockingPetStore.Get(ps, fc.getPodList()); err != nil || blocking != nil {
t.Errorf("Unexpected blocking pet %v, err %v", blocking, err)
}
saturatePetSet(t, ps, psc, fc)
// Make sure we don't leak the final blockin pet in the store
psc.syncPetSet(ps, fc.getPodList())
if p, exists, err := psc.blockingPetStore.store.GetByKey(fmt.Sprintf("%v/%v", ps.Namespace, ps.Name)); err != nil || exists {
t.Errorf("Unexpected blocking pet, err %v: %+v", err, p)
}
}