crd_test.go 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  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 master
  14. import (
  15. "context"
  16. "encoding/json"
  17. "fmt"
  18. "strings"
  19. "testing"
  20. "time"
  21. "github.com/go-openapi/spec"
  22. v1 "k8s.io/api/core/v1"
  23. networkingv1 "k8s.io/api/networking/v1"
  24. apiextensionsv1beta1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1beta1"
  25. apiextensionsclientset "k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset"
  26. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  27. "k8s.io/apimachinery/pkg/runtime/schema"
  28. "k8s.io/apimachinery/pkg/util/wait"
  29. "k8s.io/client-go/dynamic"
  30. "k8s.io/client-go/kubernetes"
  31. kubeapiservertesting "k8s.io/kubernetes/cmd/kube-apiserver/app/testing"
  32. "k8s.io/kubernetes/test/integration/etcd"
  33. "k8s.io/kubernetes/test/integration/framework"
  34. utilpointer "k8s.io/utils/pointer"
  35. )
  36. func TestCRDShadowGroup(t *testing.T) {
  37. result := kubeapiservertesting.StartTestServerOrDie(t, nil, nil, framework.SharedEtcd())
  38. defer result.TearDownFn()
  39. testNamespace := "test-crd-shadow-group"
  40. kubeclient, err := kubernetes.NewForConfig(result.ClientConfig)
  41. if err != nil {
  42. t.Fatalf("Unexpected error: %v", err)
  43. }
  44. if _, err := kubeclient.CoreV1().Namespaces().Create(context.TODO(), (&v1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: testNamespace}}), metav1.CreateOptions{}); err != nil {
  45. t.Fatal(err)
  46. }
  47. apiextensionsclient, err := apiextensionsclientset.NewForConfig(result.ClientConfig)
  48. if err != nil {
  49. t.Fatalf("Unexpected error: %v", err)
  50. }
  51. t.Logf("Creating a NetworkPolicy")
  52. nwPolicy, err := kubeclient.NetworkingV1().NetworkPolicies(testNamespace).Create(context.TODO(), &networkingv1.NetworkPolicy{
  53. ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: testNamespace},
  54. Spec: networkingv1.NetworkPolicySpec{
  55. PodSelector: metav1.LabelSelector{MatchLabels: map[string]string{"foo": "bar"}},
  56. Ingress: []networkingv1.NetworkPolicyIngressRule{},
  57. },
  58. }, metav1.CreateOptions{})
  59. if err != nil {
  60. t.Fatalf("Failed to create NetworkPolicy: %v", err)
  61. }
  62. t.Logf("Trying to shadow networking group")
  63. crd := &apiextensionsv1beta1.CustomResourceDefinition{
  64. ObjectMeta: metav1.ObjectMeta{
  65. Name: "foos." + networkingv1.GroupName,
  66. },
  67. Spec: apiextensionsv1beta1.CustomResourceDefinitionSpec{
  68. Group: networkingv1.GroupName,
  69. Version: networkingv1.SchemeGroupVersion.Version,
  70. Scope: apiextensionsv1beta1.ClusterScoped,
  71. Names: apiextensionsv1beta1.CustomResourceDefinitionNames{
  72. Plural: "foos",
  73. Kind: "Foo",
  74. },
  75. },
  76. }
  77. etcd.CreateTestCRDs(t, apiextensionsclient, true, crd)
  78. // wait to give aggregator time to update
  79. time.Sleep(2 * time.Second)
  80. t.Logf("Checking that we still see the NetworkPolicy")
  81. _, err = kubeclient.NetworkingV1().NetworkPolicies(nwPolicy.Namespace).Get(context.TODO(), nwPolicy.Name, metav1.GetOptions{})
  82. if err != nil {
  83. t.Errorf("Failed to get NetworkPolocy: %v", err)
  84. }
  85. t.Logf("Checking that crd resource does not show up in networking group")
  86. if etcd.CrdExistsInDiscovery(apiextensionsclient, crd) {
  87. t.Errorf("CRD resource shows up in discovery, but shouldn't.")
  88. }
  89. }
  90. func TestCRD(t *testing.T) {
  91. result := kubeapiservertesting.StartTestServerOrDie(t, nil, nil, framework.SharedEtcd())
  92. defer result.TearDownFn()
  93. testNamespace := "test-crd"
  94. kubeclient, err := kubernetes.NewForConfig(result.ClientConfig)
  95. if err != nil {
  96. t.Fatalf("Unexpected error: %v", err)
  97. }
  98. if _, err := kubeclient.CoreV1().Namespaces().Create(context.TODO(), (&v1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: testNamespace}}), metav1.CreateOptions{}); err != nil {
  99. t.Fatal(err)
  100. }
  101. apiextensionsclient, err := apiextensionsclientset.NewForConfig(result.ClientConfig)
  102. if err != nil {
  103. t.Fatalf("Unexpected error: %v", err)
  104. }
  105. t.Logf("Trying to create a custom resource without conflict")
  106. crd := &apiextensionsv1beta1.CustomResourceDefinition{
  107. ObjectMeta: metav1.ObjectMeta{
  108. Name: "foos.cr.bar.com",
  109. },
  110. Spec: apiextensionsv1beta1.CustomResourceDefinitionSpec{
  111. Group: "cr.bar.com",
  112. Version: "v1",
  113. Scope: apiextensionsv1beta1.NamespaceScoped,
  114. Names: apiextensionsv1beta1.CustomResourceDefinitionNames{
  115. Plural: "foos",
  116. Kind: "Foo",
  117. },
  118. },
  119. }
  120. etcd.CreateTestCRDs(t, apiextensionsclient, false, crd)
  121. t.Logf("Trying to access foos.cr.bar.com with dynamic client")
  122. dynamicClient, err := dynamic.NewForConfig(result.ClientConfig)
  123. if err != nil {
  124. t.Fatalf("Unexpected error: %v", err)
  125. }
  126. fooResource := schema.GroupVersionResource{Group: "cr.bar.com", Version: "v1", Resource: "foos"}
  127. _, err = dynamicClient.Resource(fooResource).Namespace(testNamespace).List(metav1.ListOptions{})
  128. if err != nil {
  129. t.Errorf("Failed to list foos.cr.bar.com instances: %v", err)
  130. }
  131. }
  132. func TestCRDOpenAPI(t *testing.T) {
  133. result := kubeapiservertesting.StartTestServerOrDie(t, nil, nil, framework.SharedEtcd())
  134. defer result.TearDownFn()
  135. kubeclient, err := kubernetes.NewForConfig(result.ClientConfig)
  136. if err != nil {
  137. t.Fatalf("Unexpected error: %v", err)
  138. }
  139. apiextensionsclient, err := apiextensionsclientset.NewForConfig(result.ClientConfig)
  140. if err != nil {
  141. t.Fatalf("Unexpected error: %v", err)
  142. }
  143. t.Logf("Trying to create a CustomResourceDefinitions")
  144. nonStructuralCRD := &apiextensionsv1beta1.CustomResourceDefinition{
  145. ObjectMeta: metav1.ObjectMeta{
  146. Name: "foos.nonstructural.cr.bar.com",
  147. },
  148. Spec: apiextensionsv1beta1.CustomResourceDefinitionSpec{
  149. Group: "nonstructural.cr.bar.com",
  150. Version: "v1",
  151. Scope: apiextensionsv1beta1.NamespaceScoped,
  152. Names: apiextensionsv1beta1.CustomResourceDefinitionNames{
  153. Plural: "foos",
  154. Kind: "Foo",
  155. },
  156. Validation: &apiextensionsv1beta1.CustomResourceValidation{
  157. OpenAPIV3Schema: &apiextensionsv1beta1.JSONSchemaProps{
  158. Type: "object",
  159. Properties: map[string]apiextensionsv1beta1.JSONSchemaProps{
  160. "foo": {},
  161. },
  162. },
  163. },
  164. },
  165. }
  166. structuralCRD := &apiextensionsv1beta1.CustomResourceDefinition{
  167. ObjectMeta: metav1.ObjectMeta{
  168. Name: "foos.structural.cr.bar.com",
  169. },
  170. Spec: apiextensionsv1beta1.CustomResourceDefinitionSpec{
  171. Group: "structural.cr.bar.com",
  172. Version: "v1",
  173. Scope: apiextensionsv1beta1.NamespaceScoped,
  174. Names: apiextensionsv1beta1.CustomResourceDefinitionNames{
  175. Plural: "foos",
  176. Kind: "Foo",
  177. },
  178. PreserveUnknownFields: utilpointer.BoolPtr(false),
  179. Validation: &apiextensionsv1beta1.CustomResourceValidation{
  180. OpenAPIV3Schema: &apiextensionsv1beta1.JSONSchemaProps{
  181. Type: "object",
  182. Properties: map[string]apiextensionsv1beta1.JSONSchemaProps{
  183. "foo": {Type: "string"},
  184. },
  185. },
  186. },
  187. },
  188. }
  189. etcd.CreateTestCRDs(t, apiextensionsclient, false, nonStructuralCRD)
  190. etcd.CreateTestCRDs(t, apiextensionsclient, false, structuralCRD)
  191. getPublishedSchema := func(defName string) (*spec.Schema, error) {
  192. bs, err := kubeclient.RESTClient().Get().AbsPath("openapi", "v2").DoRaw(context.TODO())
  193. if err != nil {
  194. return nil, err
  195. }
  196. spec := spec.Swagger{}
  197. if err := json.Unmarshal(bs, &spec); err != nil {
  198. return nil, err
  199. }
  200. if spec.SwaggerProps.Paths == nil {
  201. return nil, nil
  202. }
  203. d, ok := spec.SwaggerProps.Definitions[defName]
  204. if !ok {
  205. return nil, nil
  206. }
  207. return &d, nil
  208. }
  209. waitForSpec := func(crd *apiextensionsv1beta1.CustomResourceDefinition, expectedType string) {
  210. t.Logf(`Waiting for {properties: {"foo": {"type":"%s"}}} to show up in schema`, expectedType)
  211. lastMsg := ""
  212. if err := wait.PollImmediate(500*time.Millisecond, 10*time.Second, func() (bool, error) {
  213. lastMsg = ""
  214. defName := crdDefinitionName(crd)
  215. schema, err := getPublishedSchema(defName)
  216. if err != nil {
  217. lastMsg = err.Error()
  218. return false, nil
  219. }
  220. if schema == nil {
  221. lastMsg = fmt.Sprintf("spec.SwaggerProps.Definitions[%q] not found", defName)
  222. return false, nil
  223. }
  224. p, ok := schema.Properties["foo"]
  225. if !ok {
  226. lastMsg = fmt.Sprintf(`spec.SwaggerProps.Definitions[%q].Properties["foo"] not found`, defName)
  227. return false, nil
  228. }
  229. if !p.Type.Contains(expectedType) {
  230. lastMsg = fmt.Sprintf(`spec.SwaggerProps.Definitions[%q].Properties["foo"].Type should be %q, but got: %q`, defName, expectedType, p.Type)
  231. return false, nil
  232. }
  233. return true, nil
  234. }); err != nil {
  235. t.Fatalf("Failed to see %s OpenAPI spec in discovery: %v, last message: %s", structuralCRD.Name, err, lastMsg)
  236. }
  237. }
  238. t.Logf("Check that structural schema is published")
  239. waitForSpec(structuralCRD, "string")
  240. structuralCRD, err = apiextensionsclient.ApiextensionsV1beta1().CustomResourceDefinitions().Get(context.TODO(), structuralCRD.Name, metav1.GetOptions{})
  241. if err != nil {
  242. t.Fatal(err)
  243. }
  244. prop := structuralCRD.Spec.Validation.OpenAPIV3Schema.Properties["foo"]
  245. prop.Type = "boolean"
  246. structuralCRD.Spec.Validation.OpenAPIV3Schema.Properties["foo"] = prop
  247. if _, err = apiextensionsclient.ApiextensionsV1beta1().CustomResourceDefinitions().Update(context.TODO(), structuralCRD, metav1.UpdateOptions{}); err != nil {
  248. t.Fatal(err)
  249. }
  250. waitForSpec(structuralCRD, "boolean")
  251. t.Logf("Check that non-structural schema is not published")
  252. schema, err := getPublishedSchema(crdDefinitionName(nonStructuralCRD))
  253. if err != nil {
  254. t.Fatal(err)
  255. }
  256. if schema == nil {
  257. t.Fatal("expected a non-nil schema")
  258. }
  259. if foo, ok := schema.Properties["foo"]; ok {
  260. t.Fatalf("unexpected published 'foo' property: %#v", foo)
  261. }
  262. }
  263. func crdDefinitionName(crd *apiextensionsv1beta1.CustomResourceDefinition) string {
  264. sgmts := strings.Split(crd.Spec.Group, ".")
  265. reverse(sgmts)
  266. return strings.Join(append(sgmts, crd.Spec.Version, crd.Spec.Names.Kind), ".")
  267. }
  268. func reverse(s []string) {
  269. for i, j := 0, len(s)-1; i < j; i, j = i+1, j-1 {
  270. s[i], s[j] = s[j], s[i]
  271. }
  272. }