drain_test.go 31 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051
  1. /*
  2. Copyright 2015 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 drain
  14. import (
  15. "errors"
  16. "fmt"
  17. "io"
  18. "io/ioutil"
  19. "net/http"
  20. "net/url"
  21. "os"
  22. "reflect"
  23. "strconv"
  24. "strings"
  25. "sync/atomic"
  26. "testing"
  27. "time"
  28. "github.com/spf13/cobra"
  29. "k8s.io/cli-runtime/pkg/genericclioptions"
  30. appsv1 "k8s.io/api/apps/v1"
  31. batchv1 "k8s.io/api/batch/v1"
  32. corev1 "k8s.io/api/core/v1"
  33. policyv1beta1 "k8s.io/api/policy/v1beta1"
  34. apierrors "k8s.io/apimachinery/pkg/api/errors"
  35. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  36. "k8s.io/apimachinery/pkg/runtime"
  37. "k8s.io/apimachinery/pkg/runtime/schema"
  38. "k8s.io/apimachinery/pkg/types"
  39. "k8s.io/apimachinery/pkg/util/strategicpatch"
  40. "k8s.io/apimachinery/pkg/util/wait"
  41. "k8s.io/cli-runtime/pkg/printers"
  42. "k8s.io/client-go/rest/fake"
  43. cmdtesting "k8s.io/kubernetes/pkg/kubectl/cmd/testing"
  44. cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
  45. "k8s.io/kubernetes/pkg/kubectl/drain"
  46. "k8s.io/kubernetes/pkg/kubectl/scheme"
  47. utilpointer "k8s.io/utils/pointer"
  48. )
  49. const (
  50. EvictionMethod = "Eviction"
  51. DeleteMethod = "Delete"
  52. )
  53. var node *corev1.Node
  54. var cordonedNode *corev1.Node
  55. func TestMain(m *testing.M) {
  56. // Create a node.
  57. node = &corev1.Node{
  58. ObjectMeta: metav1.ObjectMeta{
  59. Name: "node",
  60. CreationTimestamp: metav1.Time{Time: time.Now()},
  61. },
  62. Status: corev1.NodeStatus{},
  63. }
  64. // A copy of the same node, but cordoned.
  65. cordonedNode = node.DeepCopy()
  66. cordonedNode.Spec.Unschedulable = true
  67. os.Exit(m.Run())
  68. }
  69. func TestCordon(t *testing.T) {
  70. tests := []struct {
  71. description string
  72. node *corev1.Node
  73. expected *corev1.Node
  74. cmd func(cmdutil.Factory, genericclioptions.IOStreams) *cobra.Command
  75. arg string
  76. expectFatal bool
  77. }{
  78. {
  79. description: "node/node syntax",
  80. node: cordonedNode,
  81. expected: node,
  82. cmd: NewCmdUncordon,
  83. arg: "node/node",
  84. expectFatal: false,
  85. },
  86. {
  87. description: "uncordon for real",
  88. node: cordonedNode,
  89. expected: node,
  90. cmd: NewCmdUncordon,
  91. arg: "node",
  92. expectFatal: false,
  93. },
  94. {
  95. description: "uncordon does nothing",
  96. node: node,
  97. expected: node,
  98. cmd: NewCmdUncordon,
  99. arg: "node",
  100. expectFatal: false,
  101. },
  102. {
  103. description: "cordon does nothing",
  104. node: cordonedNode,
  105. expected: cordonedNode,
  106. cmd: NewCmdCordon,
  107. arg: "node",
  108. expectFatal: false,
  109. },
  110. {
  111. description: "cordon for real",
  112. node: node,
  113. expected: cordonedNode,
  114. cmd: NewCmdCordon,
  115. arg: "node",
  116. expectFatal: false,
  117. },
  118. {
  119. description: "cordon missing node",
  120. node: node,
  121. expected: node,
  122. cmd: NewCmdCordon,
  123. arg: "bar",
  124. expectFatal: true,
  125. },
  126. {
  127. description: "uncordon missing node",
  128. node: node,
  129. expected: node,
  130. cmd: NewCmdUncordon,
  131. arg: "bar",
  132. expectFatal: true,
  133. },
  134. {
  135. description: "cordon for multiple nodes",
  136. node: node,
  137. expected: cordonedNode,
  138. cmd: NewCmdCordon,
  139. arg: "node node1 node2",
  140. expectFatal: false,
  141. },
  142. {
  143. description: "uncordon for multiple nodes",
  144. node: cordonedNode,
  145. expected: node,
  146. cmd: NewCmdUncordon,
  147. arg: "node node1 node2",
  148. expectFatal: false,
  149. },
  150. }
  151. for _, test := range tests {
  152. t.Run(test.description, func(t *testing.T) {
  153. tf := cmdtesting.NewTestFactory()
  154. defer tf.Cleanup()
  155. codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
  156. ns := scheme.Codecs
  157. newNode := &corev1.Node{}
  158. updated := false
  159. tf.Client = &fake.RESTClient{
  160. GroupVersion: schema.GroupVersion{Group: "", Version: "v1"},
  161. NegotiatedSerializer: ns,
  162. Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
  163. m := &MyReq{req}
  164. switch {
  165. case m.isFor("GET", "/nodes/node1"):
  166. fallthrough
  167. case m.isFor("GET", "/nodes/node2"):
  168. fallthrough
  169. case m.isFor("GET", "/nodes/node"):
  170. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, test.node)}, nil
  171. case m.isFor("GET", "/nodes/bar"):
  172. return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.StringBody("nope")}, nil
  173. case m.isFor("PATCH", "/nodes/node1"):
  174. fallthrough
  175. case m.isFor("PATCH", "/nodes/node2"):
  176. fallthrough
  177. case m.isFor("PATCH", "/nodes/node"):
  178. data, err := ioutil.ReadAll(req.Body)
  179. if err != nil {
  180. t.Fatalf("%s: unexpected error: %v", test.description, err)
  181. }
  182. defer req.Body.Close()
  183. oldJSON, err := runtime.Encode(codec, node)
  184. if err != nil {
  185. t.Fatalf("%s: unexpected error: %v", test.description, err)
  186. }
  187. appliedPatch, err := strategicpatch.StrategicMergePatch(oldJSON, data, &corev1.Node{})
  188. if err != nil {
  189. t.Fatalf("%s: unexpected error: %v", test.description, err)
  190. }
  191. if err := runtime.DecodeInto(codec, appliedPatch, newNode); err != nil {
  192. t.Fatalf("%s: unexpected error: %v", test.description, err)
  193. }
  194. if !reflect.DeepEqual(test.expected.Spec, newNode.Spec) {
  195. t.Fatalf("%s: expected:\n%v\nsaw:\n%v\n", test.description, test.expected.Spec.Unschedulable, newNode.Spec.Unschedulable)
  196. }
  197. updated = true
  198. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, newNode)}, nil
  199. default:
  200. t.Fatalf("%s: unexpected request: %v %#v\n%#v", test.description, req.Method, req.URL, req)
  201. return nil, nil
  202. }
  203. }),
  204. }
  205. tf.ClientConfigVal = cmdtesting.DefaultClientConfig()
  206. ioStreams, _, _, _ := genericclioptions.NewTestIOStreams()
  207. cmd := test.cmd(tf, ioStreams)
  208. sawFatal := false
  209. func() {
  210. defer func() {
  211. // Recover from the panic below.
  212. _ = recover()
  213. // Restore cmdutil behavior
  214. cmdutil.DefaultBehaviorOnFatal()
  215. }()
  216. cmdutil.BehaviorOnFatal(func(e string, code int) {
  217. sawFatal = true
  218. panic(e)
  219. })
  220. cmd.SetArgs(strings.Split(test.arg, " "))
  221. cmd.Execute()
  222. }()
  223. if test.expectFatal {
  224. if !sawFatal {
  225. t.Fatalf("%s: unexpected non-error", test.description)
  226. }
  227. if updated {
  228. t.Fatalf("%s: unexpected update", test.description)
  229. }
  230. }
  231. if !test.expectFatal && sawFatal {
  232. t.Fatalf("%s: unexpected error", test.description)
  233. }
  234. if !reflect.DeepEqual(test.expected.Spec, test.node.Spec) && !updated {
  235. t.Fatalf("%s: node never updated", test.description)
  236. }
  237. })
  238. }
  239. }
  240. func TestDrain(t *testing.T) {
  241. labels := make(map[string]string)
  242. labels["my_key"] = "my_value"
  243. rc := corev1.ReplicationController{
  244. ObjectMeta: metav1.ObjectMeta{
  245. Name: "rc",
  246. Namespace: "default",
  247. CreationTimestamp: metav1.Time{Time: time.Now()},
  248. Labels: labels,
  249. },
  250. Spec: corev1.ReplicationControllerSpec{
  251. Selector: labels,
  252. },
  253. }
  254. rcPod := corev1.Pod{
  255. ObjectMeta: metav1.ObjectMeta{
  256. Name: "bar",
  257. Namespace: "default",
  258. CreationTimestamp: metav1.Time{Time: time.Now()},
  259. Labels: labels,
  260. OwnerReferences: []metav1.OwnerReference{
  261. {
  262. APIVersion: "v1",
  263. Kind: "ReplicationController",
  264. Name: "rc",
  265. UID: "123",
  266. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  267. Controller: utilpointer.BoolPtr(true),
  268. },
  269. },
  270. },
  271. Spec: corev1.PodSpec{
  272. NodeName: "node",
  273. },
  274. }
  275. ds := appsv1.DaemonSet{
  276. ObjectMeta: metav1.ObjectMeta{
  277. Name: "ds",
  278. Namespace: "default",
  279. CreationTimestamp: metav1.Time{Time: time.Now()},
  280. },
  281. Spec: appsv1.DaemonSetSpec{
  282. Selector: &metav1.LabelSelector{MatchLabels: labels},
  283. },
  284. }
  285. dsPod := corev1.Pod{
  286. ObjectMeta: metav1.ObjectMeta{
  287. Name: "bar",
  288. Namespace: "default",
  289. CreationTimestamp: metav1.Time{Time: time.Now()},
  290. Labels: labels,
  291. OwnerReferences: []metav1.OwnerReference{
  292. {
  293. APIVersion: "apps/v1",
  294. Kind: "DaemonSet",
  295. Name: "ds",
  296. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  297. Controller: utilpointer.BoolPtr(true),
  298. },
  299. },
  300. },
  301. Spec: corev1.PodSpec{
  302. NodeName: "node",
  303. },
  304. }
  305. dsTerminatedPod := corev1.Pod{
  306. ObjectMeta: metav1.ObjectMeta{
  307. Name: "bar",
  308. Namespace: "default",
  309. CreationTimestamp: metav1.Time{Time: time.Now()},
  310. Labels: labels,
  311. OwnerReferences: []metav1.OwnerReference{
  312. {
  313. APIVersion: "apps/v1",
  314. Kind: "DaemonSet",
  315. Name: "ds",
  316. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  317. Controller: utilpointer.BoolPtr(true),
  318. },
  319. },
  320. },
  321. Spec: corev1.PodSpec{
  322. NodeName: "node",
  323. },
  324. Status: corev1.PodStatus{
  325. Phase: corev1.PodSucceeded,
  326. },
  327. }
  328. dsPodWithEmptyDir := corev1.Pod{
  329. ObjectMeta: metav1.ObjectMeta{
  330. Name: "bar",
  331. Namespace: "default",
  332. CreationTimestamp: metav1.Time{Time: time.Now()},
  333. Labels: labels,
  334. OwnerReferences: []metav1.OwnerReference{
  335. {
  336. APIVersion: "apps/v1",
  337. Kind: "DaemonSet",
  338. Name: "ds",
  339. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  340. Controller: utilpointer.BoolPtr(true),
  341. },
  342. },
  343. },
  344. Spec: corev1.PodSpec{
  345. NodeName: "node",
  346. Volumes: []corev1.Volume{
  347. {
  348. Name: "scratch",
  349. VolumeSource: corev1.VolumeSource{EmptyDir: &corev1.EmptyDirVolumeSource{Medium: ""}},
  350. },
  351. },
  352. },
  353. }
  354. orphanedDsPod := corev1.Pod{
  355. ObjectMeta: metav1.ObjectMeta{
  356. Name: "bar",
  357. Namespace: "default",
  358. CreationTimestamp: metav1.Time{Time: time.Now()},
  359. Labels: labels,
  360. },
  361. Spec: corev1.PodSpec{
  362. NodeName: "node",
  363. },
  364. }
  365. job := batchv1.Job{
  366. ObjectMeta: metav1.ObjectMeta{
  367. Name: "job",
  368. Namespace: "default",
  369. CreationTimestamp: metav1.Time{Time: time.Now()},
  370. },
  371. Spec: batchv1.JobSpec{
  372. Selector: &metav1.LabelSelector{MatchLabels: labels},
  373. },
  374. }
  375. jobPod := corev1.Pod{
  376. ObjectMeta: metav1.ObjectMeta{
  377. Name: "bar",
  378. Namespace: "default",
  379. CreationTimestamp: metav1.Time{Time: time.Now()},
  380. Labels: labels,
  381. OwnerReferences: []metav1.OwnerReference{
  382. {
  383. APIVersion: "v1",
  384. Kind: "Job",
  385. Name: "job",
  386. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  387. Controller: utilpointer.BoolPtr(true),
  388. },
  389. },
  390. },
  391. Spec: corev1.PodSpec{
  392. NodeName: "node",
  393. Volumes: []corev1.Volume{
  394. {
  395. Name: "scratch",
  396. VolumeSource: corev1.VolumeSource{EmptyDir: &corev1.EmptyDirVolumeSource{Medium: ""}},
  397. },
  398. },
  399. },
  400. }
  401. terminatedJobPodWithLocalStorage := corev1.Pod{
  402. ObjectMeta: metav1.ObjectMeta{
  403. Name: "bar",
  404. Namespace: "default",
  405. CreationTimestamp: metav1.Time{Time: time.Now()},
  406. Labels: labels,
  407. OwnerReferences: []metav1.OwnerReference{
  408. {
  409. APIVersion: "v1",
  410. Kind: "Job",
  411. Name: "job",
  412. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  413. Controller: utilpointer.BoolPtr(true),
  414. },
  415. },
  416. },
  417. Spec: corev1.PodSpec{
  418. NodeName: "node",
  419. Volumes: []corev1.Volume{
  420. {
  421. Name: "scratch",
  422. VolumeSource: corev1.VolumeSource{EmptyDir: &corev1.EmptyDirVolumeSource{Medium: ""}},
  423. },
  424. },
  425. },
  426. Status: corev1.PodStatus{
  427. Phase: corev1.PodSucceeded,
  428. },
  429. }
  430. rs := appsv1.ReplicaSet{
  431. ObjectMeta: metav1.ObjectMeta{
  432. Name: "rs",
  433. Namespace: "default",
  434. CreationTimestamp: metav1.Time{Time: time.Now()},
  435. Labels: labels,
  436. },
  437. Spec: appsv1.ReplicaSetSpec{
  438. Selector: &metav1.LabelSelector{MatchLabels: labels},
  439. },
  440. }
  441. rsPod := corev1.Pod{
  442. ObjectMeta: metav1.ObjectMeta{
  443. Name: "bar",
  444. Namespace: "default",
  445. CreationTimestamp: metav1.Time{Time: time.Now()},
  446. Labels: labels,
  447. OwnerReferences: []metav1.OwnerReference{
  448. {
  449. APIVersion: "v1",
  450. Kind: "ReplicaSet",
  451. Name: "rs",
  452. BlockOwnerDeletion: utilpointer.BoolPtr(true),
  453. Controller: utilpointer.BoolPtr(true),
  454. },
  455. },
  456. },
  457. Spec: corev1.PodSpec{
  458. NodeName: "node",
  459. },
  460. }
  461. nakedPod := corev1.Pod{
  462. ObjectMeta: metav1.ObjectMeta{
  463. Name: "bar",
  464. Namespace: "default",
  465. CreationTimestamp: metav1.Time{Time: time.Now()},
  466. Labels: labels,
  467. },
  468. Spec: corev1.PodSpec{
  469. NodeName: "node",
  470. },
  471. }
  472. emptydirPod := corev1.Pod{
  473. ObjectMeta: metav1.ObjectMeta{
  474. Name: "bar",
  475. Namespace: "default",
  476. CreationTimestamp: metav1.Time{Time: time.Now()},
  477. Labels: labels,
  478. },
  479. Spec: corev1.PodSpec{
  480. NodeName: "node",
  481. Volumes: []corev1.Volume{
  482. {
  483. Name: "scratch",
  484. VolumeSource: corev1.VolumeSource{EmptyDir: &corev1.EmptyDirVolumeSource{Medium: ""}},
  485. },
  486. },
  487. },
  488. }
  489. emptydirTerminatedPod := corev1.Pod{
  490. ObjectMeta: metav1.ObjectMeta{
  491. Name: "bar",
  492. Namespace: "default",
  493. CreationTimestamp: metav1.Time{Time: time.Now()},
  494. Labels: labels,
  495. },
  496. Spec: corev1.PodSpec{
  497. NodeName: "node",
  498. Volumes: []corev1.Volume{
  499. {
  500. Name: "scratch",
  501. VolumeSource: corev1.VolumeSource{EmptyDir: &corev1.EmptyDirVolumeSource{Medium: ""}},
  502. },
  503. },
  504. },
  505. Status: corev1.PodStatus{
  506. Phase: corev1.PodFailed,
  507. },
  508. }
  509. tests := []struct {
  510. description string
  511. node *corev1.Node
  512. expected *corev1.Node
  513. pods []corev1.Pod
  514. rcs []corev1.ReplicationController
  515. replicaSets []appsv1.ReplicaSet
  516. args []string
  517. expectWarning string
  518. expectFatal bool
  519. expectDelete bool
  520. }{
  521. {
  522. description: "RC-managed pod",
  523. node: node,
  524. expected: cordonedNode,
  525. pods: []corev1.Pod{rcPod},
  526. rcs: []corev1.ReplicationController{rc},
  527. args: []string{"node"},
  528. expectFatal: false,
  529. expectDelete: true,
  530. },
  531. {
  532. description: "DS-managed pod",
  533. node: node,
  534. expected: cordonedNode,
  535. pods: []corev1.Pod{dsPod},
  536. rcs: []corev1.ReplicationController{rc},
  537. args: []string{"node"},
  538. expectFatal: true,
  539. expectDelete: false,
  540. },
  541. {
  542. description: "DS-managed terminated pod",
  543. node: node,
  544. expected: cordonedNode,
  545. pods: []corev1.Pod{dsTerminatedPod},
  546. rcs: []corev1.ReplicationController{rc},
  547. args: []string{"node"},
  548. expectFatal: false,
  549. expectDelete: true,
  550. },
  551. {
  552. description: "orphaned DS-managed pod",
  553. node: node,
  554. expected: cordonedNode,
  555. pods: []corev1.Pod{orphanedDsPod},
  556. rcs: []corev1.ReplicationController{},
  557. args: []string{"node"},
  558. expectFatal: true,
  559. expectDelete: false,
  560. },
  561. {
  562. description: "orphaned DS-managed pod with --force",
  563. node: node,
  564. expected: cordonedNode,
  565. pods: []corev1.Pod{orphanedDsPod},
  566. rcs: []corev1.ReplicationController{},
  567. args: []string{"node", "--force"},
  568. expectFatal: false,
  569. expectDelete: true,
  570. expectWarning: "WARNING: deleting Pods not managed by ReplicationController, ReplicaSet, Job, DaemonSet or StatefulSet: default/bar",
  571. },
  572. {
  573. description: "DS-managed pod with --ignore-daemonsets",
  574. node: node,
  575. expected: cordonedNode,
  576. pods: []corev1.Pod{dsPod},
  577. rcs: []corev1.ReplicationController{rc},
  578. args: []string{"node", "--ignore-daemonsets"},
  579. expectFatal: false,
  580. expectDelete: false,
  581. },
  582. {
  583. description: "DS-managed pod with emptyDir with --ignore-daemonsets",
  584. node: node,
  585. expected: cordonedNode,
  586. pods: []corev1.Pod{dsPodWithEmptyDir},
  587. rcs: []corev1.ReplicationController{rc},
  588. args: []string{"node", "--ignore-daemonsets"},
  589. expectWarning: "WARNING: ignoring DaemonSet-managed Pods: default/bar",
  590. expectFatal: false,
  591. expectDelete: false,
  592. },
  593. {
  594. description: "Job-managed pod with local storage",
  595. node: node,
  596. expected: cordonedNode,
  597. pods: []corev1.Pod{jobPod},
  598. rcs: []corev1.ReplicationController{rc},
  599. args: []string{"node", "--force", "--delete-local-data=true"},
  600. expectFatal: false,
  601. expectDelete: true,
  602. },
  603. {
  604. description: "Job-managed terminated pod",
  605. node: node,
  606. expected: cordonedNode,
  607. pods: []corev1.Pod{terminatedJobPodWithLocalStorage},
  608. rcs: []corev1.ReplicationController{rc},
  609. args: []string{"node"},
  610. expectFatal: false,
  611. expectDelete: true,
  612. },
  613. {
  614. description: "RS-managed pod",
  615. node: node,
  616. expected: cordonedNode,
  617. pods: []corev1.Pod{rsPod},
  618. replicaSets: []appsv1.ReplicaSet{rs},
  619. args: []string{"node"},
  620. expectFatal: false,
  621. expectDelete: true,
  622. },
  623. {
  624. description: "naked pod",
  625. node: node,
  626. expected: cordonedNode,
  627. pods: []corev1.Pod{nakedPod},
  628. rcs: []corev1.ReplicationController{},
  629. args: []string{"node"},
  630. expectFatal: true,
  631. expectDelete: false,
  632. },
  633. {
  634. description: "naked pod with --force",
  635. node: node,
  636. expected: cordonedNode,
  637. pods: []corev1.Pod{nakedPod},
  638. rcs: []corev1.ReplicationController{},
  639. args: []string{"node", "--force"},
  640. expectFatal: false,
  641. expectDelete: true,
  642. },
  643. {
  644. description: "pod with EmptyDir",
  645. node: node,
  646. expected: cordonedNode,
  647. pods: []corev1.Pod{emptydirPod},
  648. args: []string{"node", "--force"},
  649. expectFatal: true,
  650. expectDelete: false,
  651. },
  652. {
  653. description: "terminated pod with emptyDir",
  654. node: node,
  655. expected: cordonedNode,
  656. pods: []corev1.Pod{emptydirTerminatedPod},
  657. rcs: []corev1.ReplicationController{rc},
  658. args: []string{"node"},
  659. expectFatal: false,
  660. expectDelete: true,
  661. },
  662. {
  663. description: "pod with EmptyDir and --delete-local-data",
  664. node: node,
  665. expected: cordonedNode,
  666. pods: []corev1.Pod{emptydirPod},
  667. args: []string{"node", "--force", "--delete-local-data=true"},
  668. expectFatal: false,
  669. expectDelete: true,
  670. },
  671. {
  672. description: "empty node",
  673. node: node,
  674. expected: cordonedNode,
  675. pods: []corev1.Pod{},
  676. rcs: []corev1.ReplicationController{rc},
  677. args: []string{"node"},
  678. expectFatal: false,
  679. expectDelete: false,
  680. },
  681. }
  682. testEviction := false
  683. for i := 0; i < 2; i++ {
  684. testEviction = !testEviction
  685. var currMethod string
  686. if testEviction {
  687. currMethod = EvictionMethod
  688. } else {
  689. currMethod = DeleteMethod
  690. }
  691. for _, test := range tests {
  692. t.Run(test.description, func(t *testing.T) {
  693. newNode := &corev1.Node{}
  694. var deletions, evictions int32
  695. tf := cmdtesting.NewTestFactory()
  696. defer tf.Cleanup()
  697. codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
  698. ns := scheme.Codecs
  699. tf.Client = &fake.RESTClient{
  700. GroupVersion: schema.GroupVersion{Group: "", Version: "v1"},
  701. NegotiatedSerializer: ns,
  702. Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
  703. m := &MyReq{req}
  704. switch {
  705. case req.Method == "GET" && req.URL.Path == "/api":
  706. apiVersions := metav1.APIVersions{
  707. Versions: []string{"v1"},
  708. }
  709. return cmdtesting.GenResponseWithJsonEncodedBody(apiVersions)
  710. case req.Method == "GET" && req.URL.Path == "/apis":
  711. groupList := metav1.APIGroupList{
  712. Groups: []metav1.APIGroup{
  713. {
  714. Name: "policy",
  715. PreferredVersion: metav1.GroupVersionForDiscovery{
  716. GroupVersion: "policy/v1beta1",
  717. },
  718. },
  719. },
  720. }
  721. return cmdtesting.GenResponseWithJsonEncodedBody(groupList)
  722. case req.Method == "GET" && req.URL.Path == "/api/v1":
  723. resourceList := metav1.APIResourceList{
  724. GroupVersion: "v1",
  725. }
  726. if testEviction {
  727. resourceList.APIResources = []metav1.APIResource{
  728. {
  729. Name: drain.EvictionSubresource,
  730. Kind: drain.EvictionKind,
  731. },
  732. }
  733. }
  734. return cmdtesting.GenResponseWithJsonEncodedBody(resourceList)
  735. case m.isFor("GET", "/nodes/node"):
  736. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, test.node)}, nil
  737. case m.isFor("GET", "/namespaces/default/replicationcontrollers/rc"):
  738. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &test.rcs[0])}, nil
  739. case m.isFor("GET", "/namespaces/default/daemonsets/ds"):
  740. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &ds)}, nil
  741. case m.isFor("GET", "/namespaces/default/daemonsets/missing-ds"):
  742. return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &appsv1.DaemonSet{})}, nil
  743. case m.isFor("GET", "/namespaces/default/jobs/job"):
  744. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &job)}, nil
  745. case m.isFor("GET", "/namespaces/default/replicasets/rs"):
  746. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &test.replicaSets[0])}, nil
  747. case m.isFor("GET", "/namespaces/default/pods/bar"):
  748. return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &corev1.Pod{})}, nil
  749. case m.isFor("GET", "/pods"):
  750. values, err := url.ParseQuery(req.URL.RawQuery)
  751. if err != nil {
  752. t.Fatalf("%s: unexpected error: %v", test.description, err)
  753. }
  754. getParams := make(url.Values)
  755. getParams["fieldSelector"] = []string{"spec.nodeName=node"}
  756. if !reflect.DeepEqual(getParams, values) {
  757. t.Fatalf("%s: expected:\n%v\nsaw:\n%v\n", test.description, getParams, values)
  758. }
  759. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &corev1.PodList{Items: test.pods})}, nil
  760. case m.isFor("GET", "/replicationcontrollers"):
  761. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &corev1.ReplicationControllerList{Items: test.rcs})}, nil
  762. case m.isFor("PATCH", "/nodes/node"):
  763. data, err := ioutil.ReadAll(req.Body)
  764. if err != nil {
  765. t.Fatalf("%s: unexpected error: %v", test.description, err)
  766. }
  767. defer req.Body.Close()
  768. oldJSON, err := runtime.Encode(codec, node)
  769. if err != nil {
  770. t.Fatalf("%s: unexpected error: %v", test.description, err)
  771. }
  772. appliedPatch, err := strategicpatch.StrategicMergePatch(oldJSON, data, &corev1.Node{})
  773. if err != nil {
  774. t.Fatalf("%s: unexpected error: %v", test.description, err)
  775. }
  776. if err := runtime.DecodeInto(codec, appliedPatch, newNode); err != nil {
  777. t.Fatalf("%s: unexpected error: %v", test.description, err)
  778. }
  779. if !reflect.DeepEqual(test.expected.Spec, newNode.Spec) {
  780. t.Fatalf("%s: expected:\n%v\nsaw:\n%v\n", test.description, test.expected.Spec, newNode.Spec)
  781. }
  782. return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, newNode)}, nil
  783. case m.isFor("DELETE", "/namespaces/default/pods/bar"):
  784. atomic.AddInt32(&deletions, 1)
  785. return &http.Response{StatusCode: 204, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &test.pods[0])}, nil
  786. case m.isFor("POST", "/namespaces/default/pods/bar/eviction"):
  787. atomic.AddInt32(&evictions, 1)
  788. return &http.Response{StatusCode: 201, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &policyv1beta1.Eviction{})}, nil
  789. default:
  790. t.Fatalf("%s: unexpected request: %v %#v\n%#v", test.description, req.Method, req.URL, req)
  791. return nil, nil
  792. }
  793. }),
  794. }
  795. tf.ClientConfigVal = cmdtesting.DefaultClientConfig()
  796. ioStreams, _, _, errBuf := genericclioptions.NewTestIOStreams()
  797. cmd := NewCmdDrain(tf, ioStreams)
  798. sawFatal := false
  799. fatalMsg := ""
  800. func() {
  801. defer func() {
  802. // Recover from the panic below.
  803. _ = recover()
  804. // Restore cmdutil behavior
  805. cmdutil.DefaultBehaviorOnFatal()
  806. }()
  807. cmdutil.BehaviorOnFatal(func(e string, code int) { sawFatal = true; fatalMsg = e; panic(e) })
  808. cmd.SetArgs(test.args)
  809. cmd.Execute()
  810. }()
  811. if test.expectFatal {
  812. if !sawFatal {
  813. //t.Logf("outBuf = %s", outBuf.String())
  814. //t.Logf("errBuf = %s", errBuf.String())
  815. t.Fatalf("%s: unexpected non-error when using %s", test.description, currMethod)
  816. }
  817. } else {
  818. if sawFatal {
  819. t.Fatalf("%s: unexpected error when using %s: %s", test.description, currMethod, fatalMsg)
  820. }
  821. }
  822. deleted := deletions > 0
  823. evicted := evictions > 0
  824. if test.expectDelete {
  825. // Test Delete
  826. if !testEviction && !deleted {
  827. t.Fatalf("%s: pod never deleted", test.description)
  828. }
  829. // Test Eviction
  830. if testEviction {
  831. if !evicted {
  832. t.Fatalf("%s: pod never evicted", test.description)
  833. }
  834. if evictions > 1 {
  835. t.Fatalf("%s: asked to evict same pod %d too many times", test.description, evictions-1)
  836. }
  837. }
  838. }
  839. if !test.expectDelete {
  840. if deleted {
  841. t.Fatalf("%s: unexpected delete when using %s", test.description, currMethod)
  842. }
  843. if deletions > 1 {
  844. t.Fatalf("%s: asked to deleted same pod %d too many times", test.description, deletions-1)
  845. }
  846. }
  847. if deleted && evicted {
  848. t.Fatalf("%s: same pod deleted %d times and evicted %d times", test.description, deletions, evictions)
  849. }
  850. if len(test.expectWarning) > 0 {
  851. if len(errBuf.String()) == 0 {
  852. t.Fatalf("%s: expected warning, but found no stderr output", test.description)
  853. }
  854. // Mac and Bazel on Linux behave differently when returning newlines
  855. if a, e := errBuf.String(), test.expectWarning; !strings.Contains(a, e) {
  856. t.Fatalf("%s: actual warning message did not match expected warning message.\n Expecting:\n%v\n Got:\n%v", test.description, e, a)
  857. }
  858. }
  859. })
  860. }
  861. }
  862. }
  863. func TestDeletePods(t *testing.T) {
  864. ifHasBeenCalled := map[string]bool{}
  865. tests := []struct {
  866. description string
  867. interval time.Duration
  868. timeout time.Duration
  869. expectPendingPods bool
  870. expectError bool
  871. expectedError *error
  872. getPodFn func(namespace, name string) (*corev1.Pod, error)
  873. }{
  874. {
  875. description: "Wait for deleting to complete",
  876. interval: 100 * time.Millisecond,
  877. timeout: 10 * time.Second,
  878. expectPendingPods: false,
  879. expectError: false,
  880. expectedError: nil,
  881. getPodFn: func(namespace, name string) (*corev1.Pod, error) {
  882. oldPodMap, _ := createPods(false)
  883. newPodMap, _ := createPods(true)
  884. if oldPod, found := oldPodMap[name]; found {
  885. if _, ok := ifHasBeenCalled[name]; !ok {
  886. ifHasBeenCalled[name] = true
  887. return &oldPod, nil
  888. }
  889. if oldPod.ObjectMeta.Generation < 4 {
  890. newPod := newPodMap[name]
  891. return &newPod, nil
  892. }
  893. return nil, apierrors.NewNotFound(schema.GroupResource{Resource: "pods"}, name)
  894. }
  895. return nil, apierrors.NewNotFound(schema.GroupResource{Resource: "pods"}, name)
  896. },
  897. },
  898. {
  899. description: "Deleting could timeout",
  900. interval: 200 * time.Millisecond,
  901. timeout: 3 * time.Second,
  902. expectPendingPods: true,
  903. expectError: true,
  904. expectedError: &wait.ErrWaitTimeout,
  905. getPodFn: func(namespace, name string) (*corev1.Pod, error) {
  906. oldPodMap, _ := createPods(false)
  907. if oldPod, found := oldPodMap[name]; found {
  908. return &oldPod, nil
  909. }
  910. return nil, fmt.Errorf("%q: not found", name)
  911. },
  912. },
  913. {
  914. description: "Client error could be passed out",
  915. interval: 200 * time.Millisecond,
  916. timeout: 5 * time.Second,
  917. expectPendingPods: true,
  918. expectError: true,
  919. expectedError: nil,
  920. getPodFn: func(namespace, name string) (*corev1.Pod, error) {
  921. return nil, errors.New("This is a random error for testing")
  922. },
  923. },
  924. }
  925. for _, test := range tests {
  926. t.Run(test.description, func(t *testing.T) {
  927. tf := cmdtesting.NewTestFactory()
  928. defer tf.Cleanup()
  929. o := DrainCmdOptions{
  930. PrintFlags: genericclioptions.NewPrintFlags("drained").WithTypeSetter(scheme.Scheme),
  931. }
  932. o.Out = os.Stdout
  933. o.ToPrinter = func(operation string) (printers.ResourcePrinterFunc, error) {
  934. return func(obj runtime.Object, out io.Writer) error {
  935. return nil
  936. }, nil
  937. }
  938. _, pods := createPods(false)
  939. pendingPods, err := o.waitForDelete(pods, test.interval, test.timeout, false, test.getPodFn)
  940. if test.expectError {
  941. if err == nil {
  942. t.Fatalf("%s: unexpected non-error", test.description)
  943. } else if test.expectedError != nil {
  944. if *test.expectedError != err {
  945. t.Fatalf("%s: the error does not match expected error", test.description)
  946. }
  947. }
  948. }
  949. if !test.expectError && err != nil {
  950. t.Fatalf("%s: unexpected error", test.description)
  951. }
  952. if test.expectPendingPods && len(pendingPods) == 0 {
  953. t.Fatalf("%s: unexpected empty pods", test.description)
  954. }
  955. if !test.expectPendingPods && len(pendingPods) > 0 {
  956. t.Fatalf("%s: unexpected pending pods", test.description)
  957. }
  958. })
  959. }
  960. }
  961. func createPods(ifCreateNewPods bool) (map[string]corev1.Pod, []corev1.Pod) {
  962. podMap := make(map[string]corev1.Pod)
  963. podSlice := []corev1.Pod{}
  964. for i := 0; i < 8; i++ {
  965. var uid types.UID
  966. if ifCreateNewPods {
  967. uid = types.UID(i)
  968. } else {
  969. uid = types.UID(strconv.Itoa(i) + strconv.Itoa(i))
  970. }
  971. pod := corev1.Pod{
  972. ObjectMeta: metav1.ObjectMeta{
  973. Name: "pod" + strconv.Itoa(i),
  974. Namespace: "default",
  975. UID: uid,
  976. Generation: int64(i),
  977. },
  978. }
  979. podMap[pod.Name] = pod
  980. podSlice = append(podSlice, pod)
  981. }
  982. return podMap, podSlice
  983. }
  984. type MyReq struct {
  985. Request *http.Request
  986. }
  987. func (m *MyReq) isFor(method string, path string) bool {
  988. req := m.Request
  989. return method == req.Method && (req.URL.Path == path ||
  990. req.URL.Path == strings.Join([]string{"/api/v1", path}, "") ||
  991. req.URL.Path == strings.Join([]string{"/apis/apps/v1", path}, "") ||
  992. req.URL.Path == strings.Join([]string{"/apis/batch/v1", path}, ""))
  993. }