123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430 |
- /*
- 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 (
- "bytes"
- "net/http"
- "reflect"
- "strings"
- "testing"
- "k8s.io/kubernetes/pkg/api"
- "k8s.io/kubernetes/pkg/api/testapi"
- "k8s.io/kubernetes/pkg/client/restclient"
- "k8s.io/kubernetes/pkg/client/unversioned/fake"
- "k8s.io/kubernetes/pkg/runtime"
- )
- func TestValidateLabels(t *testing.T) {
- tests := []struct {
- meta *api.ObjectMeta
- labels map[string]string
- expectErr bool
- test string
- }{
- {
- meta: &api.ObjectMeta{
- Labels: map[string]string{
- "a": "b",
- "c": "d",
- },
- },
- labels: map[string]string{
- "a": "c",
- "d": "b",
- },
- test: "one shared",
- expectErr: true,
- },
- {
- meta: &api.ObjectMeta{
- Labels: map[string]string{
- "a": "b",
- "c": "d",
- },
- },
- labels: map[string]string{
- "b": "d",
- "c": "a",
- },
- test: "second shared",
- expectErr: true,
- },
- {
- meta: &api.ObjectMeta{
- Labels: map[string]string{
- "a": "b",
- "c": "d",
- },
- },
- labels: map[string]string{
- "b": "a",
- "d": "c",
- },
- test: "no overlap",
- },
- {
- meta: &api.ObjectMeta{},
- labels: map[string]string{
- "b": "a",
- "d": "c",
- },
- test: "no labels",
- },
- }
- for _, test := range tests {
- err := validateNoOverwrites(test.meta, test.labels)
- if test.expectErr && err == nil {
- t.Errorf("%s: unexpected non-error", test.test)
- }
- if !test.expectErr && err != nil {
- t.Errorf("%s: unexpected error: %v", test.test, err)
- }
- }
- }
- func TestParseLabels(t *testing.T) {
- tests := []struct {
- labels []string
- expected map[string]string
- expectedRemove []string
- expectErr bool
- }{
- {
- labels: []string{"a=b", "c=d"},
- expected: map[string]string{"a": "b", "c": "d"},
- },
- {
- labels: []string{},
- expected: map[string]string{},
- },
- {
- labels: []string{"a=b", "c=d", "e-"},
- expected: map[string]string{"a": "b", "c": "d"},
- expectedRemove: []string{"e"},
- },
- {
- labels: []string{"ab", "c=d"},
- expectErr: true,
- },
- {
- labels: []string{"a=b", "c=d", "a-"},
- expectErr: true,
- },
- {
- labels: []string{"a="},
- expectErr: true,
- },
- {
- labels: []string{"a=%^$"},
- expectErr: true,
- },
- }
- for _, test := range tests {
- labels, remove, err := parseLabels(test.labels)
- if test.expectErr && err == nil {
- t.Errorf("unexpected non-error: %v", test)
- }
- if !test.expectErr && err != nil {
- t.Errorf("unexpected error: %v %v", err, test)
- }
- if !reflect.DeepEqual(labels, test.expected) {
- t.Errorf("expected: %v, got %v", test.expected, labels)
- }
- if !reflect.DeepEqual(remove, test.expectedRemove) {
- t.Errorf("expected: %v, got %v", test.expectedRemove, remove)
- }
- }
- }
- func TestLabelFunc(t *testing.T) {
- tests := []struct {
- obj runtime.Object
- overwrite bool
- version string
- labels map[string]string
- remove []string
- expected runtime.Object
- expectErr bool
- }{
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- labels: map[string]string{"a": "b"},
- expectErr: true,
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- labels: map[string]string{"a": "c"},
- overwrite: true,
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "c"},
- },
- },
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- labels: map[string]string{"c": "d"},
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b", "c": "d"},
- },
- },
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- labels: map[string]string{"c": "d"},
- version: "2",
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b", "c": "d"},
- ResourceVersion: "2",
- },
- },
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- labels: map[string]string{},
- remove: []string{"a"},
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{},
- },
- },
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b", "c": "d"},
- },
- },
- labels: map[string]string{"e": "f"},
- remove: []string{"a"},
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{
- "c": "d",
- "e": "f",
- },
- },
- },
- },
- {
- obj: &api.Pod{
- ObjectMeta: api.ObjectMeta{},
- },
- labels: map[string]string{"a": "b"},
- expected: &api.Pod{
- ObjectMeta: api.ObjectMeta{
- Labels: map[string]string{"a": "b"},
- },
- },
- },
- }
- for _, test := range tests {
- err := labelFunc(test.obj, test.overwrite, test.version, test.labels, test.remove)
- 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 TestLabelErrors(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 labels": {
- args: []string{"pods"},
- errFn: func(err error) bool { return strings.Contains(err.Error(), "at least one label update is required") },
- },
- "no resources": {
- args: []string{"pods-"},
- errFn: func(err error) bool { return strings.Contains(err.Error(), "one or more resources must be specified") },
- },
- "no resources 2": {
- args: []string{"pods=bar"},
- errFn: func(err error) bool { return strings.Contains(err.Error(), "one or more resources must be specified") },
- },
- "resources but no selectors": {
- args: []string{"pods", "app=bar"},
- errFn: func(err error) bool {
- return strings.Contains(err.Error(), "resource(s) were provided, but no name, label selector, or --all flag specified")
- },
- },
- "multiple resources but no selectors": {
- args: []string{"pods,deployments", "app=bar"},
- errFn: func(err error) bool {
- return strings.Contains(err.Error(), "resource(s) were provided, but no name, label selector, or --all flag specified")
- },
- },
- }
- for k, testCase := range testCases {
- f, tf, _, _ := NewAPIFactory()
- tf.Printer = &testPrinter{}
- tf.Namespace = "test"
- tf.ClientConfig = &restclient.Config{ContentConfig: restclient.ContentConfig{GroupVersion: testapi.Default.GroupVersion()}}
- buf := bytes.NewBuffer([]byte{})
- cmd := NewCmdLabel(f, buf)
- cmd.SetOutput(buf)
- for k, v := range testCase.flags {
- cmd.Flags().Set(k, v)
- }
- err := RunLabel(f, buf, cmd, testCase.args, &LabelOptions{})
- if !testCase.errFn(err) {
- t.Errorf("%s: unexpected error: %v", k, err)
- continue
- }
- if tf.Printer.(*testPrinter).Objects != nil {
- t.Errorf("unexpected print to default printer")
- }
- if buf.Len() > 0 {
- t.Errorf("buffer should be empty: %s", string(buf.Bytes()))
- }
- }
- }
- func TestLabelForResourceFromFile(t *testing.T) {
- pods, _, _ := testData()
- f, tf, codec, ns := NewAPIFactory()
- tf.Client = &fake.RESTClient{
- NegotiatedSerializer: ns,
- 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.Namespace = "test"
- tf.ClientConfig = &restclient.Config{ContentConfig: restclient.ContentConfig{GroupVersion: testapi.Default.GroupVersion()}}
- buf := bytes.NewBuffer([]byte{})
- cmd := NewCmdLabel(f, buf)
- options := &LabelOptions{
- Filenames: []string{"../../../examples/storage/cassandra/cassandra-controller.yaml"},
- }
- err := RunLabel(f, buf, cmd, []string{"a=b"}, options)
- if err != nil {
- t.Fatalf("unexpected error: %v", err)
- }
- if !strings.Contains(buf.String(), "labeled") {
- t.Errorf("did not set labels: %s", buf.String())
- }
- }
- func TestLabelMultipleObjects(t *testing.T) {
- pods, _, _ := testData()
- f, tf, codec, ns := NewAPIFactory()
- tf.Client = &fake.RESTClient{
- NegotiatedSerializer: ns,
- 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.Namespace = "test"
- tf.ClientConfig = &restclient.Config{ContentConfig: restclient.ContentConfig{GroupVersion: testapi.Default.GroupVersion()}}
- buf := bytes.NewBuffer([]byte{})
- cmd := NewCmdLabel(f, buf)
- cmd.Flags().Set("all", "true")
- if err := RunLabel(f, buf, cmd, []string{"pods", "a=b"}, &LabelOptions{}); err != nil {
- t.Fatalf("unexpected error: %v", err)
- }
- if strings.Count(buf.String(), "labeled") != len(pods.Items) {
- t.Errorf("not all labels are set: %s", buf.String())
- }
- }
|