garbage_collector.go 43 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106
  1. /*
  2. Copyright 2016 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. "fmt"
  16. "sync/atomic"
  17. "time"
  18. apps "k8s.io/api/apps/v1"
  19. batchv1 "k8s.io/api/batch/v1"
  20. batchv1beta1 "k8s.io/api/batch/v1beta1"
  21. "k8s.io/api/core/v1"
  22. apiextensionsv1beta1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1beta1"
  23. apiextensionsclientset "k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset"
  24. apiextensionstestserver "k8s.io/apiextensions-apiserver/test/integration/fixtures"
  25. "k8s.io/apimachinery/pkg/api/errors"
  26. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  27. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  28. "k8s.io/apimachinery/pkg/runtime/schema"
  29. "k8s.io/apimachinery/pkg/types"
  30. utilerrors "k8s.io/apimachinery/pkg/util/errors"
  31. "k8s.io/apimachinery/pkg/util/wait"
  32. "k8s.io/apiserver/pkg/storage/names"
  33. clientset "k8s.io/client-go/kubernetes"
  34. "k8s.io/kubernetes/test/e2e/framework"
  35. e2elog "k8s.io/kubernetes/test/e2e/framework/log"
  36. "k8s.io/kubernetes/test/e2e/framework/metrics"
  37. "github.com/onsi/ginkgo"
  38. imageutils "k8s.io/kubernetes/test/utils/image"
  39. )
  40. // estimateMaximumPods estimates how many pods the cluster can handle
  41. // with some wiggle room, to prevent pods being unable to schedule due
  42. // to max pod constraints.
  43. func estimateMaximumPods(c clientset.Interface, min, max int32) int32 {
  44. availablePods := int32(0)
  45. for _, node := range framework.GetReadySchedulableNodesOrDie(c).Items {
  46. if q, ok := node.Status.Allocatable["pods"]; ok {
  47. if num, ok := q.AsInt64(); ok {
  48. availablePods += int32(num)
  49. continue
  50. }
  51. }
  52. // best guess per node, since default maxPerCore is 10 and most nodes have at least
  53. // one core.
  54. availablePods += 10
  55. }
  56. //avoid creating exactly max pods
  57. availablePods *= 8 / 10
  58. // bound the top and bottom
  59. if availablePods > max {
  60. availablePods = max
  61. }
  62. if availablePods < min {
  63. availablePods = min
  64. }
  65. return availablePods
  66. }
  67. func getForegroundOptions() *metav1.DeleteOptions {
  68. policy := metav1.DeletePropagationForeground
  69. return &metav1.DeleteOptions{PropagationPolicy: &policy}
  70. }
  71. func getBackgroundOptions() *metav1.DeleteOptions {
  72. policy := metav1.DeletePropagationBackground
  73. return &metav1.DeleteOptions{PropagationPolicy: &policy}
  74. }
  75. func getOrphanOptions() *metav1.DeleteOptions {
  76. policy := metav1.DeletePropagationOrphan
  77. return &metav1.DeleteOptions{PropagationPolicy: &policy}
  78. }
  79. var (
  80. zero = int64(0)
  81. lablecount = int64(0)
  82. // CronJobGroupVersionResource unambiguously identifies a CronJob resource.
  83. CronJobGroupVersionResource = schema.GroupVersionResource{Group: batchv1beta1.GroupName, Version: "v1beta1", Resource: "cronjobs"}
  84. )
  85. func getPodTemplateSpec(labels map[string]string) v1.PodTemplateSpec {
  86. return v1.PodTemplateSpec{
  87. ObjectMeta: metav1.ObjectMeta{
  88. Labels: labels,
  89. },
  90. Spec: v1.PodSpec{
  91. TerminationGracePeriodSeconds: &zero,
  92. Containers: []v1.Container{
  93. {
  94. Name: "nginx",
  95. Image: imageutils.GetE2EImage(imageutils.Nginx),
  96. },
  97. },
  98. },
  99. }
  100. }
  101. func newOwnerDeployment(f *framework.Framework, deploymentName string, labels map[string]string) *apps.Deployment {
  102. replicas := int32(2)
  103. return &apps.Deployment{
  104. ObjectMeta: metav1.ObjectMeta{
  105. Name: deploymentName,
  106. },
  107. Spec: apps.DeploymentSpec{
  108. Replicas: &replicas,
  109. Selector: &metav1.LabelSelector{MatchLabels: labels},
  110. Strategy: apps.DeploymentStrategy{
  111. Type: apps.RollingUpdateDeploymentStrategyType,
  112. },
  113. Template: getPodTemplateSpec(labels),
  114. },
  115. }
  116. }
  117. func newOwnerRC(f *framework.Framework, name string, replicas int32, labels map[string]string) *v1.ReplicationController {
  118. template := getPodTemplateSpec(labels)
  119. return &v1.ReplicationController{
  120. TypeMeta: metav1.TypeMeta{
  121. Kind: "ReplicationController",
  122. APIVersion: "v1",
  123. },
  124. ObjectMeta: metav1.ObjectMeta{
  125. Namespace: f.Namespace.Name,
  126. Name: name,
  127. },
  128. Spec: v1.ReplicationControllerSpec{
  129. Replicas: &replicas,
  130. Selector: labels,
  131. Template: &template,
  132. },
  133. }
  134. }
  135. func newGCPod(name string) *v1.Pod {
  136. return &v1.Pod{
  137. TypeMeta: metav1.TypeMeta{
  138. Kind: "Pod",
  139. APIVersion: "v1",
  140. },
  141. ObjectMeta: metav1.ObjectMeta{
  142. Name: name,
  143. },
  144. Spec: v1.PodSpec{
  145. TerminationGracePeriodSeconds: new(int64),
  146. Containers: []v1.Container{
  147. {
  148. Name: "nginx",
  149. Image: imageutils.GetE2EImage(imageutils.Nginx),
  150. },
  151. },
  152. },
  153. }
  154. }
  155. // verifyRemainingObjects verifies if the number of remaining objects.
  156. // It returns error if the communication with the API server fails.
  157. func verifyRemainingObjects(f *framework.Framework, objects map[string]int) (bool, error) {
  158. var ret = true
  159. for object, num := range objects {
  160. switch object {
  161. case "Pods":
  162. pods, err := f.ClientSet.CoreV1().Pods(f.Namespace.Name).List(metav1.ListOptions{})
  163. if err != nil {
  164. return false, fmt.Errorf("failed to list pods: %v", err)
  165. }
  166. if len(pods.Items) != num {
  167. ret = false
  168. ginkgo.By(fmt.Sprintf("expected %d pods, got %d pods", num, len(pods.Items)))
  169. }
  170. case "Deployments":
  171. deployments, err := f.ClientSet.AppsV1().Deployments(f.Namespace.Name).List(metav1.ListOptions{})
  172. if err != nil {
  173. return false, fmt.Errorf("failed to list deployments: %v", err)
  174. }
  175. if len(deployments.Items) != num {
  176. ret = false
  177. ginkgo.By(fmt.Sprintf("expected %d Deployments, got %d Deployments", num, len(deployments.Items)))
  178. }
  179. case "ReplicaSets":
  180. rs, err := f.ClientSet.AppsV1().ReplicaSets(f.Namespace.Name).List(metav1.ListOptions{})
  181. if err != nil {
  182. return false, fmt.Errorf("failed to list rs: %v", err)
  183. }
  184. if len(rs.Items) != num {
  185. ret = false
  186. ginkgo.By(fmt.Sprintf("expected %d rs, got %d rs", num, len(rs.Items)))
  187. }
  188. case "ReplicationControllers":
  189. rcs, err := f.ClientSet.CoreV1().ReplicationControllers(f.Namespace.Name).List(metav1.ListOptions{})
  190. if err != nil {
  191. return false, fmt.Errorf("failed to list replication controllers: %v", err)
  192. }
  193. if len(rcs.Items) != num {
  194. ret = false
  195. ginkgo.By(fmt.Sprintf("expected %d RCs, got %d RCs", num, len(rcs.Items)))
  196. }
  197. case "CronJobs":
  198. cronJobs, err := f.ClientSet.BatchV1beta1().CronJobs(f.Namespace.Name).List(metav1.ListOptions{})
  199. if err != nil {
  200. return false, fmt.Errorf("failed to list cronjobs: %v", err)
  201. }
  202. if len(cronJobs.Items) != num {
  203. ret = false
  204. ginkgo.By(fmt.Sprintf("expected %d cronjobs, got %d cronjobs", num, len(cronJobs.Items)))
  205. }
  206. case "Jobs":
  207. jobs, err := f.ClientSet.BatchV1().Jobs(f.Namespace.Name).List(metav1.ListOptions{})
  208. if err != nil {
  209. return false, fmt.Errorf("failed to list jobs: %v", err)
  210. }
  211. if len(jobs.Items) != num {
  212. ret = false
  213. ginkgo.By(fmt.Sprintf("expected %d jobs, got %d jobs", num, len(jobs.Items)))
  214. }
  215. default:
  216. return false, fmt.Errorf("object %s is not supported", object)
  217. }
  218. }
  219. return ret, nil
  220. }
  221. func gatherMetrics(f *framework.Framework) {
  222. ginkgo.By("Gathering metrics")
  223. var summary framework.TestDataSummary
  224. grabber, err := metrics.NewMetricsGrabber(f.ClientSet, f.KubemarkExternalClusterClientSet, false, false, true, false, false)
  225. if err != nil {
  226. e2elog.Logf("Failed to create MetricsGrabber. Skipping metrics gathering.")
  227. } else {
  228. received, err := grabber.Grab()
  229. if err != nil {
  230. e2elog.Logf("MetricsGrabber failed grab metrics. Skipping metrics gathering.")
  231. } else {
  232. summary = (*framework.MetricsForE2E)(&received)
  233. e2elog.Logf(summary.PrintHumanReadable())
  234. }
  235. }
  236. }
  237. func newCronJob(name, schedule string) *batchv1beta1.CronJob {
  238. parallelism := int32(1)
  239. completions := int32(1)
  240. return &batchv1beta1.CronJob{
  241. ObjectMeta: metav1.ObjectMeta{
  242. Name: name,
  243. },
  244. TypeMeta: metav1.TypeMeta{
  245. Kind: "CronJob",
  246. },
  247. Spec: batchv1beta1.CronJobSpec{
  248. Schedule: schedule,
  249. JobTemplate: batchv1beta1.JobTemplateSpec{
  250. Spec: batchv1.JobSpec{
  251. Parallelism: &parallelism,
  252. Completions: &completions,
  253. Template: v1.PodTemplateSpec{
  254. Spec: v1.PodSpec{
  255. RestartPolicy: v1.RestartPolicyOnFailure,
  256. TerminationGracePeriodSeconds: &zero,
  257. Containers: []v1.Container{
  258. {
  259. Name: "c",
  260. Image: imageutils.GetE2EImage(imageutils.BusyBox),
  261. Command: []string{"sleep", "300"},
  262. },
  263. },
  264. },
  265. },
  266. },
  267. },
  268. },
  269. }
  270. }
  271. // getUniqLabel returns a UniqLabel based on labeLkey and labelvalue.
  272. func getUniqLabel(labelkey, labelvalue string) map[string]string {
  273. count := atomic.AddInt64(&lablecount, 1)
  274. uniqlabelkey := fmt.Sprintf("%s-%05d", labelkey, count)
  275. uniqlabelvalue := fmt.Sprintf("%s-%05d", labelvalue, count)
  276. return map[string]string{uniqlabelkey: uniqlabelvalue}
  277. }
  278. var _ = SIGDescribe("Garbage collector", func() {
  279. f := framework.NewDefaultFramework("gc")
  280. /*
  281. Release : v1.9
  282. Testname: Garbage Collector, delete replication controller, propagation policy background
  283. Description: Create a replication controller with 2 Pods. Once RC is created and the first Pod is created, delete RC with deleteOptions.PropagationPolicy set to Background. Deleting the Replication Controller MUST cause pods created by that RC to be deleted.
  284. */
  285. framework.ConformanceIt("should delete pods created by rc when not orphaning", func() {
  286. clientSet := f.ClientSet
  287. rcClient := clientSet.CoreV1().ReplicationControllers(f.Namespace.Name)
  288. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  289. rcName := "simpletest.rc"
  290. uniqLabels := getUniqLabel("gctest", "delete_pods")
  291. rc := newOwnerRC(f, rcName, 2, uniqLabels)
  292. ginkgo.By("create the rc")
  293. rc, err := rcClient.Create(rc)
  294. if err != nil {
  295. framework.Failf("Failed to create replication controller: %v", err)
  296. }
  297. // wait for rc to create some pods
  298. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  299. pods, err := podClient.List(metav1.ListOptions{})
  300. if err != nil {
  301. return false, fmt.Errorf("Failed to list pods: %v", err)
  302. }
  303. // We intentionally don't wait the number of pods to reach
  304. // rc.Spec.Replicas. We want to see if the garbage collector and the
  305. // rc manager work properly if the rc is deleted before it reaches
  306. // stasis.
  307. if len(pods.Items) > 0 {
  308. return true, nil
  309. }
  310. return false, nil
  311. }); err != nil {
  312. framework.Failf("failed to wait for the rc to create some pods: %v", err)
  313. }
  314. ginkgo.By("delete the rc")
  315. deleteOptions := getBackgroundOptions()
  316. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(rc.UID))
  317. if err := rcClient.Delete(rc.ObjectMeta.Name, deleteOptions); err != nil {
  318. framework.Failf("failed to delete the rc: %v", err)
  319. }
  320. ginkgo.By("wait for all pods to be garbage collected")
  321. // wait for the RCs and Pods to reach the expected numbers.
  322. if err := wait.Poll(5*time.Second, 60*time.Second, func() (bool, error) {
  323. objects := map[string]int{"ReplicationControllers": 0, "Pods": 0}
  324. return verifyRemainingObjects(f, objects)
  325. }); err != nil {
  326. framework.Failf("failed to wait for all pods to be deleted: %v", err)
  327. remainingPods, err := podClient.List(metav1.ListOptions{})
  328. if err != nil {
  329. framework.Failf("failed to list pods post mortem: %v", err)
  330. } else {
  331. framework.Failf("remaining pods are: %#v", remainingPods)
  332. }
  333. }
  334. gatherMetrics(f)
  335. })
  336. /*
  337. Release : v1.9
  338. Testname: Garbage Collector, delete replication controller, propagation policy orphan
  339. Description: Create a replication controller with maximum allocatable Pods between 10 and 100 replicas. Once RC is created and the all Pods are created, delete RC with deleteOptions.PropagationPolicy set to Orphan. Deleting the Replication Controller MUST cause pods created by that RC to be orphaned.
  340. */
  341. framework.ConformanceIt("should orphan pods created by rc if delete options say so", func() {
  342. clientSet := f.ClientSet
  343. rcClient := clientSet.CoreV1().ReplicationControllers(f.Namespace.Name)
  344. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  345. rcName := "simpletest.rc"
  346. uniqLabels := getUniqLabel("gctest", "orphan_pods")
  347. rc := newOwnerRC(f, rcName, estimateMaximumPods(clientSet, 10, 100), uniqLabels)
  348. ginkgo.By("create the rc")
  349. rc, err := rcClient.Create(rc)
  350. if err != nil {
  351. framework.Failf("Failed to create replication controller: %v", err)
  352. }
  353. // wait for rc to create pods
  354. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  355. rc, err := rcClient.Get(rc.Name, metav1.GetOptions{})
  356. if err != nil {
  357. return false, fmt.Errorf("Failed to get rc: %v", err)
  358. }
  359. if rc.Status.Replicas == *rc.Spec.Replicas {
  360. return true, nil
  361. }
  362. return false, nil
  363. }); err != nil {
  364. framework.Failf("failed to wait for the rc.Status.Replicas to reach rc.Spec.Replicas: %v", err)
  365. }
  366. ginkgo.By("delete the rc")
  367. deleteOptions := getOrphanOptions()
  368. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(rc.UID))
  369. if err := rcClient.Delete(rc.ObjectMeta.Name, deleteOptions); err != nil {
  370. framework.Failf("failed to delete the rc: %v", err)
  371. }
  372. ginkgo.By("wait for the rc to be deleted")
  373. // Orphaning the 100 pods takes 100 PATCH operations. The default qps of
  374. // a client is 5. If the qps is saturated, it will take 20s to orphan
  375. // the pods. However, apiserver takes hundreds of ms to finish one
  376. // PATCH, and the gc sends the patching in a single thread, so the
  377. // actual qps is less than 5. Also, the e2e tests are running in
  378. // parallel, the GC controller might get distracted by other tests.
  379. // According to the test logs, 120s is enough time.
  380. if err := wait.Poll(5*time.Second, 120*time.Second, func() (bool, error) {
  381. rcs, err := rcClient.List(metav1.ListOptions{})
  382. if err != nil {
  383. return false, fmt.Errorf("Failed to list rcs: %v", err)
  384. }
  385. if len(rcs.Items) != 0 {
  386. return false, nil
  387. }
  388. return true, nil
  389. }); err != nil {
  390. framework.Failf("%v", err)
  391. }
  392. ginkgo.By("wait for 30 seconds to see if the garbage collector mistakenly deletes the pods")
  393. time.Sleep(30 * time.Second)
  394. pods, err := podClient.List(metav1.ListOptions{})
  395. if err != nil {
  396. framework.Failf("Failed to list pods: %v", err)
  397. }
  398. if e, a := int(*(rc.Spec.Replicas)), len(pods.Items); e != a {
  399. framework.Failf("expect %d pods, got %d pods", e, a)
  400. }
  401. gatherMetrics(f)
  402. })
  403. // deleteOptions.OrphanDependents is deprecated in 1.7 and preferred to use the PropagationPolicy.
  404. // Discussion is tracked under https://github.com/kubernetes/kubernetes/issues/65427 to promote for conformance in future.
  405. ginkgo.It("should orphan pods created by rc if deleteOptions.OrphanDependents is nil", func() {
  406. clientSet := f.ClientSet
  407. rcClient := clientSet.CoreV1().ReplicationControllers(f.Namespace.Name)
  408. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  409. rcName := "simpletest.rc"
  410. uniqLabels := getUniqLabel("gctest", "orphan_pods_nil_option")
  411. rc := newOwnerRC(f, rcName, 2, uniqLabels)
  412. ginkgo.By("create the rc")
  413. rc, err := rcClient.Create(rc)
  414. if err != nil {
  415. framework.Failf("Failed to create replication controller: %v", err)
  416. }
  417. // wait for rc to create some pods
  418. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  419. rc, err := rcClient.Get(rc.Name, metav1.GetOptions{})
  420. if err != nil {
  421. return false, fmt.Errorf("Failed to get rc: %v", err)
  422. }
  423. if rc.Status.Replicas == *rc.Spec.Replicas {
  424. return true, nil
  425. }
  426. return false, nil
  427. }); err != nil {
  428. framework.Failf("failed to wait for the rc.Status.Replicas to reach rc.Spec.Replicas: %v", err)
  429. }
  430. ginkgo.By("delete the rc")
  431. deleteOptions := &metav1.DeleteOptions{}
  432. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(rc.UID))
  433. if err := rcClient.Delete(rc.ObjectMeta.Name, deleteOptions); err != nil {
  434. framework.Failf("failed to delete the rc: %v", err)
  435. }
  436. ginkgo.By("wait for 30 seconds to see if the garbage collector mistakenly deletes the pods")
  437. time.Sleep(30 * time.Second)
  438. pods, err := podClient.List(metav1.ListOptions{})
  439. if err != nil {
  440. framework.Failf("Failed to list pods: %v", err)
  441. }
  442. if e, a := int(*(rc.Spec.Replicas)), len(pods.Items); e != a {
  443. framework.Failf("expect %d pods, got %d pods", e, a)
  444. }
  445. gatherMetrics(f)
  446. })
  447. /*
  448. Release : v1.9
  449. Testname: Garbage Collector, delete deployment, propagation policy background
  450. Description: Create a deployment with a replicaset. Once replicaset is created , delete the deployment with deleteOptions.PropagationPolicy set to Background. Deleting the deployment MUST delete the replicaset created by the deployment and also the Pods that belong to the deployments MUST be deleted.
  451. */
  452. framework.ConformanceIt("should delete RS created by deployment when not orphaning", func() {
  453. clientSet := f.ClientSet
  454. deployClient := clientSet.AppsV1().Deployments(f.Namespace.Name)
  455. rsClient := clientSet.AppsV1().ReplicaSets(f.Namespace.Name)
  456. deploymentName := "simpletest.deployment"
  457. uniqLabels := getUniqLabel("gctest", "delete_rs")
  458. deployment := newOwnerDeployment(f, deploymentName, uniqLabels)
  459. ginkgo.By("create the deployment")
  460. createdDeployment, err := deployClient.Create(deployment)
  461. if err != nil {
  462. framework.Failf("Failed to create deployment: %v", err)
  463. }
  464. // wait for deployment to create some rs
  465. ginkgo.By("Wait for the Deployment to create new ReplicaSet")
  466. err = wait.PollImmediate(500*time.Millisecond, 1*time.Minute, func() (bool, error) {
  467. rsList, err := rsClient.List(metav1.ListOptions{})
  468. if err != nil {
  469. return false, fmt.Errorf("Failed to list rs: %v", err)
  470. }
  471. return len(rsList.Items) > 0, nil
  472. })
  473. if err != nil {
  474. framework.Failf("Failed to wait for the Deployment to create some ReplicaSet: %v", err)
  475. }
  476. ginkgo.By("delete the deployment")
  477. deleteOptions := getBackgroundOptions()
  478. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(createdDeployment.UID))
  479. if err := deployClient.Delete(deployment.ObjectMeta.Name, deleteOptions); err != nil {
  480. framework.Failf("failed to delete the deployment: %v", err)
  481. }
  482. ginkgo.By("wait for all rs to be garbage collected")
  483. err = wait.PollImmediate(500*time.Millisecond, 1*time.Minute, func() (bool, error) {
  484. objects := map[string]int{"Deployments": 0, "ReplicaSets": 0, "Pods": 0}
  485. return verifyRemainingObjects(f, objects)
  486. })
  487. if err != nil {
  488. errList := make([]error, 0)
  489. errList = append(errList, err)
  490. remainingRSs, err := rsClient.List(metav1.ListOptions{})
  491. if err != nil {
  492. errList = append(errList, fmt.Errorf("failed to list RSs post mortem: %v", err))
  493. } else {
  494. errList = append(errList, fmt.Errorf("remaining rs are: %#v", remainingRSs))
  495. }
  496. aggregatedError := utilerrors.NewAggregate(errList)
  497. framework.Failf("Failed to wait for all rs to be garbage collected: %v", aggregatedError)
  498. }
  499. gatherMetrics(f)
  500. })
  501. /*
  502. Release : v1.9
  503. Testname: Garbage Collector, delete deployment, propagation policy orphan
  504. Description: Create a deployment with a replicaset. Once replicaset is created , delete the deployment with deleteOptions.PropagationPolicy set to Orphan. Deleting the deployment MUST cause the replicaset created by the deployment to be orphaned, also the Pods created by the deployments MUST be orphaned.
  505. */
  506. framework.ConformanceIt("should orphan RS created by deployment when deleteOptions.PropagationPolicy is Orphan", func() {
  507. clientSet := f.ClientSet
  508. deployClient := clientSet.AppsV1().Deployments(f.Namespace.Name)
  509. rsClient := clientSet.AppsV1().ReplicaSets(f.Namespace.Name)
  510. deploymentName := "simpletest.deployment"
  511. uniqLabels := getUniqLabel("gctest", "orphan_rs")
  512. deployment := newOwnerDeployment(f, deploymentName, uniqLabels)
  513. ginkgo.By("create the deployment")
  514. createdDeployment, err := deployClient.Create(deployment)
  515. if err != nil {
  516. framework.Failf("Failed to create deployment: %v", err)
  517. }
  518. // wait for deployment to create some rs
  519. ginkgo.By("Wait for the Deployment to create new ReplicaSet")
  520. err = wait.PollImmediate(500*time.Millisecond, 1*time.Minute, func() (bool, error) {
  521. rsList, err := rsClient.List(metav1.ListOptions{})
  522. if err != nil {
  523. return false, fmt.Errorf("Failed to list rs: %v", err)
  524. }
  525. return len(rsList.Items) > 0, nil
  526. })
  527. if err != nil {
  528. framework.Failf("Failed to wait for the Deployment to create some ReplicaSet: %v", err)
  529. }
  530. ginkgo.By("delete the deployment")
  531. deleteOptions := getOrphanOptions()
  532. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(createdDeployment.UID))
  533. if err := deployClient.Delete(deployment.ObjectMeta.Name, deleteOptions); err != nil {
  534. framework.Failf("failed to delete the deployment: %v", err)
  535. }
  536. ginkgo.By("wait for 30 seconds to see if the garbage collector mistakenly deletes the rs")
  537. time.Sleep(30 * time.Second)
  538. objects := map[string]int{"Deployments": 0, "ReplicaSets": 1, "Pods": 2}
  539. ok, err := verifyRemainingObjects(f, objects)
  540. if err != nil {
  541. framework.Failf("Unexpected error while verifying remaining deployments, rs, and pods: %v", err)
  542. }
  543. if !ok {
  544. errList := make([]error, 0)
  545. remainingRSs, err := rsClient.List(metav1.ListOptions{})
  546. if err != nil {
  547. errList = append(errList, fmt.Errorf("failed to list RSs post mortem: %v", err))
  548. } else {
  549. errList = append(errList, fmt.Errorf("remaining rs post mortem: %#v", remainingRSs))
  550. }
  551. remainingDSs, err := deployClient.List(metav1.ListOptions{})
  552. if err != nil {
  553. errList = append(errList, fmt.Errorf("failed to list Deployments post mortem: %v", err))
  554. } else {
  555. errList = append(errList, fmt.Errorf("remaining deployment's post mortem: %#v", remainingDSs))
  556. }
  557. aggregatedError := utilerrors.NewAggregate(errList)
  558. framework.Failf("Failed to verify remaining deployments, rs, and pods: %v", aggregatedError)
  559. }
  560. rs, err := clientSet.AppsV1().ReplicaSets(f.Namespace.Name).List(metav1.ListOptions{})
  561. if err != nil {
  562. framework.Failf("Failed to list ReplicaSet %v", err)
  563. }
  564. for _, replicaSet := range rs.Items {
  565. if metav1.GetControllerOf(&replicaSet.ObjectMeta) != nil {
  566. framework.Failf("Found ReplicaSet with non nil ownerRef %v", replicaSet)
  567. }
  568. }
  569. gatherMetrics(f)
  570. })
  571. /*
  572. Release : v1.9
  573. Testname: Garbage Collector, delete replication controller, after owned pods
  574. Description: Create a replication controller with maximum allocatable Pods between 10 and 100 replicas. Once RC is created and the all Pods are created, delete RC with deleteOptions.PropagationPolicy set to Foreground. Deleting the Replication Controller MUST cause pods created by that RC to be deleted before the RC is deleted.
  575. */
  576. framework.ConformanceIt("should keep the rc around until all its pods are deleted if the deleteOptions says so", func() {
  577. clientSet := f.ClientSet
  578. rcClient := clientSet.CoreV1().ReplicationControllers(f.Namespace.Name)
  579. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  580. rcName := "simpletest.rc"
  581. uniqLabels := getUniqLabel("gctest", "delete_pods_foreground")
  582. rc := newOwnerRC(f, rcName, estimateMaximumPods(clientSet, 10, 100), uniqLabels)
  583. ginkgo.By("create the rc")
  584. rc, err := rcClient.Create(rc)
  585. if err != nil {
  586. framework.Failf("Failed to create replication controller: %v", err)
  587. }
  588. // wait for rc to create pods
  589. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  590. rc, err := rcClient.Get(rc.Name, metav1.GetOptions{})
  591. if err != nil {
  592. return false, fmt.Errorf("Failed to get rc: %v", err)
  593. }
  594. if rc.Status.Replicas == *rc.Spec.Replicas {
  595. return true, nil
  596. }
  597. return false, nil
  598. }); err != nil {
  599. framework.Failf("failed to wait for the rc.Status.Replicas to reach rc.Spec.Replicas: %v", err)
  600. }
  601. ginkgo.By("delete the rc")
  602. deleteOptions := getForegroundOptions()
  603. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(rc.UID))
  604. if err := rcClient.Delete(rc.ObjectMeta.Name, deleteOptions); err != nil {
  605. framework.Failf("failed to delete the rc: %v", err)
  606. }
  607. ginkgo.By("wait for the rc to be deleted")
  608. // default client QPS is 20, deleting each pod requires 2 requests, so 30s should be enough
  609. // TODO: 30s is enough assuming immediate processing of dependents following
  610. // owner deletion, but in practice there can be a long delay between owner
  611. // deletion and dependent deletion processing. For now, increase the timeout
  612. // and investigate the processing delay.
  613. if err := wait.Poll(1*time.Second, 60*time.Second, func() (bool, error) {
  614. _, err := rcClient.Get(rc.Name, metav1.GetOptions{})
  615. if err == nil {
  616. pods, _ := podClient.List(metav1.ListOptions{})
  617. e2elog.Logf("%d pods remaining", len(pods.Items))
  618. count := 0
  619. for _, pod := range pods.Items {
  620. if pod.ObjectMeta.DeletionTimestamp == nil {
  621. count++
  622. }
  623. }
  624. e2elog.Logf("%d pods has nil DeletionTimestamp", count)
  625. e2elog.Logf("")
  626. return false, nil
  627. }
  628. if errors.IsNotFound(err) {
  629. return true, nil
  630. }
  631. return false, err
  632. }); err != nil {
  633. pods, err2 := podClient.List(metav1.ListOptions{})
  634. if err2 != nil {
  635. framework.Failf("%v", err2)
  636. }
  637. e2elog.Logf("%d remaining pods are:", len(pods.Items))
  638. e2elog.Logf("The ObjectMeta of the remaining pods are:")
  639. for _, pod := range pods.Items {
  640. e2elog.Logf("%#v", pod.ObjectMeta)
  641. }
  642. framework.Failf("failed to delete the rc: %v", err)
  643. }
  644. // There shouldn't be any pods
  645. pods, err := podClient.List(metav1.ListOptions{})
  646. if err != nil {
  647. framework.Failf("%v", err)
  648. }
  649. if len(pods.Items) != 0 {
  650. framework.Failf("expected no pods, got %#v", pods)
  651. }
  652. gatherMetrics(f)
  653. })
  654. // TODO: this should be an integration test
  655. /*
  656. Release : v1.9
  657. Testname: Garbage Collector, multiple owners
  658. Description: Create a replication controller RC1, with maximum allocatable Pods between 10 and 100 replicas. Create second replication controller RC2 and set RC2 as owner for half of those replicas. Once RC1 is created and the all Pods are created, delete RC1 with deleteOptions.PropagationPolicy set to Foreground. Half of the Pods that has RC2 as owner MUST not be deleted but have a deletion timestamp. Deleting the Replication Controller MUST not delete Pods that are owned by multiple replication controllers.
  659. */
  660. framework.ConformanceIt("should not delete dependents that have both valid owner and owner that's waiting for dependents to be deleted", func() {
  661. clientSet := f.ClientSet
  662. rcClient := clientSet.CoreV1().ReplicationControllers(f.Namespace.Name)
  663. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  664. rc1Name := "simpletest-rc-to-be-deleted"
  665. replicas := int32(estimateMaximumPods(clientSet, 10, 100))
  666. halfReplicas := int(replicas / 2)
  667. uniqLabelsDeleted := getUniqLabel("gctest_d", "valid_and_pending_owners_d")
  668. rc1 := newOwnerRC(f, rc1Name, replicas, uniqLabelsDeleted)
  669. ginkgo.By("create the rc1")
  670. rc1, err := rcClient.Create(rc1)
  671. if err != nil {
  672. framework.Failf("Failed to create replication controller: %v", err)
  673. }
  674. rc2Name := "simpletest-rc-to-stay"
  675. uniqLabelsStay := getUniqLabel("gctest_s", "valid_and_pending_owners_s")
  676. rc2 := newOwnerRC(f, rc2Name, 0, uniqLabelsStay)
  677. ginkgo.By("create the rc2")
  678. rc2, err = rcClient.Create(rc2)
  679. if err != nil {
  680. framework.Failf("Failed to create replication controller: %v", err)
  681. }
  682. // wait for rc1 to be stable
  683. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  684. rc1, err := rcClient.Get(rc1.Name, metav1.GetOptions{})
  685. if err != nil {
  686. return false, fmt.Errorf("Failed to get rc: %v", err)
  687. }
  688. if rc1.Status.Replicas == *rc1.Spec.Replicas {
  689. return true, nil
  690. }
  691. return false, nil
  692. }); err != nil {
  693. framework.Failf("failed to wait for the rc.Status.Replicas to reach rc.Spec.Replicas: %v", err)
  694. }
  695. ginkgo.By(fmt.Sprintf("set half of pods created by rc %s to have rc %s as owner as well", rc1Name, rc2Name))
  696. pods, err := podClient.List(metav1.ListOptions{})
  697. framework.ExpectNoError(err, "failed to list pods in namespace: %s", f.Namespace.Name)
  698. patch := fmt.Sprintf(`{"metadata":{"ownerReferences":[{"apiVersion":"v1","kind":"ReplicationController","name":"%s","uid":"%s"}]}}`, rc2.ObjectMeta.Name, rc2.ObjectMeta.UID)
  699. for i := 0; i < halfReplicas; i++ {
  700. pod := pods.Items[i]
  701. _, err := podClient.Patch(pod.Name, types.StrategicMergePatchType, []byte(patch))
  702. framework.ExpectNoError(err, "failed to apply to pod %s in namespace %s, a strategic merge patch: %s", pod.Name, f.Namespace.Name, patch)
  703. }
  704. ginkgo.By(fmt.Sprintf("delete the rc %s", rc1Name))
  705. deleteOptions := getForegroundOptions()
  706. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(rc1.UID))
  707. if err := rcClient.Delete(rc1.ObjectMeta.Name, deleteOptions); err != nil {
  708. framework.Failf("failed to delete the rc: %v", err)
  709. }
  710. ginkgo.By("wait for the rc to be deleted")
  711. // TODO: shorten the timeout when we make GC's periodic API rediscovery more efficient.
  712. // Tracked at https://github.com/kubernetes/kubernetes/issues/50046.
  713. if err := wait.Poll(5*time.Second, 90*time.Second, func() (bool, error) {
  714. _, err := rcClient.Get(rc1.Name, metav1.GetOptions{})
  715. if err == nil {
  716. pods, _ := podClient.List(metav1.ListOptions{})
  717. e2elog.Logf("%d pods remaining", len(pods.Items))
  718. count := 0
  719. for _, pod := range pods.Items {
  720. if pod.ObjectMeta.DeletionTimestamp == nil {
  721. count++
  722. }
  723. }
  724. e2elog.Logf("%d pods has nil DeletionTimestamp", count)
  725. e2elog.Logf("")
  726. return false, nil
  727. }
  728. if errors.IsNotFound(err) {
  729. return true, nil
  730. }
  731. return false, err
  732. }); err != nil {
  733. pods, err2 := podClient.List(metav1.ListOptions{})
  734. if err2 != nil {
  735. framework.Failf("%v", err2)
  736. }
  737. e2elog.Logf("%d remaining pods are:", len(pods.Items))
  738. e2elog.Logf("ObjectMeta of remaining pods are:")
  739. for _, pod := range pods.Items {
  740. e2elog.Logf("%#v", pod.ObjectMeta)
  741. }
  742. framework.Failf("failed to delete rc %s, err: %v", rc1Name, err)
  743. }
  744. // half of the pods should still exist,
  745. pods, err = podClient.List(metav1.ListOptions{})
  746. if err != nil {
  747. framework.Failf("%v", err)
  748. }
  749. if len(pods.Items) != halfReplicas {
  750. framework.Failf("expected %d pods, got %d", halfReplicas, len(pods.Items))
  751. }
  752. for _, pod := range pods.Items {
  753. if pod.ObjectMeta.DeletionTimestamp != nil {
  754. framework.Failf("expected pod DeletionTimestamp to be nil, got %#v", pod.ObjectMeta)
  755. }
  756. // they should only have 1 ownerReference left
  757. if len(pod.ObjectMeta.OwnerReferences) != 1 {
  758. framework.Failf("expected pod to only have 1 owner, got %#v", pod.ObjectMeta.OwnerReferences)
  759. }
  760. }
  761. gatherMetrics(f)
  762. })
  763. // TODO: should be an integration test
  764. /*
  765. Release : v1.9
  766. Testname: Garbage Collector, dependency cycle
  767. Description: Create three pods, patch them with Owner references such that pod1 has pod3, pod2 has pod1 and pod3 has pod2 as owner references respectively. Delete pod1 MUST delete all pods. The dependency cycle MUST not block the garbage collection.
  768. */
  769. framework.ConformanceIt("should not be blocked by dependency circle", func() {
  770. clientSet := f.ClientSet
  771. podClient := clientSet.CoreV1().Pods(f.Namespace.Name)
  772. pod1Name := "pod1"
  773. pod1 := newGCPod(pod1Name)
  774. pod1, err := podClient.Create(pod1)
  775. framework.ExpectNoError(err, "failed to create pod %s in namespace: %s", pod1Name, f.Namespace.Name)
  776. pod2Name := "pod2"
  777. pod2 := newGCPod(pod2Name)
  778. pod2, err = podClient.Create(pod2)
  779. framework.ExpectNoError(err, "failed to create pod %s in namespace: %s", pod2Name, f.Namespace.Name)
  780. pod3Name := "pod3"
  781. pod3 := newGCPod(pod3Name)
  782. pod3, err = podClient.Create(pod3)
  783. framework.ExpectNoError(err, "failed to create pod %s in namespace: %s", pod3Name, f.Namespace.Name)
  784. // create circular dependency
  785. addRefPatch := func(name string, uid types.UID) []byte {
  786. return []byte(fmt.Sprintf(`{"metadata":{"ownerReferences":[{"apiVersion":"v1","kind":"Pod","name":"%s","uid":"%s","controller":true,"blockOwnerDeletion":true}]}}`, name, uid))
  787. }
  788. patch1 := addRefPatch(pod3.Name, pod3.UID)
  789. pod1, err = podClient.Patch(pod1.Name, types.StrategicMergePatchType, patch1)
  790. framework.ExpectNoError(err, "failed to apply to pod %s in namespace %s, a strategic merge patch: %s", pod1.Name, f.Namespace.Name, patch1)
  791. e2elog.Logf("pod1.ObjectMeta.OwnerReferences=%#v", pod1.ObjectMeta.OwnerReferences)
  792. patch2 := addRefPatch(pod1.Name, pod1.UID)
  793. pod2, err = podClient.Patch(pod2.Name, types.StrategicMergePatchType, patch2)
  794. framework.ExpectNoError(err, "failed to apply to pod %s in namespace %s, a strategic merge patch: %s", pod2.Name, f.Namespace.Name, patch2)
  795. e2elog.Logf("pod2.ObjectMeta.OwnerReferences=%#v", pod2.ObjectMeta.OwnerReferences)
  796. patch3 := addRefPatch(pod2.Name, pod2.UID)
  797. pod3, err = podClient.Patch(pod3.Name, types.StrategicMergePatchType, patch3)
  798. framework.ExpectNoError(err, "failed to apply to pod %s in namespace %s, a strategic merge patch: %s", pod3.Name, f.Namespace.Name, patch3)
  799. e2elog.Logf("pod3.ObjectMeta.OwnerReferences=%#v", pod3.ObjectMeta.OwnerReferences)
  800. // delete one pod, should result in the deletion of all pods
  801. deleteOptions := getForegroundOptions()
  802. deleteOptions.Preconditions = metav1.NewUIDPreconditions(string(pod1.UID))
  803. err = podClient.Delete(pod1.ObjectMeta.Name, deleteOptions)
  804. framework.ExpectNoError(err, "failed to delete pod %s in namespace: %s", pod1.Name, f.Namespace.Name)
  805. var pods *v1.PodList
  806. var err2 error
  807. // TODO: shorten the timeout when we make GC's periodic API rediscovery more efficient.
  808. // Tracked at https://github.com/kubernetes/kubernetes/issues/50046.
  809. if err := wait.Poll(5*time.Second, 90*time.Second, func() (bool, error) {
  810. pods, err2 = podClient.List(metav1.ListOptions{})
  811. if err2 != nil {
  812. return false, fmt.Errorf("Failed to list pods: %v", err)
  813. }
  814. if len(pods.Items) == 0 {
  815. return true, nil
  816. }
  817. return false, nil
  818. }); err != nil {
  819. e2elog.Logf("pods are %#v", pods.Items)
  820. framework.Failf("failed to wait for all pods to be deleted: %v", err)
  821. }
  822. })
  823. ginkgo.It("should support cascading deletion of custom resources", func() {
  824. config, err := framework.LoadConfig()
  825. if err != nil {
  826. framework.Failf("failed to load config: %v", err)
  827. }
  828. apiExtensionClient, err := apiextensionsclientset.NewForConfig(config)
  829. if err != nil {
  830. framework.Failf("failed to initialize apiExtensionClient: %v", err)
  831. }
  832. // Create a random custom resource definition and ensure it's available for
  833. // use.
  834. definition := apiextensionstestserver.NewRandomNameCustomResourceDefinition(apiextensionsv1beta1.ClusterScoped)
  835. defer func() {
  836. err = apiextensionstestserver.DeleteCustomResourceDefinition(definition, apiExtensionClient)
  837. if err != nil && !errors.IsNotFound(err) {
  838. framework.Failf("failed to delete CustomResourceDefinition: %v", err)
  839. }
  840. }()
  841. definition, err = apiextensionstestserver.CreateNewCustomResourceDefinition(definition, apiExtensionClient, f.DynamicClient)
  842. if err != nil {
  843. framework.Failf("failed to create CustomResourceDefinition: %v", err)
  844. }
  845. // Get a client for the custom resource.
  846. gvr := schema.GroupVersionResource{Group: definition.Spec.Group, Version: definition.Spec.Version, Resource: definition.Spec.Names.Plural}
  847. resourceClient := f.DynamicClient.Resource(gvr)
  848. apiVersion := definition.Spec.Group + "/" + definition.Spec.Version
  849. // Create a custom owner resource.
  850. ownerName := names.SimpleNameGenerator.GenerateName("owner")
  851. owner := &unstructured.Unstructured{
  852. Object: map[string]interface{}{
  853. "apiVersion": apiVersion,
  854. "kind": definition.Spec.Names.Kind,
  855. "metadata": map[string]interface{}{
  856. "name": ownerName,
  857. },
  858. },
  859. }
  860. persistedOwner, err := resourceClient.Create(owner, metav1.CreateOptions{})
  861. if err != nil {
  862. framework.Failf("failed to create owner resource %q: %v", ownerName, err)
  863. }
  864. e2elog.Logf("created owner resource %q", ownerName)
  865. // Create a custom dependent resource.
  866. dependentName := names.SimpleNameGenerator.GenerateName("dependent")
  867. dependent := &unstructured.Unstructured{
  868. Object: map[string]interface{}{
  869. "apiVersion": apiVersion,
  870. "kind": definition.Spec.Names.Kind,
  871. "metadata": map[string]interface{}{
  872. "name": dependentName,
  873. "ownerReferences": []interface{}{
  874. map[string]interface{}{
  875. "uid": string(persistedOwner.GetUID()),
  876. "apiVersion": apiVersion,
  877. "kind": definition.Spec.Names.Kind,
  878. "name": ownerName,
  879. },
  880. },
  881. },
  882. },
  883. }
  884. persistedDependent, err := resourceClient.Create(dependent, metav1.CreateOptions{})
  885. if err != nil {
  886. framework.Failf("failed to create dependent resource %q: %v", dependentName, err)
  887. }
  888. e2elog.Logf("created dependent resource %q", dependentName)
  889. // Delete the owner.
  890. background := metav1.DeletePropagationBackground
  891. err = resourceClient.Delete(ownerName, &metav1.DeleteOptions{PropagationPolicy: &background})
  892. if err != nil {
  893. framework.Failf("failed to delete owner resource %q: %v", ownerName, err)
  894. }
  895. // Ensure the dependent is deleted.
  896. if err := wait.Poll(5*time.Second, 60*time.Second, func() (bool, error) {
  897. _, err := resourceClient.Get(dependentName, metav1.GetOptions{})
  898. return errors.IsNotFound(err), nil
  899. }); err != nil {
  900. e2elog.Logf("owner: %#v", persistedOwner)
  901. e2elog.Logf("dependent: %#v", persistedDependent)
  902. framework.Failf("failed waiting for dependent resource %q to be deleted", dependentName)
  903. }
  904. // Ensure the owner is deleted.
  905. _, err = resourceClient.Get(ownerName, metav1.GetOptions{})
  906. if err == nil {
  907. framework.Failf("expected owner resource %q to be deleted", ownerName)
  908. } else {
  909. if !errors.IsNotFound(err) {
  910. framework.Failf("unexpected error getting owner resource %q: %v", ownerName, err)
  911. }
  912. }
  913. })
  914. ginkgo.It("should support orphan deletion of custom resources", func() {
  915. config, err := framework.LoadConfig()
  916. if err != nil {
  917. framework.Failf("failed to load config: %v", err)
  918. }
  919. apiExtensionClient, err := apiextensionsclientset.NewForConfig(config)
  920. if err != nil {
  921. framework.Failf("failed to initialize apiExtensionClient: %v", err)
  922. }
  923. // Create a random custom resource definition and ensure it's available for
  924. // use.
  925. definition := apiextensionstestserver.NewRandomNameCustomResourceDefinition(apiextensionsv1beta1.ClusterScoped)
  926. defer func() {
  927. err = apiextensionstestserver.DeleteCustomResourceDefinition(definition, apiExtensionClient)
  928. if err != nil && !errors.IsNotFound(err) {
  929. framework.Failf("failed to delete CustomResourceDefinition: %v", err)
  930. }
  931. }()
  932. definition, err = apiextensionstestserver.CreateNewCustomResourceDefinition(definition, apiExtensionClient, f.DynamicClient)
  933. if err != nil {
  934. framework.Failf("failed to create CustomResourceDefinition: %v", err)
  935. }
  936. // Get a client for the custom resource.
  937. gvr := schema.GroupVersionResource{Group: definition.Spec.Group, Version: definition.Spec.Version, Resource: definition.Spec.Names.Plural}
  938. resourceClient := f.DynamicClient.Resource(gvr)
  939. apiVersion := definition.Spec.Group + "/" + definition.Spec.Version
  940. // Create a custom owner resource.
  941. ownerName := names.SimpleNameGenerator.GenerateName("owner")
  942. owner := &unstructured.Unstructured{
  943. Object: map[string]interface{}{
  944. "apiVersion": apiVersion,
  945. "kind": definition.Spec.Names.Kind,
  946. "metadata": map[string]interface{}{
  947. "name": ownerName,
  948. },
  949. },
  950. }
  951. persistedOwner, err := resourceClient.Create(owner, metav1.CreateOptions{})
  952. if err != nil {
  953. framework.Failf("failed to create owner resource %q: %v", ownerName, err)
  954. }
  955. e2elog.Logf("created owner resource %q", ownerName)
  956. // Create a custom dependent resource.
  957. dependentName := names.SimpleNameGenerator.GenerateName("dependent")
  958. dependent := &unstructured.Unstructured{
  959. Object: map[string]interface{}{
  960. "apiVersion": apiVersion,
  961. "kind": definition.Spec.Names.Kind,
  962. "metadata": map[string]interface{}{
  963. "name": dependentName,
  964. "ownerReferences": []map[string]string{
  965. {
  966. "uid": string(persistedOwner.GetUID()),
  967. "apiVersion": apiVersion,
  968. "kind": definition.Spec.Names.Kind,
  969. "name": ownerName,
  970. },
  971. },
  972. },
  973. },
  974. }
  975. _, err = resourceClient.Create(dependent, metav1.CreateOptions{})
  976. if err != nil {
  977. framework.Failf("failed to create dependent resource %q: %v", dependentName, err)
  978. }
  979. e2elog.Logf("created dependent resource %q", dependentName)
  980. // Delete the owner and orphan the dependent.
  981. err = resourceClient.Delete(ownerName, getOrphanOptions())
  982. if err != nil {
  983. framework.Failf("failed to delete owner resource %q: %v", ownerName, err)
  984. }
  985. ginkgo.By("wait for the owner to be deleted")
  986. if err := wait.Poll(5*time.Second, 120*time.Second, func() (bool, error) {
  987. _, err = resourceClient.Get(ownerName, metav1.GetOptions{})
  988. if err == nil {
  989. return false, nil
  990. }
  991. if err != nil && !errors.IsNotFound(err) {
  992. return false, fmt.Errorf("Failed to get owner: %v", err)
  993. }
  994. return true, nil
  995. }); err != nil {
  996. framework.Failf("timeout in waiting for the owner to be deleted: %v", err)
  997. }
  998. // Wait 30s and ensure the dependent is not deleted.
  999. ginkgo.By("wait for 30 seconds to see if the garbage collector mistakenly deletes the dependent crd")
  1000. if err := wait.Poll(5*time.Second, 30*time.Second, func() (bool, error) {
  1001. _, err := resourceClient.Get(dependentName, metav1.GetOptions{})
  1002. return false, err
  1003. }); err != nil && err != wait.ErrWaitTimeout {
  1004. framework.Failf("failed to ensure the dependent is not deleted: %v", err)
  1005. }
  1006. })
  1007. ginkgo.It("should delete jobs and pods created by cronjob", func() {
  1008. framework.SkipIfMissingResource(f.DynamicClient, CronJobGroupVersionResource, f.Namespace.Name)
  1009. ginkgo.By("Create the cronjob")
  1010. cronJob := newCronJob("simple", "*/1 * * * ?")
  1011. cronJob, err := f.ClientSet.BatchV1beta1().CronJobs(f.Namespace.Name).Create(cronJob)
  1012. framework.ExpectNoError(err, "failed to create cronjob: %+v, in namespace: %s", cronJob, f.Namespace.Name)
  1013. ginkgo.By("Wait for the CronJob to create new Job")
  1014. err = wait.PollImmediate(500*time.Millisecond, 2*time.Minute, func() (bool, error) {
  1015. jobs, err := f.ClientSet.BatchV1().Jobs(f.Namespace.Name).List(metav1.ListOptions{})
  1016. if err != nil {
  1017. return false, fmt.Errorf("Failed to list jobs: %v", err)
  1018. }
  1019. return len(jobs.Items) > 0, nil
  1020. })
  1021. if err != nil {
  1022. framework.Failf("Failed to wait for the CronJob to create some Jobs: %v", err)
  1023. }
  1024. ginkgo.By("Delete the cronjob")
  1025. if err := f.ClientSet.BatchV1beta1().CronJobs(f.Namespace.Name).Delete(cronJob.Name, getBackgroundOptions()); err != nil {
  1026. framework.Failf("Failed to delete the CronJob: %v", err)
  1027. }
  1028. ginkgo.By("Verify if cronjob does not leave jobs nor pods behind")
  1029. err = wait.PollImmediate(500*time.Millisecond, 1*time.Minute, func() (bool, error) {
  1030. objects := map[string]int{"CronJobs": 0, "Jobs": 0, "Pods": 0}
  1031. return verifyRemainingObjects(f, objects)
  1032. })
  1033. if err != nil {
  1034. framework.Failf("Failed to wait for all jobs and pods to be deleted: %v", err)
  1035. }
  1036. gatherMetrics(f)
  1037. })
  1038. })