mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-25 18:09:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			643 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			643 lines
		
	
	
		
			18 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 cmd
 | |
| 
 | |
| import (
 | |
| 	"net/http"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"k8s.io/api/core/v1"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/runtime"
 | |
| 	"k8s.io/apimachinery/pkg/runtime/schema"
 | |
| 	"k8s.io/cli-runtime/pkg/genericclioptions"
 | |
| 	"k8s.io/client-go/rest/fake"
 | |
| 	cmdtesting "k8s.io/kubernetes/pkg/kubectl/cmd/testing"
 | |
| 	"k8s.io/kubernetes/pkg/kubectl/scheme"
 | |
| )
 | |
| 
 | |
| func TestValidateAnnotationOverwrites(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		meta        *metav1.ObjectMeta
 | |
| 		annotations map[string]string
 | |
| 		expectErr   bool
 | |
| 		scenario    string
 | |
| 	}{
 | |
| 		{
 | |
| 			meta: &metav1.ObjectMeta{
 | |
| 				Annotations: map[string]string{
 | |
| 					"a": "A",
 | |
| 					"b": "B",
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{
 | |
| 				"a": "a",
 | |
| 				"c": "C",
 | |
| 			},
 | |
| 			scenario:  "share first annotation",
 | |
| 			expectErr: true,
 | |
| 		},
 | |
| 		{
 | |
| 			meta: &metav1.ObjectMeta{
 | |
| 				Annotations: map[string]string{
 | |
| 					"a": "A",
 | |
| 					"c": "C",
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{
 | |
| 				"b": "B",
 | |
| 				"c": "c",
 | |
| 			},
 | |
| 			scenario:  "share second annotation",
 | |
| 			expectErr: true,
 | |
| 		},
 | |
| 		{
 | |
| 			meta: &metav1.ObjectMeta{
 | |
| 				Annotations: map[string]string{
 | |
| 					"a": "A",
 | |
| 					"c": "C",
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{
 | |
| 				"b": "B",
 | |
| 				"d": "D",
 | |
| 			},
 | |
| 			scenario: "no overlap",
 | |
| 		},
 | |
| 		{
 | |
| 			meta: &metav1.ObjectMeta{},
 | |
| 			annotations: map[string]string{
 | |
| 				"a": "A",
 | |
| 				"b": "B",
 | |
| 			},
 | |
| 			scenario: "no annotations",
 | |
| 		},
 | |
| 	}
 | |
| 	for _, test := range tests {
 | |
| 		err := validateNoAnnotationOverwrites(test.meta, test.annotations)
 | |
| 		if test.expectErr && err == nil {
 | |
| 			t.Errorf("%s: unexpected non-error", test.scenario)
 | |
| 		} else if !test.expectErr && err != nil {
 | |
| 			t.Errorf("%s: unexpected error: %v", test.scenario, err)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestParseAnnotations(t *testing.T) {
 | |
| 	testURL := "https://test.com/index.htm?id=123#u=user-name"
 | |
| 	testJSON := `'{"kind":"SerializedReference","apiVersion":"v1","reference":{"kind":"ReplicationController","namespace":"default","name":"my-nginx","uid":"c544ee78-2665-11e5-8051-42010af0c213","apiVersion":"v1","resourceVersion":"61368"}}'`
 | |
| 	tests := []struct {
 | |
| 		annotations    []string
 | |
| 		expected       map[string]string
 | |
| 		expectedRemove []string
 | |
| 		scenario       string
 | |
| 		expectedErr    string
 | |
| 		expectErr      bool
 | |
| 	}{
 | |
| 		{
 | |
| 			annotations:    []string{"a=b", "c=d"},
 | |
| 			expected:       map[string]string{"a": "b", "c": "d"},
 | |
| 			expectedRemove: []string{},
 | |
| 			scenario:       "add two annotations",
 | |
| 			expectErr:      false,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations:    []string{"url=" + testURL, "fake.kubernetes.io/annotation=" + testJSON},
 | |
| 			expected:       map[string]string{"url": testURL, "fake.kubernetes.io/annotation": testJSON},
 | |
| 			expectedRemove: []string{},
 | |
| 			scenario:       "add annotations with special characters",
 | |
| 			expectErr:      false,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations:    []string{},
 | |
| 			expected:       map[string]string{},
 | |
| 			expectedRemove: []string{},
 | |
| 			scenario:       "add no annotations",
 | |
| 			expectErr:      false,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations:    []string{"a=b", "c=d", "e-"},
 | |
| 			expected:       map[string]string{"a": "b", "c": "d"},
 | |
| 			expectedRemove: []string{"e"},
 | |
| 			scenario:       "add two annotations, remove one",
 | |
| 			expectErr:      false,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations: []string{"ab", "c=d"},
 | |
| 			expectedErr: "invalid annotation format: ab",
 | |
| 			scenario:    "incorrect annotation input (missing =value)",
 | |
| 			expectErr:   true,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations:    []string{"a="},
 | |
| 			expected:       map[string]string{"a": ""},
 | |
| 			expectedRemove: []string{},
 | |
| 			scenario:       "add valid annotation with empty value",
 | |
| 			expectErr:      false,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations: []string{"ab", "a="},
 | |
| 			expectedErr: "invalid annotation format: ab",
 | |
| 			scenario:    "incorrect annotation input (missing =value)",
 | |
| 			expectErr:   true,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations: []string{"-"},
 | |
| 			expectedErr: "invalid annotation format: -",
 | |
| 			scenario:    "incorrect annotation input (missing key)",
 | |
| 			expectErr:   true,
 | |
| 		},
 | |
| 		{
 | |
| 			annotations: []string{"=bar"},
 | |
| 			expectedErr: "invalid annotation format: =bar",
 | |
| 			scenario:    "incorrect annotation input (missing key)",
 | |
| 			expectErr:   true,
 | |
| 		},
 | |
| 	}
 | |
| 	for _, test := range tests {
 | |
| 		annotations, remove, err := parseAnnotations(test.annotations)
 | |
| 		switch {
 | |
| 		case test.expectErr && err == nil:
 | |
| 			t.Errorf("%s: unexpected non-error, should return %v", test.scenario, test.expectedErr)
 | |
| 		case test.expectErr && err.Error() != test.expectedErr:
 | |
| 			t.Errorf("%s: unexpected error %v, expected %v", test.scenario, err, test.expectedErr)
 | |
| 		case !test.expectErr && err != nil:
 | |
| 			t.Errorf("%s: unexpected error %v", test.scenario, err)
 | |
| 		case !test.expectErr && !reflect.DeepEqual(annotations, test.expected):
 | |
| 			t.Errorf("%s: expected %v, got %v", test.scenario, test.expected, annotations)
 | |
| 		case !test.expectErr && !reflect.DeepEqual(remove, test.expectedRemove):
 | |
| 			t.Errorf("%s: expected %v, got %v", test.scenario, test.expectedRemove, remove)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestValidateAnnotations(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		removeAnnotations []string
 | |
| 		newAnnotations    map[string]string
 | |
| 		expectedErr       string
 | |
| 		scenario          string
 | |
| 	}{
 | |
| 		{
 | |
| 			expectedErr:       "can not both modify and remove the following annotation(s) in the same command: a",
 | |
| 			removeAnnotations: []string{"a"},
 | |
| 			newAnnotations:    map[string]string{"a": "b", "c": "d"},
 | |
| 			scenario:          "remove an added annotation",
 | |
| 		},
 | |
| 		{
 | |
| 			expectedErr:       "can not both modify and remove the following annotation(s) in the same command: a, c",
 | |
| 			removeAnnotations: []string{"a", "c"},
 | |
| 			newAnnotations:    map[string]string{"a": "b", "c": "d"},
 | |
| 			scenario:          "remove added annotations",
 | |
| 		},
 | |
| 	}
 | |
| 	for _, test := range tests {
 | |
| 		if err := validateAnnotations(test.removeAnnotations, test.newAnnotations); err == nil {
 | |
| 			t.Errorf("%s: unexpected non-error", test.scenario)
 | |
| 		} else if err.Error() != test.expectedErr {
 | |
| 			t.Errorf("%s: expected error %s, got %s", test.scenario, test.expectedErr, err.Error())
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestUpdateAnnotations(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		obj         runtime.Object
 | |
| 		overwrite   bool
 | |
| 		version     string
 | |
| 		annotations map[string]string
 | |
| 		remove      []string
 | |
| 		expected    runtime.Object
 | |
| 		expectErr   bool
 | |
| 	}{
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"a": "b"},
 | |
| 			expectErr:   true,
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"a": "c"},
 | |
| 			overwrite:   true,
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "c"},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"c": "d"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b", "c": "d"},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"c": "d"},
 | |
| 			version:     "2",
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations:     map[string]string{"a": "b", "c": "d"},
 | |
| 					ResourceVersion: "2",
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{},
 | |
| 			remove:      []string{"a"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b", "c": "d"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"e": "f"},
 | |
| 			remove:      []string{"a"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{
 | |
| 						"c": "d",
 | |
| 						"e": "f",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b", "c": "d"},
 | |
| 				},
 | |
| 			},
 | |
| 			annotations: map[string]string{"e": "f"},
 | |
| 			remove:      []string{"g"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{
 | |
| 						"a": "b",
 | |
| 						"c": "d",
 | |
| 						"e": "f",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b", "c": "d"},
 | |
| 				},
 | |
| 			},
 | |
| 			remove: []string{"e"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{
 | |
| 						"a": "b",
 | |
| 						"c": "d",
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			obj: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{},
 | |
| 			},
 | |
| 			annotations: map[string]string{"a": "b"},
 | |
| 			expected: &v1.Pod{
 | |
| 				ObjectMeta: metav1.ObjectMeta{
 | |
| 					Annotations: map[string]string{"a": "b"},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 	for _, test := range tests {
 | |
| 		options := &AnnotateOptions{
 | |
| 			overwrite:         test.overwrite,
 | |
| 			newAnnotations:    test.annotations,
 | |
| 			removeAnnotations: test.remove,
 | |
| 			resourceVersion:   test.version,
 | |
| 		}
 | |
| 		err := options.updateAnnotations(test.obj)
 | |
| 		if test.expectErr {
 | |
| 			if err == nil {
 | |
| 				t.Errorf("unexpected non-error: %v", test)
 | |
| 			}
 | |
| 			continue
 | |
| 		}
 | |
| 		if !test.expectErr && err != nil {
 | |
| 			t.Errorf("unexpected error: %v %v", err, test)
 | |
| 		}
 | |
| 		if !reflect.DeepEqual(test.obj, test.expected) {
 | |
| 			t.Errorf("expected: %v, got %v", test.expected, test.obj)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAnnotateErrors(t *testing.T) {
 | |
| 	testCases := map[string]struct {
 | |
| 		args  []string
 | |
| 		flags map[string]string
 | |
| 		errFn func(error) bool
 | |
| 	}{
 | |
| 		"no args": {
 | |
| 			args:  []string{},
 | |
| 			errFn: func(err error) bool { return strings.Contains(err.Error(), "one or more resources must be specified") },
 | |
| 		},
 | |
| 		"not enough annotations": {
 | |
| 			args: []string{"pods"},
 | |
| 			errFn: func(err error) bool {
 | |
| 				return strings.Contains(err.Error(), "at least one annotation update is required")
 | |
| 			},
 | |
| 		},
 | |
| 		"wrong annotations": {
 | |
| 			args: []string{"pods", "-"},
 | |
| 			errFn: func(err error) bool {
 | |
| 				return strings.Contains(err.Error(), "at least one annotation update is required")
 | |
| 			},
 | |
| 		},
 | |
| 		"wrong annotations 2": {
 | |
| 			args: []string{"pods", "=bar"},
 | |
| 			errFn: func(err error) bool {
 | |
| 				return strings.Contains(err.Error(), "at least one annotation update is required")
 | |
| 			},
 | |
| 		},
 | |
| 		"no resources remove annotations": {
 | |
| 			args:  []string{"pods-"},
 | |
| 			errFn: func(err error) bool { return strings.Contains(err.Error(), "one or more resources must be specified") },
 | |
| 		},
 | |
| 		"no resources add annotations": {
 | |
| 			args:  []string{"pods=bar"},
 | |
| 			errFn: func(err error) bool { return strings.Contains(err.Error(), "one or more resources must be specified") },
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for k, testCase := range testCases {
 | |
| 		t.Run(k, func(t *testing.T) {
 | |
| 			tf := cmdtesting.NewTestFactory().WithNamespace("test")
 | |
| 			defer tf.Cleanup()
 | |
| 
 | |
| 			tf.ClientConfigVal = defaultClientConfig()
 | |
| 
 | |
| 			iostreams, _, bufOut, bufErr := genericclioptions.NewTestIOStreams()
 | |
| 			cmd := NewCmdAnnotate("kubectl", tf, iostreams)
 | |
| 			cmd.SetOutput(bufOut)
 | |
| 
 | |
| 			for k, v := range testCase.flags {
 | |
| 				cmd.Flags().Set(k, v)
 | |
| 			}
 | |
| 			options := NewAnnotateOptions(iostreams)
 | |
| 			err := options.Complete(tf, cmd, testCase.args)
 | |
| 			if err == nil {
 | |
| 				err = options.Validate()
 | |
| 			}
 | |
| 			if !testCase.errFn(err) {
 | |
| 				t.Errorf("%s: unexpected error: %v", k, err)
 | |
| 				return
 | |
| 			}
 | |
| 			if bufOut.Len() > 0 {
 | |
| 				t.Errorf("buffer should be empty: %s", string(bufOut.Bytes()))
 | |
| 			}
 | |
| 			if bufErr.Len() > 0 {
 | |
| 				t.Errorf("buffer should be empty: %s", string(bufErr.Bytes()))
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAnnotateObject(t *testing.T) {
 | |
| 	pods, _, _ := testData()
 | |
| 
 | |
| 	tf := cmdtesting.NewTestFactory().WithNamespace("test")
 | |
| 	defer tf.Cleanup()
 | |
| 
 | |
| 	codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
 | |
| 
 | |
| 	tf.UnstructuredClient = &fake.RESTClient{
 | |
| 		GroupVersion:         schema.GroupVersion{Group: "testgroup", Version: "v1"},
 | |
| 		NegotiatedSerializer: unstructuredSerializer,
 | |
| 		Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
 | |
| 			switch req.Method {
 | |
| 			case "GET":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/pods/foo":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			case "PATCH":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/pods/foo":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			default:
 | |
| 				t.Fatalf("unexpected request: %s %#v\n%#v", req.Method, req.URL, req)
 | |
| 				return nil, nil
 | |
| 			}
 | |
| 		}),
 | |
| 	}
 | |
| 	tf.ClientConfigVal = defaultClientConfig()
 | |
| 
 | |
| 	iostreams, _, bufOut, _ := genericclioptions.NewTestIOStreams()
 | |
| 	cmd := NewCmdAnnotate("kubectl", tf, iostreams)
 | |
| 	cmd.SetOutput(bufOut)
 | |
| 	options := NewAnnotateOptions(iostreams)
 | |
| 	args := []string{"pods/foo", "a=b", "c-"}
 | |
| 	if err := options.Complete(tf, cmd, args); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.Validate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.RunAnnotate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAnnotateObjectFromFile(t *testing.T) {
 | |
| 	pods, _, _ := testData()
 | |
| 
 | |
| 	tf := cmdtesting.NewTestFactory().WithNamespace("test")
 | |
| 	defer tf.Cleanup()
 | |
| 
 | |
| 	codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
 | |
| 
 | |
| 	tf.UnstructuredClient = &fake.RESTClient{
 | |
| 		GroupVersion:         schema.GroupVersion{Group: "testgroup", Version: "v1"},
 | |
| 		NegotiatedSerializer: unstructuredSerializer,
 | |
| 		Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
 | |
| 			switch req.Method {
 | |
| 			case "GET":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/replicationcontrollers/cassandra":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			case "PATCH":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/replicationcontrollers/cassandra":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			default:
 | |
| 				t.Fatalf("unexpected request: %s %#v\n%#v", req.Method, req.URL, req)
 | |
| 				return nil, nil
 | |
| 			}
 | |
| 		}),
 | |
| 	}
 | |
| 	tf.ClientConfigVal = defaultClientConfig()
 | |
| 
 | |
| 	iostreams, _, bufOut, _ := genericclioptions.NewTestIOStreams()
 | |
| 	cmd := NewCmdAnnotate("kubectl", tf, iostreams)
 | |
| 	cmd.SetOutput(bufOut)
 | |
| 	options := NewAnnotateOptions(iostreams)
 | |
| 	options.Filenames = []string{"../../../test/e2e/testing-manifests/statefulset/cassandra/controller.yaml"}
 | |
| 	args := []string{"a=b", "c-"}
 | |
| 	if err := options.Complete(tf, cmd, args); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.Validate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.RunAnnotate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAnnotateLocal(t *testing.T) {
 | |
| 	tf := cmdtesting.NewTestFactory().WithNamespace("test")
 | |
| 	defer tf.Cleanup()
 | |
| 
 | |
| 	tf.UnstructuredClient = &fake.RESTClient{
 | |
| 		GroupVersion:         schema.GroupVersion{Group: "testgroup", Version: "v1"},
 | |
| 		NegotiatedSerializer: unstructuredSerializer,
 | |
| 		Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
 | |
| 			t.Fatalf("unexpected request: %s %#v\n%#v", req.Method, req.URL, req)
 | |
| 			return nil, nil
 | |
| 		}),
 | |
| 	}
 | |
| 	tf.ClientConfigVal = defaultClientConfig()
 | |
| 
 | |
| 	iostreams, _, _, _ := genericclioptions.NewTestIOStreams()
 | |
| 	cmd := NewCmdAnnotate("kubectl", tf, iostreams)
 | |
| 	options := NewAnnotateOptions(iostreams)
 | |
| 	options.local = true
 | |
| 	options.Filenames = []string{"../../../test/e2e/testing-manifests/statefulset/cassandra/controller.yaml"}
 | |
| 	args := []string{"a=b"}
 | |
| 	if err := options.Complete(tf, cmd, args); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.Validate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.RunAnnotate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAnnotateMultipleObjects(t *testing.T) {
 | |
| 	pods, _, _ := testData()
 | |
| 
 | |
| 	tf := cmdtesting.NewTestFactory().WithNamespace("test")
 | |
| 	defer tf.Cleanup()
 | |
| 
 | |
| 	codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
 | |
| 	tf.UnstructuredClient = &fake.RESTClient{
 | |
| 		GroupVersion:         schema.GroupVersion{Group: "testgroup", Version: "v1"},
 | |
| 		NegotiatedSerializer: unstructuredSerializer,
 | |
| 		Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
 | |
| 			switch req.Method {
 | |
| 			case "GET":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/pods":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, pods)}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			case "PATCH":
 | |
| 				switch req.URL.Path {
 | |
| 				case "/namespaces/test/pods/foo":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
 | |
| 				case "/namespaces/test/pods/bar":
 | |
| 					return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[1])}, nil
 | |
| 				default:
 | |
| 					t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
 | |
| 					return nil, nil
 | |
| 				}
 | |
| 			default:
 | |
| 				t.Fatalf("unexpected request: %s %#v\n%#v", req.Method, req.URL, req)
 | |
| 				return nil, nil
 | |
| 			}
 | |
| 		}),
 | |
| 	}
 | |
| 	tf.ClientConfigVal = defaultClientConfig()
 | |
| 
 | |
| 	iostreams, _, _, _ := genericclioptions.NewTestIOStreams()
 | |
| 	cmd := NewCmdAnnotate("kubectl", tf, iostreams)
 | |
| 	cmd.SetOutput(iostreams.Out)
 | |
| 	options := NewAnnotateOptions(iostreams)
 | |
| 	options.all = true
 | |
| 	args := []string{"pods", "a=b", "c-"}
 | |
| 	if err := options.Complete(tf, cmd, args); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.Validate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| 	if err := options.RunAnnotate(); err != nil {
 | |
| 		t.Fatalf("unexpected error: %v", err)
 | |
| 	}
 | |
| }
 |