dryrun_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. /*
  2. Copyright 2018 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 dryrun
  14. import (
  15. "context"
  16. "testing"
  17. v1 "k8s.io/api/core/v1"
  18. apiextensionsclientset "k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset"
  19. apierrors "k8s.io/apimachinery/pkg/api/errors"
  20. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  21. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  22. "k8s.io/apimachinery/pkg/runtime/schema"
  23. "k8s.io/apimachinery/pkg/types"
  24. "k8s.io/apimachinery/pkg/util/sets"
  25. "k8s.io/apiserver/pkg/features"
  26. utilfeature "k8s.io/apiserver/pkg/util/feature"
  27. "k8s.io/client-go/dynamic"
  28. "k8s.io/client-go/kubernetes"
  29. featuregatetesting "k8s.io/component-base/featuregate/testing"
  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. )
  34. // Only add kinds to this list when this a virtual resource with get and create verbs that doesn't actually
  35. // store into it's kind. We've used this downstream for mappings before.
  36. var kindWhiteList = sets.NewString()
  37. // namespace used for all tests, do not change this
  38. const testNamespace = "dryrunnamespace"
  39. func DryRunCreateTest(t *testing.T, rsc dynamic.ResourceInterface, obj *unstructured.Unstructured, gvResource schema.GroupVersionResource) {
  40. createdObj, err := rsc.Create(obj, metav1.CreateOptions{DryRun: []string{metav1.DryRunAll}})
  41. if err != nil {
  42. t.Fatalf("failed to dry-run create stub for %s: %#v", gvResource, err)
  43. }
  44. if obj.GroupVersionKind() != createdObj.GroupVersionKind() {
  45. t.Fatalf("created object doesn't have the same gvk as original object: got %v, expected %v",
  46. createdObj.GroupVersionKind(),
  47. obj.GroupVersionKind())
  48. }
  49. if _, err := rsc.Get(obj.GetName(), metav1.GetOptions{}); !apierrors.IsNotFound(err) {
  50. t.Fatalf("object shouldn't exist: %v", err)
  51. }
  52. }
  53. func DryRunPatchTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  54. patch := []byte(`{"metadata":{"annotations":{"patch": "true"}}}`)
  55. obj, err := rsc.Patch(name, types.MergePatchType, patch, metav1.PatchOptions{DryRun: []string{metav1.DryRunAll}})
  56. if err != nil {
  57. t.Fatalf("failed to dry-run patch object: %v", err)
  58. }
  59. if v := obj.GetAnnotations()["patch"]; v != "true" {
  60. t.Fatalf("dry-run patched annotations should be returned, got: %v", obj.GetAnnotations())
  61. }
  62. obj, err = rsc.Get(obj.GetName(), metav1.GetOptions{})
  63. if err != nil {
  64. t.Fatalf("failed to get object: %v", err)
  65. }
  66. if v := obj.GetAnnotations()["patch"]; v == "true" {
  67. t.Fatalf("dry-run patched annotations should not be persisted, got: %v", obj.GetAnnotations())
  68. }
  69. }
  70. func getReplicasOrFail(t *testing.T, obj *unstructured.Unstructured) int64 {
  71. t.Helper()
  72. replicas, found, err := unstructured.NestedInt64(obj.UnstructuredContent(), "spec", "replicas")
  73. if err != nil {
  74. t.Fatalf("failed to get int64 for replicas: %v", err)
  75. }
  76. if !found {
  77. t.Fatal("object doesn't have spec.replicas")
  78. }
  79. return replicas
  80. }
  81. func DryRunScalePatchTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  82. obj, err := rsc.Get(name, metav1.GetOptions{}, "scale")
  83. if apierrors.IsNotFound(err) {
  84. return
  85. }
  86. if err != nil {
  87. t.Fatalf("failed to get object: %v", err)
  88. }
  89. replicas := getReplicasOrFail(t, obj)
  90. patch := []byte(`{"spec":{"replicas":10}}`)
  91. patchedObj, err := rsc.Patch(name, types.MergePatchType, patch, metav1.PatchOptions{DryRun: []string{metav1.DryRunAll}}, "scale")
  92. if err != nil {
  93. t.Fatalf("failed to dry-run patch object: %v", err)
  94. }
  95. if newReplicas := getReplicasOrFail(t, patchedObj); newReplicas != 10 {
  96. t.Fatalf("dry-run patch to replicas didn't return new value: %v", newReplicas)
  97. }
  98. persistedObj, err := rsc.Get(name, metav1.GetOptions{}, "scale")
  99. if err != nil {
  100. t.Fatalf("failed to get scale sub-resource")
  101. }
  102. if newReplicas := getReplicasOrFail(t, persistedObj); newReplicas != replicas {
  103. t.Fatalf("number of replicas changed, expected %v, got %v", replicas, newReplicas)
  104. }
  105. }
  106. func DryRunScaleUpdateTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  107. obj, err := rsc.Get(name, metav1.GetOptions{}, "scale")
  108. if apierrors.IsNotFound(err) {
  109. return
  110. }
  111. if err != nil {
  112. t.Fatalf("failed to get object: %v", err)
  113. }
  114. replicas := getReplicasOrFail(t, obj)
  115. if err := unstructured.SetNestedField(obj.Object, int64(10), "spec", "replicas"); err != nil {
  116. t.Fatalf("failed to set spec.replicas: %v", err)
  117. }
  118. updatedObj, err := rsc.Update(obj, metav1.UpdateOptions{DryRun: []string{metav1.DryRunAll}}, "scale")
  119. if err != nil {
  120. t.Fatalf("failed to dry-run update scale sub-resource: %v", err)
  121. }
  122. if newReplicas := getReplicasOrFail(t, updatedObj); newReplicas != 10 {
  123. t.Fatalf("dry-run update to replicas didn't return new value: %v", newReplicas)
  124. }
  125. persistedObj, err := rsc.Get(name, metav1.GetOptions{}, "scale")
  126. if err != nil {
  127. t.Fatalf("failed to get scale sub-resource")
  128. }
  129. if newReplicas := getReplicasOrFail(t, persistedObj); newReplicas != replicas {
  130. t.Fatalf("number of replicas changed, expected %v, got %v", replicas, newReplicas)
  131. }
  132. }
  133. func DryRunUpdateTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  134. var err error
  135. var obj *unstructured.Unstructured
  136. for i := 0; i < 3; i++ {
  137. obj, err = rsc.Get(name, metav1.GetOptions{})
  138. if err != nil {
  139. t.Fatalf("failed to retrieve object: %v", err)
  140. }
  141. obj.SetAnnotations(map[string]string{"update": "true"})
  142. obj, err = rsc.Update(obj, metav1.UpdateOptions{DryRun: []string{metav1.DryRunAll}})
  143. if err == nil || !apierrors.IsConflict(err) {
  144. break
  145. }
  146. }
  147. if err != nil {
  148. t.Fatalf("failed to dry-run update resource: %v", err)
  149. }
  150. if v := obj.GetAnnotations()["update"]; v != "true" {
  151. t.Fatalf("dry-run updated annotations should be returned, got: %v", obj.GetAnnotations())
  152. }
  153. obj, err = rsc.Get(obj.GetName(), metav1.GetOptions{})
  154. if err != nil {
  155. t.Fatalf("failed to get object: %v", err)
  156. }
  157. if v := obj.GetAnnotations()["update"]; v == "true" {
  158. t.Fatalf("dry-run updated annotations should not be persisted, got: %v", obj.GetAnnotations())
  159. }
  160. }
  161. func DryRunDeleteCollectionTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  162. err := rsc.DeleteCollection(&metav1.DeleteOptions{DryRun: []string{metav1.DryRunAll}}, metav1.ListOptions{})
  163. if err != nil {
  164. t.Fatalf("dry-run delete collection failed: %v", err)
  165. }
  166. obj, err := rsc.Get(name, metav1.GetOptions{})
  167. if err != nil {
  168. t.Fatalf("failed to get object: %v", err)
  169. }
  170. ts := obj.GetDeletionTimestamp()
  171. if ts != nil {
  172. t.Fatalf("object has a deletion timestamp after dry-run delete collection")
  173. }
  174. }
  175. func DryRunDeleteTest(t *testing.T, rsc dynamic.ResourceInterface, name string) {
  176. err := rsc.Delete(name, &metav1.DeleteOptions{DryRun: []string{metav1.DryRunAll}})
  177. if err != nil {
  178. t.Fatalf("dry-run delete failed: %v", err)
  179. }
  180. obj, err := rsc.Get(name, metav1.GetOptions{})
  181. if err != nil {
  182. t.Fatalf("failed to get object: %v", err)
  183. }
  184. ts := obj.GetDeletionTimestamp()
  185. if ts != nil {
  186. t.Fatalf("object has a deletion timestamp after dry-run delete")
  187. }
  188. }
  189. // TestDryRun tests dry-run on all types.
  190. func TestDryRun(t *testing.T) {
  191. defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.DryRun, true)()
  192. // start API server
  193. s, err := kubeapiservertesting.StartTestServer(t, kubeapiservertesting.NewDefaultTestServerOptions(), []string{
  194. "--disable-admission-plugins=ServiceAccount,StorageObjectInUseProtection",
  195. "--runtime-config=api/all=true",
  196. }, framework.SharedEtcd())
  197. if err != nil {
  198. t.Fatal(err)
  199. }
  200. defer s.TearDownFn()
  201. client, err := kubernetes.NewForConfig(s.ClientConfig)
  202. if err != nil {
  203. t.Fatal(err)
  204. }
  205. dynamicClient, err := dynamic.NewForConfig(s.ClientConfig)
  206. if err != nil {
  207. t.Fatal(err)
  208. }
  209. // create CRDs so we can make sure that custom resources do not get lost
  210. etcd.CreateTestCRDs(t, apiextensionsclientset.NewForConfigOrDie(s.ClientConfig), false, etcd.GetCustomResourceDefinitionData()...)
  211. if _, err := client.CoreV1().Namespaces().Create(context.TODO(), &v1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: testNamespace}}, metav1.CreateOptions{}); err != nil {
  212. t.Fatal(err)
  213. }
  214. dryrunData := etcd.GetEtcdStorageData()
  215. // dry run specific stub overrides
  216. for resource, stub := range map[schema.GroupVersionResource]string{
  217. // need to change event's namespace field to match dry run test
  218. gvr("", "v1", "events"): `{"involvedObject": {"namespace": "dryrunnamespace"}, "message": "some data here", "metadata": {"name": "event1"}}`,
  219. } {
  220. data := dryrunData[resource]
  221. data.Stub = stub
  222. dryrunData[resource] = data
  223. }
  224. // gather resources to test
  225. _, resources, err := client.Discovery().ServerGroupsAndResources()
  226. if err != nil {
  227. t.Fatalf("Failed to get ServerGroupsAndResources with error: %+v", err)
  228. }
  229. for _, resourceToTest := range etcd.GetResources(t, resources) {
  230. t.Run(resourceToTest.Mapping.Resource.String(), func(t *testing.T) {
  231. mapping := resourceToTest.Mapping
  232. gvk := resourceToTest.Mapping.GroupVersionKind
  233. gvResource := resourceToTest.Mapping.Resource
  234. kind := gvk.Kind
  235. if kindWhiteList.Has(kind) {
  236. t.Skip("whitelisted")
  237. }
  238. testData, hasTest := dryrunData[gvResource]
  239. if !hasTest {
  240. t.Fatalf("no test data for %s. Please add a test for your new type to etcd.GetEtcdStorageData().", gvResource)
  241. }
  242. rsc, obj, err := etcd.JSONToUnstructured(testData.Stub, testNamespace, mapping, dynamicClient)
  243. if err != nil {
  244. t.Fatalf("failed to unmarshal stub (%v): %v", testData.Stub, err)
  245. }
  246. name := obj.GetName()
  247. DryRunCreateTest(t, rsc, obj, gvResource)
  248. if _, err := rsc.Create(obj, metav1.CreateOptions{}); err != nil {
  249. t.Fatalf("failed to create stub for %s: %#v", gvResource, err)
  250. }
  251. DryRunUpdateTest(t, rsc, name)
  252. DryRunPatchTest(t, rsc, name)
  253. DryRunScalePatchTest(t, rsc, name)
  254. DryRunScaleUpdateTest(t, rsc, name)
  255. if resourceToTest.HasDeleteCollection {
  256. DryRunDeleteCollectionTest(t, rsc, name)
  257. }
  258. DryRunDeleteTest(t, rsc, name)
  259. if err = rsc.Delete(obj.GetName(), metav1.NewDeleteOptions(0)); err != nil {
  260. t.Fatalf("deleting final object failed: %v", err)
  261. }
  262. })
  263. }
  264. }
  265. func gvr(g, v, r string) schema.GroupVersionResource {
  266. return schema.GroupVersionResource{Group: g, Version: v, Resource: r}
  267. }