
Some of the main changes: - add dryrun to annotate (can push this in a different PR if requested) - use Complete(), Validate() and RunX() - don't place dynamic variables in the options (only user options and args) - call the NewBuilder() in the Run function. You can now do diff between these files and they are as identical as possible.
483 lines
12 KiB
Go
483 lines
12 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 (
|
|
"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/kubectl/resource"
|
|
"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="},
|
|
expected: map[string]string{"a": ""},
|
|
},
|
|
{
|
|
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)
|
|
}
|
|
opts := LabelOptions{}
|
|
err := opts.Complete(f, buf, cmd, testCase.args)
|
|
if err == nil {
|
|
err = opts.Validate()
|
|
}
|
|
if err == nil {
|
|
err = opts.RunLabel(f, cmd)
|
|
}
|
|
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)
|
|
opts := LabelOptions{FilenameOptions: resource.FilenameOptions{
|
|
Filenames: []string{"../../../examples/storage/cassandra/cassandra-controller.yaml"}}}
|
|
err := opts.Complete(f, buf, cmd, []string{"a=b"})
|
|
if err == nil {
|
|
err = opts.Validate()
|
|
}
|
|
if err == nil {
|
|
err = opts.RunLabel(f, cmd)
|
|
}
|
|
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 TestLabelLocal(t *testing.T) {
|
|
f, tf, _, ns := NewAPIFactory()
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: ns,
|
|
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.Namespace = "test"
|
|
tf.ClientConfig = &restclient.Config{ContentConfig: restclient.ContentConfig{GroupVersion: testapi.Default.GroupVersion()}}
|
|
|
|
buf := bytes.NewBuffer([]byte{})
|
|
cmd := NewCmdLabel(f, buf)
|
|
cmd.Flags().Set("local", "true")
|
|
opts := LabelOptions{FilenameOptions: resource.FilenameOptions{
|
|
Filenames: []string{"../../../examples/storage/cassandra/cassandra-controller.yaml"}}}
|
|
err := opts.Complete(f, buf, cmd, []string{"a=b"})
|
|
if err == nil {
|
|
err = opts.Validate()
|
|
}
|
|
if err == nil {
|
|
err = opts.RunLabel(f, cmd)
|
|
}
|
|
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")
|
|
|
|
opts := LabelOptions{}
|
|
err := opts.Complete(f, buf, cmd, []string{"pods", "a=b"})
|
|
if err == nil {
|
|
err = opts.Validate()
|
|
}
|
|
if err == nil {
|
|
err = opts.RunLabel(f, cmd)
|
|
}
|
|
if 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())
|
|
}
|
|
}
|