mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-26 02:55:32 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			708 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			708 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2014 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 kubectl
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"k8s.io/apimachinery/pkg/api/errors"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/runtime"
 | |
| 	"k8s.io/apimachinery/pkg/runtime/schema"
 | |
| 	"k8s.io/apimachinery/pkg/util/uuid"
 | |
| 	"k8s.io/apimachinery/pkg/watch"
 | |
| 	testcore "k8s.io/client-go/testing"
 | |
| 	"k8s.io/kubernetes/pkg/api"
 | |
| 	"k8s.io/kubernetes/pkg/apis/batch"
 | |
| 	"k8s.io/kubernetes/pkg/apis/extensions"
 | |
| 	"k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset/fake"
 | |
| 	coreclient "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset/typed/core/internalversion"
 | |
| )
 | |
| 
 | |
| func TestReplicationControllerStop(t *testing.T) {
 | |
| 	name := "foo"
 | |
| 	ns := "default"
 | |
| 	tests := []struct {
 | |
| 		Name            string
 | |
| 		Objs            []runtime.Object
 | |
| 		StopError       error
 | |
| 		ExpectedActions []string
 | |
| 	}{
 | |
| 		{
 | |
| 			Name: "OnlyOneRC",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&api.ReplicationControllerList{ // LIST
 | |
| 					Items: []api.ReplicationController{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError:       nil,
 | |
| 			ExpectedActions: []string{"get", "list", "get", "update", "get", "delete"},
 | |
| 		},
 | |
| 		{
 | |
| 			Name: "NoOverlapping",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&api.ReplicationControllerList{ // LIST
 | |
| 					Items: []api.ReplicationController{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "baz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k3": "v3"}},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError:       nil,
 | |
| 			ExpectedActions: []string{"get", "list", "get", "update", "get", "delete"},
 | |
| 		},
 | |
| 		{
 | |
| 			Name: "OverlappingError",
 | |
| 			Objs: []runtime.Object{
 | |
| 
 | |
| 				&api.ReplicationControllerList{ // LIST
 | |
| 					Items: []api.ReplicationController{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "baz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1", "k2": "v2"}},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError:       fmt.Errorf("Detected overlapping controllers for rc foo: baz, please manage deletion individually with --cascade=false."),
 | |
| 			ExpectedActions: []string{"get", "list"},
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			Name: "OverlappingButSafeDelete",
 | |
| 			Objs: []runtime.Object{
 | |
| 
 | |
| 				&api.ReplicationControllerList{ // LIST
 | |
| 					Items: []api.ReplicationController{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "baz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1", "k2": "v2", "k3": "v3"}},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "zaz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1", "k2": "v2"}},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 
 | |
| 			StopError:       fmt.Errorf("Detected overlapping controllers for rc foo: baz,zaz, please manage deletion individually with --cascade=false."),
 | |
| 			ExpectedActions: []string{"get", "list"},
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			Name: "TwoExactMatchRCs",
 | |
| 			Objs: []runtime.Object{
 | |
| 
 | |
| 				&api.ReplicationControllerList{ // LIST
 | |
| 					Items: []api.ReplicationController{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "zaz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: api.ReplicationControllerSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: map[string]string{"k1": "v1"}},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 
 | |
| 			StopError:       nil,
 | |
| 			ExpectedActions: []string{"get", "list", "delete"},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		copiedForWatch, err := api.Scheme.Copy(test.Objs[0])
 | |
| 		if err != nil {
 | |
| 			t.Fatalf("%s unexpected error: %v", test.Name, err)
 | |
| 		}
 | |
| 		fake := fake.NewSimpleClientset(test.Objs...)
 | |
| 		fakeWatch := watch.NewFake()
 | |
| 		fake.PrependWatchReactor("replicationcontrollers", testcore.DefaultWatchReactor(fakeWatch, nil))
 | |
| 
 | |
| 		go func() {
 | |
| 			fakeWatch.Add(copiedForWatch)
 | |
| 		}()
 | |
| 
 | |
| 		reaper := ReplicationControllerReaper{fake.Core(), time.Millisecond, time.Millisecond}
 | |
| 		err = reaper.Stop(ns, name, 0, nil)
 | |
| 		if !reflect.DeepEqual(err, test.StopError) {
 | |
| 			t.Errorf("%s unexpected error: %v", test.Name, err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		actions := fake.Actions()
 | |
| 		if len(actions) != len(test.ExpectedActions) {
 | |
| 			t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
 | |
| 			continue
 | |
| 		}
 | |
| 		for i, verb := range test.ExpectedActions {
 | |
| 			if actions[i].GetResource().GroupResource() != api.Resource("replicationcontrollers") {
 | |
| 				t.Errorf("%s unexpected action: %+v, expected %s-replicationController", test.Name, actions[i], verb)
 | |
| 			}
 | |
| 			if actions[i].GetVerb() != verb {
 | |
| 				t.Errorf("%s unexpected action: %+v, expected %s-replicationController", test.Name, actions[i], verb)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestReplicaSetStop(t *testing.T) {
 | |
| 	name := "foo"
 | |
| 	ns := "default"
 | |
| 	tests := []struct {
 | |
| 		Name            string
 | |
| 		Objs            []runtime.Object
 | |
| 		StopError       error
 | |
| 		ExpectedActions []string
 | |
| 	}{
 | |
| 		{
 | |
| 			Name: "OnlyOneRS",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&extensions.ReplicaSetList{ // LIST
 | |
| 					Items: []extensions.ReplicaSet{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: extensions.ReplicaSetSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k1": "v1"}},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError:       nil,
 | |
| 			ExpectedActions: []string{"get", "get", "update", "get", "get", "delete"},
 | |
| 		},
 | |
| 		{
 | |
| 			Name: "NoOverlapping",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&extensions.ReplicaSetList{ // LIST
 | |
| 					Items: []extensions.ReplicaSet{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "baz",
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: extensions.ReplicaSetSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k3": "v3"}},
 | |
| 							},
 | |
| 						},
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: extensions.ReplicaSetSpec{
 | |
| 								Replicas: 0,
 | |
| 								Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k1": "v1"}},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError:       nil,
 | |
| 			ExpectedActions: []string{"get", "get", "update", "get", "get", "delete"},
 | |
| 		},
 | |
| 		// TODO: Implement tests for overlapping replica sets, similar to replication controllers,
 | |
| 		// when the overlapping checks are implemented for replica sets.
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		fake := fake.NewSimpleClientset(test.Objs...)
 | |
| 		reaper := ReplicaSetReaper{fake.Extensions(), time.Millisecond, time.Millisecond}
 | |
| 		err := reaper.Stop(ns, name, 0, nil)
 | |
| 		if !reflect.DeepEqual(err, test.StopError) {
 | |
| 			t.Errorf("%s unexpected error: %v", test.Name, err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		actions := fake.Actions()
 | |
| 		if len(actions) != len(test.ExpectedActions) {
 | |
| 			t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
 | |
| 			continue
 | |
| 		}
 | |
| 		for i, verb := range test.ExpectedActions {
 | |
| 			if actions[i].GetResource().GroupResource() != extensions.Resource("replicasets") {
 | |
| 				t.Errorf("%s unexpected action: %+v, expected %s-replicaSet", test.Name, actions[i], verb)
 | |
| 			}
 | |
| 			if actions[i].GetVerb() != verb {
 | |
| 				t.Errorf("%s unexpected action: %+v, expected %s-replicaSet", test.Name, actions[i], verb)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestJobStop(t *testing.T) {
 | |
| 	name := "foo"
 | |
| 	ns := "default"
 | |
| 	zero := int32(0)
 | |
| 	tests := []struct {
 | |
| 		Name            string
 | |
| 		Objs            []runtime.Object
 | |
| 		StopError       error
 | |
| 		ExpectedActions []string
 | |
| 	}{
 | |
| 		{
 | |
| 			Name: "OnlyOneJob",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&batch.JobList{ // LIST
 | |
| 					Items: []batch.Job{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: batch.JobSpec{
 | |
| 								Parallelism: &zero,
 | |
| 								Selector: &metav1.LabelSelector{
 | |
| 									MatchLabels: map[string]string{"k1": "v1"},
 | |
| 								},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError: nil,
 | |
| 			ExpectedActions: []string{"get:jobs", "get:jobs", "update:jobs",
 | |
| 				"get:jobs", "get:jobs", "list:pods", "delete:jobs"},
 | |
| 		},
 | |
| 		{
 | |
| 			Name: "JobWithDeadPods",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&batch.JobList{ // LIST
 | |
| 					Items: []batch.Job{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 							},
 | |
| 							Spec: batch.JobSpec{
 | |
| 								Parallelism: &zero,
 | |
| 								Selector: &metav1.LabelSelector{
 | |
| 									MatchLabels: map[string]string{"k1": "v1"},
 | |
| 								},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 				&api.PodList{ // LIST
 | |
| 					Items: []api.Pod{
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "pod1",
 | |
| 								Namespace: ns,
 | |
| 								Labels:    map[string]string{"k1": "v1"},
 | |
| 							},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError: nil,
 | |
| 			ExpectedActions: []string{"get:jobs", "get:jobs", "update:jobs",
 | |
| 				"get:jobs", "get:jobs", "list:pods", "delete:pods", "delete:jobs"},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		fake := fake.NewSimpleClientset(test.Objs...)
 | |
| 		reaper := JobReaper{fake.Batch(), fake.Core(), time.Millisecond, time.Millisecond}
 | |
| 		err := reaper.Stop(ns, name, 0, nil)
 | |
| 		if !reflect.DeepEqual(err, test.StopError) {
 | |
| 			t.Errorf("%s unexpected error: %v", test.Name, err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		actions := fake.Actions()
 | |
| 		if len(actions) != len(test.ExpectedActions) {
 | |
| 			t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
 | |
| 			continue
 | |
| 		}
 | |
| 		for i, expAction := range test.ExpectedActions {
 | |
| 			action := strings.Split(expAction, ":")
 | |
| 			if actions[i].GetVerb() != action[0] {
 | |
| 				t.Errorf("%s unexpected verb: %+v, expected %s", test.Name, actions[i], expAction)
 | |
| 			}
 | |
| 			if actions[i].GetResource().Resource != action[1] {
 | |
| 				t.Errorf("%s unexpected resource: %+v, expected %s", test.Name, actions[i], expAction)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDeploymentStop(t *testing.T) {
 | |
| 	name := "foo"
 | |
| 	ns := "default"
 | |
| 	deployment := extensions.Deployment{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Name:      name,
 | |
| 			UID:       uuid.NewUUID(),
 | |
| 			Namespace: ns,
 | |
| 		},
 | |
| 		Spec: extensions.DeploymentSpec{
 | |
| 			Replicas: 0,
 | |
| 			Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k1": "v1"}},
 | |
| 		},
 | |
| 		Status: extensions.DeploymentStatus{
 | |
| 			Replicas: 0,
 | |
| 		},
 | |
| 	}
 | |
| 	trueVar := true
 | |
| 	tests := []struct {
 | |
| 		Name            string
 | |
| 		Objs            []runtime.Object
 | |
| 		StopError       error
 | |
| 		ExpectedActions []string
 | |
| 	}{
 | |
| 		{
 | |
| 			Name: "SimpleDeployment",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&extensions.Deployment{ // GET
 | |
| 					ObjectMeta: metav1.ObjectMeta{
 | |
| 						Name:      name,
 | |
| 						Namespace: ns,
 | |
| 					},
 | |
| 					Spec: extensions.DeploymentSpec{
 | |
| 						Replicas: 0,
 | |
| 						Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k1": "v1"}},
 | |
| 					},
 | |
| 					Status: extensions.DeploymentStatus{
 | |
| 						Replicas: 0,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError: nil,
 | |
| 			ExpectedActions: []string{"get:deployments", "update:deployments",
 | |
| 				"get:deployments", "list:replicasets", "delete:deployments"},
 | |
| 		},
 | |
| 		{
 | |
| 			Name: "Deployment with single replicaset",
 | |
| 			Objs: []runtime.Object{
 | |
| 				&deployment, // GET
 | |
| 				&extensions.ReplicaSetList{ // LIST
 | |
| 					Items: []extensions.ReplicaSet{
 | |
| 						// ReplicaSet owned by this Deployment.
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      name,
 | |
| 								Namespace: ns,
 | |
| 								Labels:    map[string]string{"k1": "v1"},
 | |
| 								OwnerReferences: []metav1.OwnerReference{
 | |
| 									{
 | |
| 										APIVersion: extensions.SchemeGroupVersion.String(),
 | |
| 										Kind:       "Deployment",
 | |
| 										Name:       deployment.Name,
 | |
| 										UID:        deployment.UID,
 | |
| 										Controller: &trueVar,
 | |
| 									},
 | |
| 								},
 | |
| 							},
 | |
| 							Spec: extensions.ReplicaSetSpec{},
 | |
| 						},
 | |
| 						// ReplicaSet owned by something else (should be ignored).
 | |
| 						{
 | |
| 							ObjectMeta: metav1.ObjectMeta{
 | |
| 								Name:      "rs2",
 | |
| 								Namespace: ns,
 | |
| 								Labels:    map[string]string{"k1": "v1"},
 | |
| 								OwnerReferences: []metav1.OwnerReference{
 | |
| 									{
 | |
| 										APIVersion: extensions.SchemeGroupVersion.String(),
 | |
| 										Kind:       "Deployment",
 | |
| 										Name:       "somethingelse",
 | |
| 										UID:        uuid.NewUUID(),
 | |
| 										Controller: &trueVar,
 | |
| 									},
 | |
| 								},
 | |
| 							},
 | |
| 							Spec: extensions.ReplicaSetSpec{},
 | |
| 						},
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			StopError: nil,
 | |
| 			ExpectedActions: []string{"get:deployments", "update:deployments",
 | |
| 				"get:deployments", "list:replicasets", "get:replicasets",
 | |
| 				"get:replicasets", "update:replicasets", "get:replicasets",
 | |
| 				"get:replicasets", "delete:replicasets", "delete:deployments"},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		fake := fake.NewSimpleClientset(test.Objs...)
 | |
| 		reaper := DeploymentReaper{fake.Extensions(), fake.Extensions(), time.Millisecond, time.Millisecond}
 | |
| 		err := reaper.Stop(ns, name, 0, nil)
 | |
| 		if !reflect.DeepEqual(err, test.StopError) {
 | |
| 			t.Errorf("%s unexpected error: %v", test.Name, err)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		actions := fake.Actions()
 | |
| 		if len(actions) != len(test.ExpectedActions) {
 | |
| 			t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
 | |
| 			continue
 | |
| 		}
 | |
| 		for i, expAction := range test.ExpectedActions {
 | |
| 			action := strings.Split(expAction, ":")
 | |
| 			if actions[i].GetVerb() != action[0] {
 | |
| 				t.Errorf("%s unexpected verb: %+v, expected %s", test.Name, actions[i], expAction)
 | |
| 			}
 | |
| 			if actions[i].GetResource().Resource != action[1] {
 | |
| 				t.Errorf("%s unexpected resource: %+v, expected %s", test.Name, actions[i], expAction)
 | |
| 			}
 | |
| 			if len(action) == 3 && actions[i].GetSubresource() != action[2] {
 | |
| 				t.Errorf("%s unexpected subresource: %+v, expected %s", test.Name, actions[i], expAction)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type noSuchPod struct {
 | |
| 	coreclient.PodInterface
 | |
| }
 | |
| 
 | |
| func (c *noSuchPod) Get(name string, options metav1.GetOptions) (*api.Pod, error) {
 | |
| 	return nil, fmt.Errorf("%s does not exist", name)
 | |
| }
 | |
| 
 | |
| type noDeletePod struct {
 | |
| 	coreclient.PodInterface
 | |
| }
 | |
| 
 | |
| func (c *noDeletePod) Delete(name string, o *metav1.DeleteOptions) error {
 | |
| 	return fmt.Errorf("I'm afraid I can't do that, Dave")
 | |
| }
 | |
| 
 | |
| type reaperFake struct {
 | |
| 	*fake.Clientset
 | |
| 	noSuchPod, noDeletePod bool
 | |
| }
 | |
| 
 | |
| func (c *reaperFake) Core() coreclient.CoreInterface {
 | |
| 	return &reaperCoreFake{c.Clientset.Core(), c.noSuchPod, c.noDeletePod}
 | |
| }
 | |
| 
 | |
| type reaperCoreFake struct {
 | |
| 	coreclient.CoreInterface
 | |
| 	noSuchPod, noDeletePod bool
 | |
| }
 | |
| 
 | |
| func (c *reaperCoreFake) Pods(namespace string) coreclient.PodInterface {
 | |
| 	pods := c.CoreInterface.Pods(namespace)
 | |
| 	if c.noSuchPod {
 | |
| 		return &noSuchPod{pods}
 | |
| 	}
 | |
| 	if c.noDeletePod {
 | |
| 		return &noDeletePod{pods}
 | |
| 	}
 | |
| 	return pods
 | |
| }
 | |
| 
 | |
| func pod() *api.Pod {
 | |
| 	return &api.Pod{ObjectMeta: metav1.ObjectMeta{Namespace: metav1.NamespaceDefault, Name: "foo"}}
 | |
| }
 | |
| 
 | |
| func TestSimpleStop(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		fake        *reaperFake
 | |
| 		kind        schema.GroupKind
 | |
| 		actions     []testcore.Action
 | |
| 		expectError bool
 | |
| 		test        string
 | |
| 	}{
 | |
| 		{
 | |
| 			fake: &reaperFake{
 | |
| 				Clientset: fake.NewSimpleClientset(pod()),
 | |
| 			},
 | |
| 			kind: api.Kind("Pod"),
 | |
| 			actions: []testcore.Action{
 | |
| 				testcore.NewGetAction(api.Resource("pods").WithVersion(""), metav1.NamespaceDefault, "foo"),
 | |
| 				testcore.NewDeleteAction(api.Resource("pods").WithVersion(""), metav1.NamespaceDefault, "foo"),
 | |
| 			},
 | |
| 			expectError: false,
 | |
| 			test:        "stop pod succeeds",
 | |
| 		},
 | |
| 		{
 | |
| 			fake: &reaperFake{
 | |
| 				Clientset: fake.NewSimpleClientset(),
 | |
| 				noSuchPod: true,
 | |
| 			},
 | |
| 			kind:        api.Kind("Pod"),
 | |
| 			actions:     []testcore.Action{},
 | |
| 			expectError: true,
 | |
| 			test:        "stop pod fails, no pod",
 | |
| 		},
 | |
| 		{
 | |
| 			fake: &reaperFake{
 | |
| 				Clientset:   fake.NewSimpleClientset(pod()),
 | |
| 				noDeletePod: true,
 | |
| 			},
 | |
| 			kind: api.Kind("Pod"),
 | |
| 			actions: []testcore.Action{
 | |
| 				testcore.NewGetAction(api.Resource("pods").WithVersion(""), metav1.NamespaceDefault, "foo"),
 | |
| 			},
 | |
| 			expectError: true,
 | |
| 			test:        "stop pod fails, can't delete",
 | |
| 		},
 | |
| 	}
 | |
| 	for _, test := range tests {
 | |
| 		fake := test.fake
 | |
| 		reaper, err := ReaperFor(test.kind, fake)
 | |
| 		if err != nil {
 | |
| 			t.Errorf("unexpected error: %v (%s)", err, test.test)
 | |
| 		}
 | |
| 		err = reaper.Stop("default", "foo", 0, nil)
 | |
| 		if err != nil && !test.expectError {
 | |
| 			t.Errorf("unexpected error: %v (%s)", err, test.test)
 | |
| 		}
 | |
| 		if err == nil {
 | |
| 			if test.expectError {
 | |
| 				t.Errorf("unexpected non-error: %v (%s)", err, test.test)
 | |
| 			}
 | |
| 		}
 | |
| 		actions := fake.Actions()
 | |
| 		if len(test.actions) != len(actions) {
 | |
| 			t.Errorf("unexpected actions: %v; expected %v (%s)", actions, test.actions, test.test)
 | |
| 		}
 | |
| 		for i, action := range actions {
 | |
| 			testAction := test.actions[i]
 | |
| 			if action != testAction {
 | |
| 				t.Errorf("unexpected action: %#v; expected %v (%s)", action, testAction, test.test)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDeploymentNotFoundError(t *testing.T) {
 | |
| 	name := "foo"
 | |
| 	ns := "default"
 | |
| 	deployment := &extensions.Deployment{
 | |
| 		ObjectMeta: metav1.ObjectMeta{
 | |
| 			Name:      name,
 | |
| 			Namespace: ns,
 | |
| 		},
 | |
| 		Spec: extensions.DeploymentSpec{
 | |
| 			Replicas: 0,
 | |
| 			Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"k1": "v1"}},
 | |
| 		},
 | |
| 		Status: extensions.DeploymentStatus{
 | |
| 			Replicas: 0,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	fake := fake.NewSimpleClientset(
 | |
| 		deployment,
 | |
| 		&extensions.ReplicaSetList{Items: []extensions.ReplicaSet{
 | |
| 			{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Name:      name,
 | |
| 					Namespace: ns,
 | |
| 				},
 | |
| 				Spec: extensions.ReplicaSetSpec{},
 | |
| 			},
 | |
| 		},
 | |
| 		},
 | |
| 	)
 | |
| 	fake.AddReactor("get", "replicasets", func(action testcore.Action) (handled bool, ret runtime.Object, err error) {
 | |
| 		return true, nil, ScaleError{ActualError: errors.NewNotFound(api.Resource("replicaset"), "doesn't-matter")}
 | |
| 	})
 | |
| 
 | |
| 	reaper := DeploymentReaper{fake.Extensions(), fake.Extensions(), time.Millisecond, time.Millisecond}
 | |
| 	if err := reaper.Stop(ns, name, 0, nil); err != nil {
 | |
| 		t.Fatalf("unexpected error: %#v", err)
 | |
| 	}
 | |
| }
 |