crd_test.go 8.4 KB

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