123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- /*
- 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 namespace
- import (
- "fmt"
- "net/http"
- "net/http/httptest"
- "path"
- "strings"
- "sync"
- "testing"
- "k8s.io/kubernetes/pkg/api"
- "k8s.io/kubernetes/pkg/api/errors"
- "k8s.io/kubernetes/pkg/api/unversioned"
- clientset "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
- "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset/fake"
- "k8s.io/kubernetes/pkg/client/restclient"
- "k8s.io/kubernetes/pkg/client/testing/core"
- "k8s.io/kubernetes/pkg/client/typed/dynamic"
- "k8s.io/kubernetes/pkg/runtime"
- "k8s.io/kubernetes/pkg/util/sets"
- )
- func TestFinalized(t *testing.T) {
- testNamespace := &api.Namespace{
- Spec: api.NamespaceSpec{
- Finalizers: []api.FinalizerName{"a", "b"},
- },
- }
- if finalized(testNamespace) {
- t.Errorf("Unexpected result, namespace is not finalized")
- }
- testNamespace.Spec.Finalizers = []api.FinalizerName{}
- if !finalized(testNamespace) {
- t.Errorf("Expected object to be finalized")
- }
- }
- func TestFinalizeNamespaceFunc(t *testing.T) {
- mockClient := &fake.Clientset{}
- testNamespace := &api.Namespace{
- ObjectMeta: api.ObjectMeta{
- Name: "test",
- ResourceVersion: "1",
- },
- Spec: api.NamespaceSpec{
- Finalizers: []api.FinalizerName{"kubernetes", "other"},
- },
- }
- finalizeNamespace(mockClient, testNamespace, api.FinalizerKubernetes)
- actions := mockClient.Actions()
- if len(actions) != 1 {
- t.Errorf("Expected 1 mock client action, but got %v", len(actions))
- }
- if !actions[0].Matches("create", "namespaces") || actions[0].GetSubresource() != "finalize" {
- t.Errorf("Expected finalize-namespace action %v", actions[0])
- }
- finalizers := actions[0].(core.CreateAction).GetObject().(*api.Namespace).Spec.Finalizers
- if len(finalizers) != 1 {
- t.Errorf("There should be a single finalizer remaining")
- }
- if "other" != string(finalizers[0]) {
- t.Errorf("Unexpected finalizer value, %v", finalizers[0])
- }
- }
- func testSyncNamespaceThatIsTerminating(t *testing.T, versions *unversioned.APIVersions) {
- now := unversioned.Now()
- namespaceName := "test"
- testNamespacePendingFinalize := &api.Namespace{
- ObjectMeta: api.ObjectMeta{
- Name: namespaceName,
- ResourceVersion: "1",
- DeletionTimestamp: &now,
- },
- Spec: api.NamespaceSpec{
- Finalizers: []api.FinalizerName{"kubernetes"},
- },
- Status: api.NamespaceStatus{
- Phase: api.NamespaceTerminating,
- },
- }
- testNamespaceFinalizeComplete := &api.Namespace{
- ObjectMeta: api.ObjectMeta{
- Name: namespaceName,
- ResourceVersion: "1",
- DeletionTimestamp: &now,
- },
- Spec: api.NamespaceSpec{},
- Status: api.NamespaceStatus{
- Phase: api.NamespaceTerminating,
- },
- }
- // when doing a delete all of content, we will do a GET of a collection, and DELETE of a collection by default
- dynamicClientActionSet := sets.NewString()
- groupVersionResources := testGroupVersionResources()
- for _, groupVersionResource := range groupVersionResources {
- urlPath := path.Join([]string{
- dynamic.LegacyAPIPathResolverFunc(groupVersionResource.GroupVersion()),
- groupVersionResource.Group,
- groupVersionResource.Version,
- "namespaces",
- namespaceName,
- groupVersionResource.Resource,
- }...)
- dynamicClientActionSet.Insert((&fakeAction{method: "GET", path: urlPath}).String())
- dynamicClientActionSet.Insert((&fakeAction{method: "DELETE", path: urlPath}).String())
- }
- scenarios := map[string]struct {
- testNamespace *api.Namespace
- kubeClientActionSet sets.String
- dynamicClientActionSet sets.String
- }{
- "pending-finalize": {
- testNamespace: testNamespacePendingFinalize,
- kubeClientActionSet: sets.NewString(
- strings.Join([]string{"get", "namespaces", ""}, "-"),
- strings.Join([]string{"create", "namespaces", "finalize"}, "-"),
- strings.Join([]string{"list", "pods", ""}, "-"),
- strings.Join([]string{"delete", "namespaces", ""}, "-"),
- ),
- dynamicClientActionSet: dynamicClientActionSet,
- },
- "complete-finalize": {
- testNamespace: testNamespaceFinalizeComplete,
- kubeClientActionSet: sets.NewString(
- strings.Join([]string{"get", "namespaces", ""}, "-"),
- strings.Join([]string{"delete", "namespaces", ""}, "-"),
- ),
- dynamicClientActionSet: sets.NewString(),
- },
- }
- for scenario, testInput := range scenarios {
- testHandler := &fakeActionHandler{statusCode: 200}
- srv, clientConfig := testServerAndClientConfig(testHandler.ServeHTTP)
- defer srv.Close()
- mockClient := fake.NewSimpleClientset(testInput.testNamespace)
- clientPool := dynamic.NewClientPool(clientConfig, dynamic.LegacyAPIPathResolverFunc)
- err := syncNamespace(mockClient, clientPool, operationNotSupportedCache{}, groupVersionResources, testInput.testNamespace, api.FinalizerKubernetes)
- if err != nil {
- t.Errorf("scenario %s - Unexpected error when synching namespace %v", scenario, err)
- }
- // validate traffic from kube client
- actionSet := sets.NewString()
- for _, action := range mockClient.Actions() {
- actionSet.Insert(strings.Join([]string{action.GetVerb(), action.GetResource().Resource, action.GetSubresource()}, "-"))
- }
- if !actionSet.Equal(testInput.kubeClientActionSet) {
- t.Errorf("scenario %s - mock client expected actions:\n%v\n but got:\n%v\nDifference:\n%v", scenario,
- testInput.kubeClientActionSet, actionSet, testInput.kubeClientActionSet.Difference(actionSet))
- }
- // validate traffic from dynamic client
- actionSet = sets.NewString()
- for _, action := range testHandler.actions {
- actionSet.Insert(action.String())
- }
- if !actionSet.Equal(testInput.dynamicClientActionSet) {
- t.Errorf("scenario %s - dynamic client expected actions:\n%v\n but got:\n%v\nDifference:\n%v", scenario,
- testInput.dynamicClientActionSet, actionSet, testInput.dynamicClientActionSet.Difference(actionSet))
- }
- }
- }
- func TestRetryOnConflictError(t *testing.T) {
- mockClient := &fake.Clientset{}
- numTries := 0
- retryOnce := func(kubeClient clientset.Interface, namespace *api.Namespace) (*api.Namespace, error) {
- numTries++
- if numTries <= 1 {
- return namespace, errors.NewConflict(api.Resource("namespaces"), namespace.Name, fmt.Errorf("ERROR!"))
- }
- return namespace, nil
- }
- namespace := &api.Namespace{}
- _, err := retryOnConflictError(mockClient, namespace, retryOnce)
- if err != nil {
- t.Errorf("Unexpected error %v", err)
- }
- if numTries != 2 {
- t.Errorf("Expected %v, but got %v", 2, numTries)
- }
- }
- func TestSyncNamespaceThatIsTerminatingNonExperimental(t *testing.T) {
- testSyncNamespaceThatIsTerminating(t, &unversioned.APIVersions{})
- }
- func TestSyncNamespaceThatIsTerminatingV1Beta1(t *testing.T) {
- testSyncNamespaceThatIsTerminating(t, &unversioned.APIVersions{Versions: []string{"extensions/v1beta1"}})
- }
- func TestSyncNamespaceThatIsActive(t *testing.T) {
- mockClient := &fake.Clientset{}
- testNamespace := &api.Namespace{
- ObjectMeta: api.ObjectMeta{
- Name: "test",
- ResourceVersion: "1",
- },
- Spec: api.NamespaceSpec{
- Finalizers: []api.FinalizerName{"kubernetes"},
- },
- Status: api.NamespaceStatus{
- Phase: api.NamespaceActive,
- },
- }
- err := syncNamespace(mockClient, nil, operationNotSupportedCache{}, testGroupVersionResources(), testNamespace, api.FinalizerKubernetes)
- if err != nil {
- t.Errorf("Unexpected error when synching namespace %v", err)
- }
- if len(mockClient.Actions()) != 0 {
- t.Errorf("Expected no action from controller, but got: %v", mockClient.Actions())
- }
- }
- // testServerAndClientConfig returns a server that listens and a config that can reference it
- func testServerAndClientConfig(handler func(http.ResponseWriter, *http.Request)) (*httptest.Server, *restclient.Config) {
- srv := httptest.NewServer(http.HandlerFunc(handler))
- config := &restclient.Config{
- Host: srv.URL,
- }
- return srv, config
- }
- // fakeAction records information about requests to aid in testing.
- type fakeAction struct {
- method string
- path string
- }
- // String returns method=path to aid in testing
- func (f *fakeAction) String() string {
- return strings.Join([]string{f.method, f.path}, "=")
- }
- // fakeActionHandler holds a list of fakeActions received
- type fakeActionHandler struct {
- // statusCode returned by this handler
- statusCode int
- lock sync.Mutex
- actions []fakeAction
- }
- // ServeHTTP logs the action that occurred and always returns the associated status code
- func (f *fakeActionHandler) ServeHTTP(response http.ResponseWriter, request *http.Request) {
- f.lock.Lock()
- defer f.lock.Unlock()
- f.actions = append(f.actions, fakeAction{method: request.Method, path: request.URL.Path})
- response.Header().Set("Content-Type", runtime.ContentTypeJSON)
- response.WriteHeader(f.statusCode)
- response.Write([]byte("{\"kind\": \"List\",\"items\":null}"))
- }
- // testGroupVersionResources returns a mocked up set of resources across different api groups for testing namespace controller.
- func testGroupVersionResources() []unversioned.GroupVersionResource {
- results := []unversioned.GroupVersionResource{}
- results = append(results, unversioned.GroupVersionResource{Group: "", Version: "v1", Resource: "pods"})
- results = append(results, unversioned.GroupVersionResource{Group: "", Version: "v1", Resource: "services"})
- results = append(results, unversioned.GroupVersionResource{Group: "extensions", Version: "v1beta1", Resource: "deployments"})
- return results
- }
|