/* Copyright 2014 Google Inc. All rights reserved. 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 ( "bytes" "encoding/json" "fmt" "io" "reflect" "strings" "testing" "time" "github.com/GoogleCloudPlatform/kubernetes/pkg/api" "github.com/GoogleCloudPlatform/kubernetes/pkg/api/testapi" "github.com/GoogleCloudPlatform/kubernetes/pkg/api/v1beta1" "github.com/GoogleCloudPlatform/kubernetes/pkg/runtime" "github.com/GoogleCloudPlatform/kubernetes/pkg/util" "github.com/ghodss/yaml" ) type testStruct struct { api.TypeMeta `json:",inline"` api.ObjectMeta `json:"metadata,omitempty"` Key string `json:"Key"` Map map[string]int `json:"Map"` StringList []string `json:"StringList"` IntList []int `json:"IntList"` } func (ts *testStruct) IsAnAPIObject() {} func init() { api.Scheme.AddKnownTypes("", &testStruct{}) api.Scheme.AddKnownTypes(testapi.Version(), &testStruct{}) } var testData = testStruct{ Key: "testValue", Map: map[string]int{"TestSubkey": 1}, StringList: []string{"a", "b", "c"}, IntList: []int{1, 2, 3}, } func TestVersionedPrinter(t *testing.T) { original := &testStruct{Key: "value"} p := NewVersionedPrinter( ResourcePrinterFunc(func(obj runtime.Object, w io.Writer) error { if obj == original { t.Fatalf("object should not be identical: %#v", obj) } if obj.(*testStruct).Key != "value" { t.Fatalf("object was not converted: %#v", obj) } return nil }), api.Scheme, testapi.Version(), ) if err := p.PrintObj(original, nil); err != nil { t.Errorf("unexpected error: %v", err) } } func TestYAMLPrinter(t *testing.T) { testPrinter(t, &YAMLPrinter{}, yaml.Unmarshal) } func TestJSONPrinter(t *testing.T) { testPrinter(t, &JSONPrinter{}, json.Unmarshal) } func TestPrintDefault(t *testing.T) { printer, found, err := GetPrinter("", "") if err != nil { t.Fatalf("unexpected error: %#v", err) } if found { t.Errorf("no printer should have been found: %#v / %v", printer, err) } } type internalType struct { Name string } func (*internalType) IsAnAPIObject() { } func TestPrintJSONForObject(t *testing.T) { buf := bytes.NewBuffer([]byte{}) printer, found, err := GetPrinter("json", "") if err != nil || !found { t.Fatalf("unexpected error: %#v", err) } if err := printer.PrintObj(&internalType{Name: "foo"}, buf); err != nil { t.Fatalf("unexpected error: %#v", err) } obj := map[string]interface{}{} if err := json.Unmarshal(buf.Bytes(), &obj); err != nil { t.Fatalf("unexpected error: %#v\n%s", err, buf.String()) } if obj["Name"] != "foo" { t.Errorf("unexpected field: %#v", obj) } } func TestPrintJSON(t *testing.T) { buf := bytes.NewBuffer([]byte{}) printer, found, err := GetPrinter("json", "") if err != nil || !found { t.Fatalf("unexpected error: %#v", err) } printer.PrintObj(&api.Pod{ObjectMeta: api.ObjectMeta{Name: "foo"}}, buf) obj := map[string]interface{}{} if err := json.Unmarshal(buf.Bytes(), &obj); err != nil { t.Errorf("unexpected error: %#v\n%s", err, buf.String()) } } func TestPrintYAML(t *testing.T) { buf := bytes.NewBuffer([]byte{}) printer, found, err := GetPrinter("yaml", "") if err != nil || !found { t.Fatalf("unexpected error: %#v", err) } printer.PrintObj(&api.Pod{ObjectMeta: api.ObjectMeta{Name: "foo"}}, buf) obj := map[string]interface{}{} if err := yaml.Unmarshal(buf.Bytes(), &obj); err != nil { t.Errorf("unexpected error: %#v\n%s", err, buf.String()) } } func TestPrintTemplate(t *testing.T) { buf := bytes.NewBuffer([]byte{}) printer, found, err := GetPrinter("template", "{{.id}}") if err != nil || !found { t.Fatalf("unexpected error: %#v", err) } err = printer.PrintObj(&v1beta1.Pod{TypeMeta: v1beta1.TypeMeta{ID: "foo"}}, buf) if err != nil { t.Fatalf("unexpected error: %#v", err) } if buf.String() != "foo" { t.Errorf("unexpected output: %s", buf.String()) } } func TestPrintEmptyTemplate(t *testing.T) { if _, _, err := GetPrinter("template", ""); err == nil { t.Errorf("unexpected non-error") } } func TestPrintBadTemplate(t *testing.T) { if _, _, err := GetPrinter("template", "{{ .Name"); err == nil { t.Errorf("unexpected non-error") } } func TestPrintBadTemplateFile(t *testing.T) { if _, _, err := GetPrinter("templatefile", ""); err == nil { t.Errorf("unexpected non-error") } } func testPrinter(t *testing.T, printer ResourcePrinter, unmarshalFunc func(data []byte, v interface{}) error) { buf := bytes.NewBuffer([]byte{}) err := printer.PrintObj(&testData, buf) if err != nil { t.Fatal(err) } var poutput testStruct // Verify that given function runs without error. err = unmarshalFunc(buf.Bytes(), &poutput) if err != nil { t.Fatal(err) } // Use real decode function to undo the versioning process. poutput = testStruct{} err = runtime.YAMLDecoder(testapi.Codec()).DecodeInto(buf.Bytes(), &poutput) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(testData, poutput) { t.Errorf("Test data and unmarshaled data are not equal: %v", util.ObjectDiff(poutput, testData)) } obj := &api.Pod{ ObjectMeta: api.ObjectMeta{Name: "foo"}, } buf.Reset() printer.PrintObj(obj, buf) var objOut api.Pod // Verify that given function runs without error. err = unmarshalFunc(buf.Bytes(), &objOut) if err != nil { t.Fatalf("unexpected error: %#v", err) } // Use real decode function to undo the versioning process. objOut = api.Pod{} err = runtime.YAMLDecoder(testapi.Codec()).DecodeInto(buf.Bytes(), &objOut) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(obj, &objOut) { t.Errorf("Unexpected inequality:\n%v", util.ObjectDiff(obj, &objOut)) } } type TestPrintType struct { Data string } func (*TestPrintType) IsAnAPIObject() {} type TestUnknownType struct{} func (*TestUnknownType) IsAnAPIObject() {} func PrintCustomType(obj *TestPrintType, w io.Writer) error { _, err := fmt.Fprintf(w, "%s", obj.Data) return err } func ErrorPrintHandler(obj *TestPrintType, w io.Writer) error { return fmt.Errorf("ErrorPrintHandler error") } func TestCustomTypePrinting(t *testing.T) { columns := []string{"Data"} printer := NewHumanReadablePrinter(false) printer.Handler(columns, PrintCustomType) obj := TestPrintType{"test object"} buffer := &bytes.Buffer{} err := printer.PrintObj(&obj, buffer) if err != nil { t.Fatalf("An error occurred printing the custom type: %#v", err) } expectedOutput := "Data\ntest object" if buffer.String() != expectedOutput { t.Errorf("The data was not printed as expected. Expected:\n%s\nGot:\n%s", expectedOutput, buffer.String()) } } func TestPrintHandlerError(t *testing.T) { columns := []string{"Data"} printer := NewHumanReadablePrinter(false) printer.Handler(columns, ErrorPrintHandler) obj := TestPrintType{"test object"} buffer := &bytes.Buffer{} err := printer.PrintObj(&obj, buffer) if err == nil || err.Error() != "ErrorPrintHandler error" { t.Errorf("Did not get the expected error: %#v", err) } } func TestUnknownTypePrinting(t *testing.T) { printer := NewHumanReadablePrinter(false) buffer := &bytes.Buffer{} err := printer.PrintObj(&TestUnknownType{}, buffer) if err == nil { t.Errorf("An error was expected from printing unknown type") } } func TestTemplateEmitsVersionedObjects(t *testing.T) { // kind is always blank in memory and set on the wire printer, err := NewTemplatePrinter([]byte(`{{.kind}}`)) if err != nil { t.Fatalf("tmpl fail: %v", err) } obj, err := api.Scheme.ConvertToVersion(&api.Pod{}, "v1beta1") if err != nil { t.Fatalf("unexpected error: %v", err) } buffer := &bytes.Buffer{} err = printer.PrintObj(obj, buffer) if err != nil { t.Fatalf("print fail: %v", err) } if e, a := "Pod", string(buffer.Bytes()); e != a { t.Errorf("Expected %v, got %v", e, a) } } func TestTemplatePanic(t *testing.T) { tmpl := `{{and ((index .currentState.info "foo").state.running.startedAt) .currentState.info.net.state.running.startedAt}}` printer, err := NewTemplatePrinter([]byte(tmpl)) if err != nil { t.Fatalf("tmpl fail: %v", err) } buffer := &bytes.Buffer{} err = printer.PrintObj(&api.Pod{}, buffer) if err == nil { t.Fatalf("expected that template to crash") } if buffer.String() == "" { t.Errorf("no debugging info was printed") } } func TestTemplateStrings(t *testing.T) { // This unit tests the "exists" function as well as the template from update.sh table := map[string]struct { pod api.Pod expect string }{ "nilInfo": {api.Pod{}, "false"}, "emptyInfo": {api.Pod{Status: api.PodStatus{ContainerStatuses: []api.ContainerStatus{}}}, "false"}, "fooExists": { api.Pod{ Status: api.PodStatus{ ContainerStatuses: []api.ContainerStatus{ { Name: "foo", }, }, }, }, "false", }, "barExists": { api.Pod{ Status: api.PodStatus{ ContainerStatuses: []api.ContainerStatus{ { Name: "bar", }, }, }, }, "false", }, "bothExist": { api.Pod{ Status: api.PodStatus{ ContainerStatuses: []api.ContainerStatus{ { Name: "foo", }, { Name: "bar", }, }, }, }, "false", }, "oneValid": { api.Pod{ Status: api.PodStatus{ ContainerStatuses: []api.ContainerStatus{ { Name: "foo", }, { Name: "bar", State: api.ContainerState{ Running: &api.ContainerStateRunning{ StartedAt: util.Time{}, }, }, }, }, }, }, "false", }, "bothValid": { api.Pod{ Status: api.PodStatus{ ContainerStatuses: []api.ContainerStatus{ { Name: "foo", State: api.ContainerState{ Running: &api.ContainerStateRunning{ StartedAt: util.Time{}, }, }, }, { Name: "bar", State: api.ContainerState{ Running: &api.ContainerStateRunning{ StartedAt: util.Time{}, }, }, }, }, }, }, "true", }, } // The point of this test is to verify that the below template works. If you change this // template, you need to update hack/e2e-suite/update.sh. tmpl := `{{and (exists . "currentState" "info" "foo" "state" "running") (exists . "currentState" "info" "bar" "state" "running")}}` useThisToDebug := ` a: {{exists . "currentState"}} b: {{exists . "currentState" "info"}} c: {{exists . "currentState" "info" "foo"}} d: {{exists . "currentState" "info" "foo" "state"}} e: {{exists . "currentState" "info" "foo" "state" "running"}} f: {{exists . "currentState" "info" "foo" "state" "running" "startedAt"}}` _ = useThisToDebug // don't complain about unused var p, err := NewTemplatePrinter([]byte(tmpl)) if err != nil { t.Fatalf("tmpl fail: %v", err) } printer := NewVersionedPrinter(p, api.Scheme, "v1beta1") for name, item := range table { buffer := &bytes.Buffer{} err = printer.PrintObj(&item.pod, buffer) if err != nil { t.Errorf("%v: unexpected err: %v", name, err) continue } if e, a := item.expect, buffer.String(); e != a { t.Errorf("%v: expected %v, got %v", name, e, a) } } } func TestPrinters(t *testing.T) { om := func(name string) api.ObjectMeta { return api.ObjectMeta{Name: name} } templatePrinter, err := NewTemplatePrinter([]byte("{{.name}}")) if err != nil { t.Fatal(err) } templatePrinter2, err := NewTemplatePrinter([]byte("{{len .items}}")) if err != nil { t.Fatal(err) } printers := map[string]ResourcePrinter{ "humanReadable": NewHumanReadablePrinter(true), "humanReadableHeaders": NewHumanReadablePrinter(false), "json": &JSONPrinter{}, "yaml": &YAMLPrinter{}, "template": templatePrinter, "template2": templatePrinter2, } objects := map[string]runtime.Object{ "pod": &api.Pod{ObjectMeta: om("pod")}, "emptyPodList": &api.PodList{}, "nonEmptyPodList": &api.PodList{Items: []api.Pod{{}}}, "endpoints": &api.Endpoints{ Subsets: []api.EndpointSubset{{ Addresses: []api.EndpointAddress{{IP: "127.0.0.1"}, {IP: "localhost"}}, Ports: []api.EndpointPort{{Port: 8080}}, }}}, } // map of printer name to set of objects it should fail on. expectedErrors := map[string]util.StringSet{ "template2": util.NewStringSet("pod", "emptyPodList", "endpoints"), } for pName, p := range printers { for oName, obj := range objects { b := &bytes.Buffer{} if err := p.PrintObj(obj, b); err != nil { if set, found := expectedErrors[pName]; found && set.Has(oName) { // expected error continue } t.Errorf("printer '%v', object '%v'; error: '%v'", pName, oName, err) } } } } func TestPrintEventsResultSorted(t *testing.T) { // Arrange printer := NewHumanReadablePrinter(false /* noHeaders */) obj := api.EventList{ Items: []api.Event{ { Source: api.EventSource{Component: "kubelet"}, Message: "Item 1", FirstTimestamp: util.NewTime(time.Date(2014, time.January, 15, 0, 0, 0, 0, time.UTC)), LastTimestamp: util.NewTime(time.Date(2014, time.January, 15, 0, 0, 0, 0, time.UTC)), Count: 1, }, { Source: api.EventSource{Component: "scheduler"}, Message: "Item 2", FirstTimestamp: util.NewTime(time.Date(1987, time.June, 17, 0, 0, 0, 0, time.UTC)), LastTimestamp: util.NewTime(time.Date(1987, time.June, 17, 0, 0, 0, 0, time.UTC)), Count: 1, }, { Source: api.EventSource{Component: "kubelet"}, Message: "Item 3", FirstTimestamp: util.NewTime(time.Date(2002, time.December, 25, 0, 0, 0, 0, time.UTC)), LastTimestamp: util.NewTime(time.Date(2002, time.December, 25, 0, 0, 0, 0, time.UTC)), Count: 1, }, }, } buffer := &bytes.Buffer{} // Act err := printer.PrintObj(&obj, buffer) // Assert if err != nil { t.Fatalf("An error occurred printing the EventList: %#v", err) } out := buffer.String() VerifyDatesInOrder(out, "\n" /* rowDelimiter */, " " /* columnDelimiter */, t) } func TestPrintMinionStatus(t *testing.T) { printer := NewHumanReadablePrinter(false) table := []struct { minion api.Node status string }{ { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo1"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{{Type: api.NodeReady, Status: api.ConditionTrue}}}, }, status: "Ready", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo3"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{ {Type: api.NodeReady, Status: api.ConditionTrue}, {Type: api.NodeReady, Status: api.ConditionTrue}}}, }, status: "Ready", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo4"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{{Type: api.NodeReady, Status: api.ConditionFalse}}}, }, status: "NotReady", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo5"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{{Type: "InvalidValue", Status: api.ConditionTrue}}}, }, status: "Unknown", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo6"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{{}}}, }, status: "Unknown", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo7"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{ {Type: api.NodeSchedulable, Status: api.ConditionTrue}, {Type: api.NodeReady, Status: api.ConditionTrue}}}, }, status: "Schedulable,Ready", }, { minion: api.Node{ ObjectMeta: api.ObjectMeta{Name: "foo8"}, Status: api.NodeStatus{Conditions: []api.NodeCondition{ {Type: api.NodeSchedulable, Status: api.ConditionFalse}, {Type: api.NodeReady, Status: api.ConditionFalse}}}, }, status: "NotSchedulable,NotReady", }, } for _, test := range table { buffer := &bytes.Buffer{} err := printer.PrintObj(&test.minion, buffer) if err != nil { t.Fatalf("An error occurred printing Minion: %#v", err) } if !contains(strings.Fields(buffer.String()), test.status) { t.Fatalf("Expect printing minion %s with status %#v, got: %#v", test.minion.Name, test.status, buffer.String()) } } } func contains(fields []string, field string) bool { for _, v := range fields { if v == field { return true } } return false } func TestPrintHumanReadableService(t *testing.T) { tests := []api.Service{ { Spec: api.ServiceSpec{ PortalIP: "1.2.3.4", PublicIPs: []string{ "2.3.4.5", "3.4.5.6", }, Ports: []api.ServicePort{ { Port: 80, Protocol: "TCP", }, }, }, }, { Spec: api.ServiceSpec{ PortalIP: "1.2.3.4", Ports: []api.ServicePort{ { Port: 80, Protocol: "TCP", }, { Port: 8090, Protocol: "UDP", }, { Port: 8000, Protocol: "TCP", }, }, }, }, { Spec: api.ServiceSpec{ PortalIP: "1.2.3.4", PublicIPs: []string{ "2.3.4.5", }, Ports: []api.ServicePort{ { Port: 80, Protocol: "TCP", }, { Port: 8090, Protocol: "UDP", }, { Port: 8000, Protocol: "TCP", }, }, }, }, { Spec: api.ServiceSpec{ PortalIP: "1.2.3.4", PublicIPs: []string{ "2.3.4.5", "4.5.6.7", "5.6.7.8", }, Ports: []api.ServicePort{ { Port: 80, Protocol: "TCP", }, { Port: 8090, Protocol: "UDP", }, { Port: 8000, Protocol: "TCP", }, }, }, }, } for _, svc := range tests { buff := bytes.Buffer{} printService(&svc, &buff) output := string(buff.Bytes()) ip := svc.Spec.PortalIP if !strings.Contains(output, ip) { t.Errorf("expected to contain portal ip %s, but doesn't: %s", ip, output) } for _, ip = range svc.Spec.PublicIPs { if !strings.Contains(output, ip) { t.Errorf("expected to contain public ip %s, but doesn't: %s", ip, output) } } for _, port := range svc.Spec.Ports { portSpec := fmt.Sprintf("%d/%s", port.Port, port.Protocol) if !strings.Contains(output, portSpec) { t.Errorf("expected to contain port: %s, but doesn't: %s", portSpec, output) } } // Max of # ports and (# public ip + portal ip) count := len(svc.Spec.Ports) if len(svc.Spec.PublicIPs)+1 > count { count = len(svc.Spec.PublicIPs) + 1 } if count != strings.Count(output, "\n") { t.Errorf("expected %d newlines, found %d", count, strings.Count(output, "\n")) } } }