table_conversion.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. /*
  2. Copyright 2017 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package apimachinery
  14. import (
  15. "bytes"
  16. "context"
  17. "fmt"
  18. "text/tabwriter"
  19. "github.com/onsi/ginkgo"
  20. "github.com/onsi/gomega"
  21. authorizationv1 "k8s.io/api/authorization/v1"
  22. v1 "k8s.io/api/core/v1"
  23. apierrors "k8s.io/apimachinery/pkg/api/errors"
  24. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  25. metav1beta1 "k8s.io/apimachinery/pkg/apis/meta/v1beta1"
  26. "k8s.io/client-go/util/workqueue"
  27. utilversion "k8s.io/apimachinery/pkg/util/version"
  28. "k8s.io/cli-runtime/pkg/printers"
  29. "k8s.io/kubernetes/test/e2e/framework"
  30. e2eskipper "k8s.io/kubernetes/test/e2e/framework/skipper"
  31. imageutils "k8s.io/kubernetes/test/utils/image"
  32. )
  33. var serverPrintVersion = utilversion.MustParseSemantic("v1.10.0")
  34. var _ = SIGDescribe("Servers with support for Table transformation", func() {
  35. f := framework.NewDefaultFramework("tables")
  36. ginkgo.BeforeEach(func() {
  37. e2eskipper.SkipUnlessServerVersionGTE(serverPrintVersion, f.ClientSet.Discovery())
  38. })
  39. ginkgo.It("should return pod details", func() {
  40. ns := f.Namespace.Name
  41. c := f.ClientSet
  42. podName := "pod-1"
  43. framework.Logf("Creating pod %s", podName)
  44. _, err := c.CoreV1().Pods(ns).Create(context.TODO(), newTablePod(podName), metav1.CreateOptions{})
  45. framework.ExpectNoError(err, "failed to create pod %s in namespace: %s", podName, ns)
  46. table := &metav1beta1.Table{}
  47. err = c.CoreV1().RESTClient().Get().Resource("pods").Namespace(ns).Name(podName).SetHeader("Accept", "application/json;as=Table;v=v1beta1;g=meta.k8s.io").Do(context.TODO()).Into(table)
  48. framework.ExpectNoError(err, "failed to get pod %s in Table form in namespace: %s", podName, ns)
  49. framework.Logf("Table: %#v", table)
  50. gomega.Expect(len(table.ColumnDefinitions)).To(gomega.BeNumerically(">", 2))
  51. framework.ExpectEqual(len(table.Rows), 1)
  52. framework.ExpectEqual(len(table.Rows[0].Cells), len(table.ColumnDefinitions))
  53. framework.ExpectEqual(table.ColumnDefinitions[0].Name, "Name")
  54. framework.ExpectEqual(table.Rows[0].Cells[0], podName)
  55. out := printTable(table)
  56. gomega.Expect(out).To(gomega.MatchRegexp("^NAME\\s"))
  57. gomega.Expect(out).To(gomega.MatchRegexp("\npod-1\\s"))
  58. framework.Logf("Table:\n%s", out)
  59. })
  60. ginkgo.It("should return chunks of table results for list calls", func() {
  61. ns := f.Namespace.Name
  62. c := f.ClientSet
  63. client := c.CoreV1().PodTemplates(ns)
  64. ginkgo.By("creating a large number of resources")
  65. workqueue.ParallelizeUntil(context.TODO(), 5, 20, func(i int) {
  66. for tries := 3; tries >= 0; tries-- {
  67. _, err := client.Create(context.TODO(), &v1.PodTemplate{
  68. ObjectMeta: metav1.ObjectMeta{
  69. Name: fmt.Sprintf("template-%04d", i),
  70. },
  71. Template: v1.PodTemplateSpec{
  72. Spec: v1.PodSpec{
  73. Containers: []v1.Container{
  74. {Name: "test", Image: "test2"},
  75. },
  76. },
  77. },
  78. }, metav1.CreateOptions{})
  79. if err == nil {
  80. return
  81. }
  82. framework.Logf("Got an error creating template %d: %v", i, err)
  83. }
  84. framework.Failf("Unable to create template %d, exiting", i)
  85. })
  86. pagedTable := &metav1beta1.Table{}
  87. err := c.CoreV1().RESTClient().Get().Namespace(ns).Resource("podtemplates").
  88. VersionedParams(&metav1.ListOptions{Limit: 2}, metav1.ParameterCodec).
  89. SetHeader("Accept", "application/json;as=Table;v=v1beta1;g=meta.k8s.io").
  90. Do(context.TODO()).Into(pagedTable)
  91. framework.ExpectNoError(err, "failed to get pod templates in Table form in namespace: %s", ns)
  92. framework.ExpectEqual(len(pagedTable.Rows), 2)
  93. framework.ExpectNotEqual(pagedTable.ResourceVersion, "")
  94. framework.ExpectNotEqual(pagedTable.SelfLink, "")
  95. framework.ExpectNotEqual(pagedTable.Continue, "")
  96. framework.ExpectEqual(pagedTable.Rows[0].Cells[0], "template-0000")
  97. framework.ExpectEqual(pagedTable.Rows[1].Cells[0], "template-0001")
  98. err = c.CoreV1().RESTClient().Get().Namespace(ns).Resource("podtemplates").
  99. VersionedParams(&metav1.ListOptions{Continue: pagedTable.Continue}, metav1.ParameterCodec).
  100. SetHeader("Accept", "application/json;as=Table;v=v1beta1;g=meta.k8s.io").
  101. Do(context.TODO()).Into(pagedTable)
  102. framework.ExpectNoError(err, "failed to get pod templates in Table form in namespace: %s", ns)
  103. gomega.Expect(len(pagedTable.Rows)).To(gomega.BeNumerically(">", 0))
  104. framework.ExpectEqual(pagedTable.Rows[0].Cells[0], "template-0002")
  105. })
  106. ginkgo.It("should return generic metadata details across all namespaces for nodes", func() {
  107. c := f.ClientSet
  108. table := &metav1beta1.Table{}
  109. err := c.CoreV1().RESTClient().Get().Resource("nodes").SetHeader("Accept", "application/json;as=Table;v=v1beta1;g=meta.k8s.io").Do(context.TODO()).Into(table)
  110. framework.ExpectNoError(err, "failed to get nodes in Table form across all namespaces")
  111. framework.Logf("Table: %#v", table)
  112. gomega.Expect(len(table.ColumnDefinitions)).To(gomega.BeNumerically(">=", 2))
  113. gomega.Expect(len(table.Rows)).To(gomega.BeNumerically(">=", 1))
  114. framework.ExpectEqual(len(table.Rows[0].Cells), len(table.ColumnDefinitions))
  115. framework.ExpectEqual(table.ColumnDefinitions[0].Name, "Name")
  116. framework.ExpectNotEqual(table.ResourceVersion, "")
  117. framework.ExpectNotEqual(table.SelfLink, "")
  118. out := printTable(table)
  119. gomega.Expect(out).To(gomega.MatchRegexp("^NAME\\s"))
  120. framework.Logf("Table:\n%s", out)
  121. })
  122. /*
  123. Release : v1.16
  124. Testname: API metadata HTTP return
  125. Description: Issue a HTTP request to the API.
  126. HTTP request MUST return a HTTP status code of 406.
  127. */
  128. framework.ConformanceIt("should return a 406 for a backend which does not implement metadata", func() {
  129. c := f.ClientSet
  130. table := &metav1beta1.Table{}
  131. sar := &authorizationv1.SelfSubjectAccessReview{
  132. Spec: authorizationv1.SelfSubjectAccessReviewSpec{
  133. NonResourceAttributes: &authorizationv1.NonResourceAttributes{
  134. Path: "/",
  135. Verb: "get",
  136. },
  137. },
  138. }
  139. err := c.AuthorizationV1().RESTClient().Post().Resource("selfsubjectaccessreviews").SetHeader("Accept", "application/json;as=Table;v=v1;g=meta.k8s.io").Body(sar).Do(context.TODO()).Into(table)
  140. framework.ExpectError(err, "failed to return error when posting self subject access review: %+v, to a backend that does not implement metadata", sar)
  141. framework.ExpectEqual(err.(apierrors.APIStatus).Status().Code, int32(406))
  142. })
  143. })
  144. func printTable(table *metav1beta1.Table) string {
  145. buf := &bytes.Buffer{}
  146. tw := tabwriter.NewWriter(buf, 5, 8, 1, ' ', 0)
  147. printer := printers.NewTablePrinter(printers.PrintOptions{})
  148. err := printer.PrintObj(table, tw)
  149. framework.ExpectNoError(err, "failed to print table: %+v", table)
  150. tw.Flush()
  151. return buf.String()
  152. }
  153. func newTablePod(podName string) *v1.Pod {
  154. containerName := fmt.Sprintf("%s-container", podName)
  155. port := 8080
  156. pod := &v1.Pod{
  157. ObjectMeta: metav1.ObjectMeta{
  158. Name: podName,
  159. },
  160. Spec: v1.PodSpec{
  161. Containers: []v1.Container{
  162. {
  163. Name: containerName,
  164. Image: imageutils.GetE2EImage(imageutils.Agnhost),
  165. Args: []string{"porter"},
  166. Env: []v1.EnvVar{{Name: fmt.Sprintf("SERVE_PORT_%d", port), Value: "foo"}},
  167. Ports: []v1.ContainerPort{{ContainerPort: int32(port)}},
  168. },
  169. },
  170. RestartPolicy: v1.RestartPolicyNever,
  171. },
  172. }
  173. return pod
  174. }