index_test.go 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674
  1. /*
  2. Copyright 2014 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 persistentvolume
  14. import (
  15. "sort"
  16. "testing"
  17. v1 "k8s.io/api/core/v1"
  18. "k8s.io/apimachinery/pkg/api/resource"
  19. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  20. utilfeature "k8s.io/apiserver/pkg/util/feature"
  21. "k8s.io/client-go/kubernetes/scheme"
  22. ref "k8s.io/client-go/tools/reference"
  23. featuregatetesting "k8s.io/component-base/featuregate/testing"
  24. pvutil "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/util"
  25. "k8s.io/kubernetes/pkg/features"
  26. "k8s.io/kubernetes/pkg/volume/util"
  27. )
  28. func makePVC(size string, modfn func(*v1.PersistentVolumeClaim)) *v1.PersistentVolumeClaim {
  29. fs := v1.PersistentVolumeFilesystem
  30. pvc := v1.PersistentVolumeClaim{
  31. ObjectMeta: metav1.ObjectMeta{
  32. Name: "claim01",
  33. Namespace: "myns",
  34. },
  35. Spec: v1.PersistentVolumeClaimSpec{
  36. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany, v1.ReadWriteOnce},
  37. Resources: v1.ResourceRequirements{
  38. Requests: v1.ResourceList{
  39. v1.ResourceName(v1.ResourceStorage): resource.MustParse(size),
  40. },
  41. },
  42. VolumeMode: &fs,
  43. },
  44. }
  45. if modfn != nil {
  46. modfn(&pvc)
  47. }
  48. return &pvc
  49. }
  50. func makeVolumeModePVC(size string, mode *v1.PersistentVolumeMode, modfn func(*v1.PersistentVolumeClaim)) *v1.PersistentVolumeClaim {
  51. pvc := v1.PersistentVolumeClaim{
  52. ObjectMeta: metav1.ObjectMeta{
  53. Name: "claim01",
  54. Namespace: "myns",
  55. },
  56. Spec: v1.PersistentVolumeClaimSpec{
  57. VolumeMode: mode,
  58. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  59. Resources: v1.ResourceRequirements{
  60. Requests: v1.ResourceList{
  61. v1.ResourceName(v1.ResourceStorage): resource.MustParse(size),
  62. },
  63. },
  64. },
  65. }
  66. if modfn != nil {
  67. modfn(&pvc)
  68. }
  69. return &pvc
  70. }
  71. func TestMatchVolume(t *testing.T) {
  72. volList := newPersistentVolumeOrderedIndex()
  73. for _, pv := range createTestVolumes() {
  74. volList.store.Add(pv)
  75. }
  76. scenarios := map[string]struct {
  77. expectedMatch string
  78. claim *v1.PersistentVolumeClaim
  79. }{
  80. "successful-match-gce-10": {
  81. expectedMatch: "gce-pd-10",
  82. claim: makePVC("8G", nil),
  83. },
  84. "successful-match-nfs-5": {
  85. expectedMatch: "nfs-5",
  86. claim: makePVC("5G", func(pvc *v1.PersistentVolumeClaim) {
  87. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany, v1.ReadWriteOnce, v1.ReadWriteMany}
  88. }),
  89. },
  90. "successful-skip-1g-bound-volume": {
  91. expectedMatch: "gce-pd-5",
  92. claim: makePVC("1G", nil),
  93. },
  94. "successful-no-match": {
  95. expectedMatch: "",
  96. claim: makePVC("999G", nil),
  97. },
  98. "successful-no-match-due-to-label": {
  99. expectedMatch: "",
  100. claim: makePVC("999G", func(pvc *v1.PersistentVolumeClaim) {
  101. pvc.Spec.Selector = &metav1.LabelSelector{
  102. MatchLabels: map[string]string{
  103. "should-not-exist": "true",
  104. },
  105. }
  106. }),
  107. },
  108. "successful-no-match-due-to-size-constraint-with-label-selector": {
  109. expectedMatch: "",
  110. claim: makePVC("20000G", func(pvc *v1.PersistentVolumeClaim) {
  111. pvc.Spec.Selector = &metav1.LabelSelector{
  112. MatchLabels: map[string]string{
  113. "should-exist": "true",
  114. },
  115. }
  116. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany, v1.ReadWriteOnce}
  117. }),
  118. },
  119. "successful-match-due-with-constraint-and-label-selector": {
  120. expectedMatch: "gce-pd-2",
  121. claim: makePVC("20000G", func(pvc *v1.PersistentVolumeClaim) {
  122. pvc.Spec.Selector = &metav1.LabelSelector{
  123. MatchLabels: map[string]string{
  124. "should-exist": "true",
  125. },
  126. }
  127. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  128. }),
  129. },
  130. "successful-match-with-class": {
  131. expectedMatch: "gce-pd-silver1",
  132. claim: makePVC("1G", func(pvc *v1.PersistentVolumeClaim) {
  133. pvc.Spec.Selector = &metav1.LabelSelector{
  134. MatchLabels: map[string]string{
  135. "should-exist": "true",
  136. },
  137. }
  138. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  139. pvc.Spec.StorageClassName = &classSilver
  140. }),
  141. },
  142. "successful-match-with-class-and-labels": {
  143. expectedMatch: "gce-pd-silver2",
  144. claim: makePVC("1G", func(pvc *v1.PersistentVolumeClaim) {
  145. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  146. pvc.Spec.StorageClassName = &classSilver
  147. }),
  148. },
  149. "successful-match-very-large": {
  150. expectedMatch: "local-pd-very-large",
  151. // we keep the pvc size less than int64 so that in case the pv overflows
  152. // the pvc does not overflow equally and give us false matching signals.
  153. claim: makePVC("1E", func(pvc *v1.PersistentVolumeClaim) {
  154. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  155. pvc.Spec.StorageClassName = &classLarge
  156. }),
  157. },
  158. "successful-match-exact-extremely-large": {
  159. expectedMatch: "local-pd-extremely-large",
  160. claim: makePVC("800E", func(pvc *v1.PersistentVolumeClaim) {
  161. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  162. pvc.Spec.StorageClassName = &classLarge
  163. }),
  164. },
  165. "successful-no-match-way-too-large": {
  166. expectedMatch: "",
  167. claim: makePVC("950E", func(pvc *v1.PersistentVolumeClaim) {
  168. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  169. pvc.Spec.StorageClassName = &classLarge
  170. }),
  171. },
  172. }
  173. for name, scenario := range scenarios {
  174. volume, err := volList.findBestMatchForClaim(scenario.claim, false)
  175. if err != nil {
  176. t.Errorf("Unexpected error matching volume by claim: %v", err)
  177. }
  178. if len(scenario.expectedMatch) != 0 && volume == nil {
  179. t.Errorf("Expected match but received nil volume for scenario: %s", name)
  180. }
  181. if len(scenario.expectedMatch) != 0 && volume != nil && string(volume.UID) != scenario.expectedMatch {
  182. t.Errorf("Expected %s but got volume %s in scenario %s", scenario.expectedMatch, volume.UID, name)
  183. }
  184. if len(scenario.expectedMatch) == 0 && volume != nil {
  185. t.Errorf("Unexpected match for scenario: %s, matched with %s instead", name, volume.UID)
  186. }
  187. }
  188. }
  189. func TestMatchingWithBoundVolumes(t *testing.T) {
  190. fs := v1.PersistentVolumeFilesystem
  191. volumeIndex := newPersistentVolumeOrderedIndex()
  192. // two similar volumes, one is bound
  193. pv1 := &v1.PersistentVolume{
  194. ObjectMeta: metav1.ObjectMeta{
  195. UID: "gce-pd-1",
  196. Name: "gce001",
  197. },
  198. Spec: v1.PersistentVolumeSpec{
  199. Capacity: v1.ResourceList{
  200. v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G"),
  201. },
  202. PersistentVolumeSource: v1.PersistentVolumeSource{
  203. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  204. },
  205. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadOnlyMany},
  206. // this one we're pretending is already bound
  207. ClaimRef: &v1.ObjectReference{UID: "abc123"},
  208. VolumeMode: &fs,
  209. },
  210. Status: v1.PersistentVolumeStatus{
  211. Phase: v1.VolumeBound,
  212. },
  213. }
  214. pv2 := &v1.PersistentVolume{
  215. ObjectMeta: metav1.ObjectMeta{
  216. UID: "gce-pd-2",
  217. Name: "gce002",
  218. },
  219. Spec: v1.PersistentVolumeSpec{
  220. Capacity: v1.ResourceList{
  221. v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G"),
  222. },
  223. PersistentVolumeSource: v1.PersistentVolumeSource{
  224. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  225. },
  226. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadOnlyMany},
  227. VolumeMode: &fs,
  228. },
  229. Status: v1.PersistentVolumeStatus{
  230. Phase: v1.VolumeAvailable,
  231. },
  232. }
  233. volumeIndex.store.Add(pv1)
  234. volumeIndex.store.Add(pv2)
  235. claim := &v1.PersistentVolumeClaim{
  236. ObjectMeta: metav1.ObjectMeta{
  237. Name: "claim01",
  238. Namespace: "myns",
  239. },
  240. Spec: v1.PersistentVolumeClaimSpec{
  241. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany, v1.ReadWriteOnce},
  242. Resources: v1.ResourceRequirements{
  243. Requests: v1.ResourceList{
  244. v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G"),
  245. },
  246. },
  247. VolumeMode: &fs,
  248. },
  249. }
  250. volume, err := volumeIndex.findBestMatchForClaim(claim, false)
  251. if err != nil {
  252. t.Fatalf("Unexpected error matching volume by claim: %v", err)
  253. }
  254. if volume == nil {
  255. t.Fatalf("Unexpected nil volume. Expected %s", pv2.Name)
  256. }
  257. if pv2.Name != volume.Name {
  258. t.Errorf("Expected %s but got volume %s instead", pv2.Name, volume.Name)
  259. }
  260. }
  261. func TestListByAccessModes(t *testing.T) {
  262. volList := newPersistentVolumeOrderedIndex()
  263. for _, pv := range createTestVolumes() {
  264. volList.store.Add(pv)
  265. }
  266. volumes, err := volList.listByAccessModes([]v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadOnlyMany})
  267. if err != nil {
  268. t.Error("Unexpected error retrieving volumes by access modes:", err)
  269. }
  270. sort.Sort(byCapacity{volumes})
  271. for i, expected := range []string{"gce-pd-1", "gce-pd-5", "gce-pd-10"} {
  272. if string(volumes[i].UID) != expected {
  273. t.Errorf("Incorrect ordering of persistent volumes. Expected %s but got %s", expected, volumes[i].UID)
  274. }
  275. }
  276. volumes, err = volList.listByAccessModes([]v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadOnlyMany, v1.ReadWriteMany})
  277. if err != nil {
  278. t.Error("Unexpected error retrieving volumes by access modes:", err)
  279. }
  280. sort.Sort(byCapacity{volumes})
  281. for i, expected := range []string{"nfs-1", "nfs-5", "nfs-10", "local-pd-very-large", "local-pd-extremely-large"} {
  282. if string(volumes[i].UID) != expected {
  283. t.Errorf("Incorrect ordering of persistent volumes. Expected %s but got %s", expected, volumes[i].UID)
  284. }
  285. }
  286. }
  287. func TestAllPossibleAccessModes(t *testing.T) {
  288. index := newPersistentVolumeOrderedIndex()
  289. for _, pv := range createTestVolumes() {
  290. index.store.Add(pv)
  291. }
  292. // the mock PVs creates contain 2 types of accessmodes: RWO+ROX and RWO+ROW+RWX
  293. possibleModes := index.allPossibleMatchingAccessModes([]v1.PersistentVolumeAccessMode{v1.ReadWriteOnce})
  294. if len(possibleModes) != 3 {
  295. t.Errorf("Expected 3 arrays of modes that match RWO, but got %v", len(possibleModes))
  296. }
  297. for _, m := range possibleModes {
  298. if !util.AccessModesContains(m, v1.ReadWriteOnce) {
  299. t.Errorf("AccessModes does not contain %s", v1.ReadWriteOnce)
  300. }
  301. }
  302. possibleModes = index.allPossibleMatchingAccessModes([]v1.PersistentVolumeAccessMode{v1.ReadWriteMany})
  303. if len(possibleModes) != 1 {
  304. t.Errorf("Expected 1 array of modes that match RWX, but got %v", len(possibleModes))
  305. }
  306. if !util.AccessModesContains(possibleModes[0], v1.ReadWriteMany) {
  307. t.Errorf("AccessModes does not contain %s", v1.ReadWriteOnce)
  308. }
  309. }
  310. func TestFindingVolumeWithDifferentAccessModes(t *testing.T) {
  311. fs := v1.PersistentVolumeFilesystem
  312. gce := &v1.PersistentVolume{
  313. ObjectMeta: metav1.ObjectMeta{UID: "001", Name: "gce"},
  314. Spec: v1.PersistentVolumeSpec{
  315. Capacity: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G")},
  316. PersistentVolumeSource: v1.PersistentVolumeSource{GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{}},
  317. AccessModes: []v1.PersistentVolumeAccessMode{
  318. v1.ReadWriteOnce,
  319. v1.ReadOnlyMany,
  320. },
  321. VolumeMode: &fs,
  322. },
  323. Status: v1.PersistentVolumeStatus{
  324. Phase: v1.VolumeAvailable,
  325. },
  326. }
  327. ebs := &v1.PersistentVolume{
  328. ObjectMeta: metav1.ObjectMeta{UID: "002", Name: "ebs"},
  329. Spec: v1.PersistentVolumeSpec{
  330. Capacity: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G")},
  331. PersistentVolumeSource: v1.PersistentVolumeSource{AWSElasticBlockStore: &v1.AWSElasticBlockStoreVolumeSource{}},
  332. AccessModes: []v1.PersistentVolumeAccessMode{
  333. v1.ReadWriteOnce,
  334. },
  335. VolumeMode: &fs,
  336. },
  337. Status: v1.PersistentVolumeStatus{
  338. Phase: v1.VolumeAvailable,
  339. },
  340. }
  341. nfs := &v1.PersistentVolume{
  342. ObjectMeta: metav1.ObjectMeta{UID: "003", Name: "nfs"},
  343. Spec: v1.PersistentVolumeSpec{
  344. Capacity: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G")},
  345. PersistentVolumeSource: v1.PersistentVolumeSource{NFS: &v1.NFSVolumeSource{}},
  346. AccessModes: []v1.PersistentVolumeAccessMode{
  347. v1.ReadWriteOnce,
  348. v1.ReadOnlyMany,
  349. v1.ReadWriteMany,
  350. },
  351. VolumeMode: &fs,
  352. },
  353. Status: v1.PersistentVolumeStatus{
  354. Phase: v1.VolumeAvailable,
  355. },
  356. }
  357. claim := &v1.PersistentVolumeClaim{
  358. ObjectMeta: metav1.ObjectMeta{
  359. Name: "claim01",
  360. Namespace: "myns",
  361. },
  362. Spec: v1.PersistentVolumeClaimSpec{
  363. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  364. Resources: v1.ResourceRequirements{Requests: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G")}},
  365. VolumeMode: &fs,
  366. },
  367. }
  368. index := newPersistentVolumeOrderedIndex()
  369. index.store.Add(gce)
  370. index.store.Add(ebs)
  371. index.store.Add(nfs)
  372. volume, _ := index.findBestMatchForClaim(claim, false)
  373. if volume.Name != ebs.Name {
  374. t.Errorf("Expected %s but got volume %s instead", ebs.Name, volume.Name)
  375. }
  376. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadOnlyMany}
  377. volume, _ = index.findBestMatchForClaim(claim, false)
  378. if volume.Name != gce.Name {
  379. t.Errorf("Expected %s but got volume %s instead", gce.Name, volume.Name)
  380. }
  381. // order of the requested modes should not matter
  382. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany, v1.ReadWriteOnce, v1.ReadOnlyMany}
  383. volume, _ = index.findBestMatchForClaim(claim, false)
  384. if volume.Name != nfs.Name {
  385. t.Errorf("Expected %s but got volume %s instead", nfs.Name, volume.Name)
  386. }
  387. // fewer modes requested should still match
  388. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany}
  389. volume, _ = index.findBestMatchForClaim(claim, false)
  390. if volume.Name != nfs.Name {
  391. t.Errorf("Expected %s but got volume %s instead", nfs.Name, volume.Name)
  392. }
  393. // pretend the exact match is bound. should get the next level up of modes.
  394. ebs.Spec.ClaimRef = &v1.ObjectReference{}
  395. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  396. volume, _ = index.findBestMatchForClaim(claim, false)
  397. if volume.Name != gce.Name {
  398. t.Errorf("Expected %s but got volume %s instead", gce.Name, volume.Name)
  399. }
  400. // continue up the levels of modes.
  401. gce.Spec.ClaimRef = &v1.ObjectReference{}
  402. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  403. volume, _ = index.findBestMatchForClaim(claim, false)
  404. if volume.Name != nfs.Name {
  405. t.Errorf("Expected %s but got volume %s instead", nfs.Name, volume.Name)
  406. }
  407. // partial mode request
  408. gce.Spec.ClaimRef = nil
  409. claim.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany}
  410. volume, _ = index.findBestMatchForClaim(claim, false)
  411. if volume.Name != gce.Name {
  412. t.Errorf("Expected %s but got volume %s instead", gce.Name, volume.Name)
  413. }
  414. }
  415. func createTestVolumes() []*v1.PersistentVolume {
  416. fs := v1.PersistentVolumeFilesystem
  417. // these volumes are deliberately out-of-order to test indexing and sorting
  418. return []*v1.PersistentVolume{
  419. {
  420. ObjectMeta: metav1.ObjectMeta{
  421. UID: "gce-pd-10",
  422. Name: "gce003",
  423. },
  424. Spec: v1.PersistentVolumeSpec{
  425. Capacity: v1.ResourceList{
  426. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G"),
  427. },
  428. PersistentVolumeSource: v1.PersistentVolumeSource{
  429. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  430. },
  431. AccessModes: []v1.PersistentVolumeAccessMode{
  432. v1.ReadWriteOnce,
  433. v1.ReadOnlyMany,
  434. },
  435. VolumeMode: &fs,
  436. },
  437. Status: v1.PersistentVolumeStatus{
  438. Phase: v1.VolumeAvailable,
  439. },
  440. },
  441. {
  442. ObjectMeta: metav1.ObjectMeta{
  443. UID: "gce-pd-20",
  444. Name: "gce004",
  445. },
  446. Spec: v1.PersistentVolumeSpec{
  447. Capacity: v1.ResourceList{
  448. v1.ResourceName(v1.ResourceStorage): resource.MustParse("20G"),
  449. },
  450. PersistentVolumeSource: v1.PersistentVolumeSource{
  451. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  452. },
  453. AccessModes: []v1.PersistentVolumeAccessMode{
  454. v1.ReadWriteOnce,
  455. v1.ReadOnlyMany,
  456. },
  457. // this one we're pretending is already bound
  458. ClaimRef: &v1.ObjectReference{UID: "def456"},
  459. VolumeMode: &fs,
  460. },
  461. Status: v1.PersistentVolumeStatus{
  462. Phase: v1.VolumeBound,
  463. },
  464. },
  465. {
  466. ObjectMeta: metav1.ObjectMeta{
  467. UID: "nfs-5",
  468. Name: "nfs002",
  469. },
  470. Spec: v1.PersistentVolumeSpec{
  471. Capacity: v1.ResourceList{
  472. v1.ResourceName(v1.ResourceStorage): resource.MustParse("5G"),
  473. },
  474. PersistentVolumeSource: v1.PersistentVolumeSource{
  475. Glusterfs: &v1.GlusterfsPersistentVolumeSource{},
  476. },
  477. AccessModes: []v1.PersistentVolumeAccessMode{
  478. v1.ReadWriteOnce,
  479. v1.ReadOnlyMany,
  480. v1.ReadWriteMany,
  481. },
  482. VolumeMode: &fs,
  483. },
  484. Status: v1.PersistentVolumeStatus{
  485. Phase: v1.VolumeAvailable,
  486. },
  487. },
  488. {
  489. ObjectMeta: metav1.ObjectMeta{
  490. UID: "gce-pd-1",
  491. Name: "gce001",
  492. },
  493. Spec: v1.PersistentVolumeSpec{
  494. Capacity: v1.ResourceList{
  495. v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G"),
  496. },
  497. PersistentVolumeSource: v1.PersistentVolumeSource{
  498. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  499. },
  500. AccessModes: []v1.PersistentVolumeAccessMode{
  501. v1.ReadWriteOnce,
  502. v1.ReadOnlyMany,
  503. },
  504. // this one we're pretending is already bound
  505. ClaimRef: &v1.ObjectReference{UID: "abc123"},
  506. VolumeMode: &fs,
  507. },
  508. Status: v1.PersistentVolumeStatus{
  509. Phase: v1.VolumeBound,
  510. },
  511. },
  512. {
  513. ObjectMeta: metav1.ObjectMeta{
  514. UID: "nfs-10",
  515. Name: "nfs003",
  516. },
  517. Spec: v1.PersistentVolumeSpec{
  518. Capacity: v1.ResourceList{
  519. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G"),
  520. },
  521. PersistentVolumeSource: v1.PersistentVolumeSource{
  522. Glusterfs: &v1.GlusterfsPersistentVolumeSource{},
  523. },
  524. AccessModes: []v1.PersistentVolumeAccessMode{
  525. v1.ReadWriteOnce,
  526. v1.ReadOnlyMany,
  527. v1.ReadWriteMany,
  528. },
  529. VolumeMode: &fs,
  530. },
  531. Status: v1.PersistentVolumeStatus{
  532. Phase: v1.VolumeAvailable,
  533. },
  534. },
  535. {
  536. ObjectMeta: metav1.ObjectMeta{
  537. UID: "gce-pd-5",
  538. Name: "gce002",
  539. },
  540. Spec: v1.PersistentVolumeSpec{
  541. Capacity: v1.ResourceList{
  542. v1.ResourceName(v1.ResourceStorage): resource.MustParse("5G"),
  543. },
  544. PersistentVolumeSource: v1.PersistentVolumeSource{
  545. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  546. },
  547. AccessModes: []v1.PersistentVolumeAccessMode{
  548. v1.ReadWriteOnce,
  549. v1.ReadOnlyMany,
  550. },
  551. VolumeMode: &fs,
  552. },
  553. Status: v1.PersistentVolumeStatus{
  554. Phase: v1.VolumeAvailable,
  555. },
  556. },
  557. {
  558. ObjectMeta: metav1.ObjectMeta{
  559. UID: "nfs-1",
  560. Name: "nfs001",
  561. },
  562. Spec: v1.PersistentVolumeSpec{
  563. Capacity: v1.ResourceList{
  564. v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G"),
  565. },
  566. PersistentVolumeSource: v1.PersistentVolumeSource{
  567. Glusterfs: &v1.GlusterfsPersistentVolumeSource{},
  568. },
  569. AccessModes: []v1.PersistentVolumeAccessMode{
  570. v1.ReadWriteOnce,
  571. v1.ReadOnlyMany,
  572. v1.ReadWriteMany,
  573. },
  574. VolumeMode: &fs,
  575. },
  576. Status: v1.PersistentVolumeStatus{
  577. Phase: v1.VolumeAvailable,
  578. },
  579. },
  580. {
  581. ObjectMeta: metav1.ObjectMeta{
  582. UID: "gce-pd-2",
  583. Name: "gce0022",
  584. Labels: map[string]string{
  585. "should-exist": "true",
  586. },
  587. },
  588. Spec: v1.PersistentVolumeSpec{
  589. Capacity: v1.ResourceList{
  590. v1.ResourceName(v1.ResourceStorage): resource.MustParse("20000G"),
  591. },
  592. PersistentVolumeSource: v1.PersistentVolumeSource{
  593. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  594. },
  595. AccessModes: []v1.PersistentVolumeAccessMode{
  596. v1.ReadWriteOnce,
  597. },
  598. VolumeMode: &fs,
  599. },
  600. Status: v1.PersistentVolumeStatus{
  601. Phase: v1.VolumeAvailable,
  602. },
  603. },
  604. {
  605. ObjectMeta: metav1.ObjectMeta{
  606. UID: "gce-pd-silver1",
  607. Name: "gce0023",
  608. Labels: map[string]string{
  609. "should-exist": "true",
  610. },
  611. },
  612. Spec: v1.PersistentVolumeSpec{
  613. Capacity: v1.ResourceList{
  614. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10000G"),
  615. },
  616. PersistentVolumeSource: v1.PersistentVolumeSource{
  617. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  618. },
  619. AccessModes: []v1.PersistentVolumeAccessMode{
  620. v1.ReadWriteOnce,
  621. },
  622. StorageClassName: classSilver,
  623. VolumeMode: &fs,
  624. },
  625. Status: v1.PersistentVolumeStatus{
  626. Phase: v1.VolumeAvailable,
  627. },
  628. },
  629. {
  630. ObjectMeta: metav1.ObjectMeta{
  631. UID: "gce-pd-silver2",
  632. Name: "gce0024",
  633. },
  634. Spec: v1.PersistentVolumeSpec{
  635. Capacity: v1.ResourceList{
  636. v1.ResourceName(v1.ResourceStorage): resource.MustParse("100G"),
  637. },
  638. PersistentVolumeSource: v1.PersistentVolumeSource{
  639. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  640. },
  641. AccessModes: []v1.PersistentVolumeAccessMode{
  642. v1.ReadWriteOnce,
  643. },
  644. StorageClassName: classSilver,
  645. VolumeMode: &fs,
  646. },
  647. Status: v1.PersistentVolumeStatus{
  648. Phase: v1.VolumeAvailable,
  649. },
  650. },
  651. {
  652. ObjectMeta: metav1.ObjectMeta{
  653. UID: "gce-pd-gold",
  654. Name: "gce0025",
  655. },
  656. Spec: v1.PersistentVolumeSpec{
  657. Capacity: v1.ResourceList{
  658. v1.ResourceName(v1.ResourceStorage): resource.MustParse("50G"),
  659. },
  660. PersistentVolumeSource: v1.PersistentVolumeSource{
  661. GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{},
  662. },
  663. AccessModes: []v1.PersistentVolumeAccessMode{
  664. v1.ReadWriteOnce,
  665. },
  666. StorageClassName: classGold,
  667. VolumeMode: &fs,
  668. },
  669. Status: v1.PersistentVolumeStatus{
  670. Phase: v1.VolumeAvailable,
  671. },
  672. },
  673. {
  674. ObjectMeta: metav1.ObjectMeta{
  675. UID: "local-pd-very-large",
  676. Name: "local001",
  677. },
  678. Spec: v1.PersistentVolumeSpec{
  679. Capacity: v1.ResourceList{
  680. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200E"),
  681. },
  682. PersistentVolumeSource: v1.PersistentVolumeSource{
  683. Local: &v1.LocalVolumeSource{},
  684. },
  685. AccessModes: []v1.PersistentVolumeAccessMode{
  686. v1.ReadWriteOnce,
  687. v1.ReadOnlyMany,
  688. v1.ReadWriteMany,
  689. },
  690. StorageClassName: classLarge,
  691. VolumeMode: &fs,
  692. },
  693. Status: v1.PersistentVolumeStatus{
  694. Phase: v1.VolumeAvailable,
  695. },
  696. },
  697. {
  698. ObjectMeta: metav1.ObjectMeta{
  699. UID: "local-pd-extremely-large",
  700. Name: "local002",
  701. },
  702. Spec: v1.PersistentVolumeSpec{
  703. Capacity: v1.ResourceList{
  704. v1.ResourceName(v1.ResourceStorage): resource.MustParse("800E"),
  705. },
  706. PersistentVolumeSource: v1.PersistentVolumeSource{
  707. Local: &v1.LocalVolumeSource{},
  708. },
  709. AccessModes: []v1.PersistentVolumeAccessMode{
  710. v1.ReadWriteOnce,
  711. v1.ReadOnlyMany,
  712. v1.ReadWriteMany,
  713. },
  714. StorageClassName: classLarge,
  715. VolumeMode: &fs,
  716. },
  717. Status: v1.PersistentVolumeStatus{
  718. Phase: v1.VolumeAvailable,
  719. },
  720. },
  721. {
  722. ObjectMeta: metav1.ObjectMeta{
  723. UID: "affinity-pv",
  724. Name: "affinity001",
  725. },
  726. Spec: v1.PersistentVolumeSpec{
  727. Capacity: v1.ResourceList{
  728. v1.ResourceName(v1.ResourceStorage): resource.MustParse("100G"),
  729. },
  730. PersistentVolumeSource: v1.PersistentVolumeSource{
  731. Local: &v1.LocalVolumeSource{},
  732. },
  733. AccessModes: []v1.PersistentVolumeAccessMode{
  734. v1.ReadWriteOnce,
  735. v1.ReadOnlyMany,
  736. },
  737. StorageClassName: classWait,
  738. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value1"),
  739. VolumeMode: &fs,
  740. },
  741. Status: v1.PersistentVolumeStatus{
  742. Phase: v1.VolumeAvailable,
  743. },
  744. },
  745. {
  746. ObjectMeta: metav1.ObjectMeta{
  747. UID: "affinity-pv2",
  748. Name: "affinity002",
  749. },
  750. Spec: v1.PersistentVolumeSpec{
  751. Capacity: v1.ResourceList{
  752. v1.ResourceName(v1.ResourceStorage): resource.MustParse("150G"),
  753. },
  754. PersistentVolumeSource: v1.PersistentVolumeSource{
  755. Local: &v1.LocalVolumeSource{},
  756. },
  757. AccessModes: []v1.PersistentVolumeAccessMode{
  758. v1.ReadWriteOnce,
  759. v1.ReadOnlyMany,
  760. },
  761. StorageClassName: classWait,
  762. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value1"),
  763. VolumeMode: &fs,
  764. },
  765. Status: v1.PersistentVolumeStatus{
  766. Phase: v1.VolumeAvailable,
  767. },
  768. },
  769. {
  770. ObjectMeta: metav1.ObjectMeta{
  771. UID: "affinity-prebound",
  772. Name: "affinity003",
  773. },
  774. Spec: v1.PersistentVolumeSpec{
  775. Capacity: v1.ResourceList{
  776. v1.ResourceName(v1.ResourceStorage): resource.MustParse("100G"),
  777. },
  778. PersistentVolumeSource: v1.PersistentVolumeSource{
  779. Local: &v1.LocalVolumeSource{},
  780. },
  781. AccessModes: []v1.PersistentVolumeAccessMode{
  782. v1.ReadWriteOnce,
  783. v1.ReadOnlyMany,
  784. },
  785. StorageClassName: classWait,
  786. ClaimRef: &v1.ObjectReference{Name: "claim02", Namespace: "myns"},
  787. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value1"),
  788. VolumeMode: &fs,
  789. },
  790. Status: v1.PersistentVolumeStatus{
  791. Phase: v1.VolumeAvailable,
  792. },
  793. },
  794. {
  795. ObjectMeta: metav1.ObjectMeta{
  796. UID: "affinity-pv3",
  797. Name: "affinity003",
  798. },
  799. Spec: v1.PersistentVolumeSpec{
  800. Capacity: v1.ResourceList{
  801. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200G"),
  802. },
  803. PersistentVolumeSource: v1.PersistentVolumeSource{
  804. Local: &v1.LocalVolumeSource{},
  805. },
  806. AccessModes: []v1.PersistentVolumeAccessMode{
  807. v1.ReadWriteOnce,
  808. v1.ReadOnlyMany,
  809. },
  810. StorageClassName: classWait,
  811. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value3"),
  812. VolumeMode: &fs,
  813. },
  814. Status: v1.PersistentVolumeStatus{
  815. Phase: v1.VolumeAvailable,
  816. },
  817. },
  818. {
  819. ObjectMeta: metav1.ObjectMeta{
  820. UID: "affinity-pv4-pending",
  821. Name: "affinity004-pending",
  822. },
  823. Spec: v1.PersistentVolumeSpec{
  824. Capacity: v1.ResourceList{
  825. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200G"),
  826. },
  827. PersistentVolumeSource: v1.PersistentVolumeSource{
  828. Local: &v1.LocalVolumeSource{},
  829. },
  830. AccessModes: []v1.PersistentVolumeAccessMode{
  831. v1.ReadWriteOnce,
  832. v1.ReadOnlyMany,
  833. },
  834. StorageClassName: classWait,
  835. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value4"),
  836. VolumeMode: &fs,
  837. },
  838. Status: v1.PersistentVolumeStatus{
  839. Phase: v1.VolumePending,
  840. },
  841. },
  842. {
  843. ObjectMeta: metav1.ObjectMeta{
  844. UID: "affinity-pv4-failed",
  845. Name: "affinity004-failed",
  846. },
  847. Spec: v1.PersistentVolumeSpec{
  848. Capacity: v1.ResourceList{
  849. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200G"),
  850. },
  851. PersistentVolumeSource: v1.PersistentVolumeSource{
  852. Local: &v1.LocalVolumeSource{},
  853. },
  854. AccessModes: []v1.PersistentVolumeAccessMode{
  855. v1.ReadWriteOnce,
  856. v1.ReadOnlyMany,
  857. },
  858. StorageClassName: classWait,
  859. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value4"),
  860. VolumeMode: &fs,
  861. },
  862. Status: v1.PersistentVolumeStatus{
  863. Phase: v1.VolumeFailed,
  864. },
  865. },
  866. {
  867. ObjectMeta: metav1.ObjectMeta{
  868. UID: "affinity-pv4-released",
  869. Name: "affinity004-released",
  870. },
  871. Spec: v1.PersistentVolumeSpec{
  872. Capacity: v1.ResourceList{
  873. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200G"),
  874. },
  875. PersistentVolumeSource: v1.PersistentVolumeSource{
  876. Local: &v1.LocalVolumeSource{},
  877. },
  878. AccessModes: []v1.PersistentVolumeAccessMode{
  879. v1.ReadWriteOnce,
  880. v1.ReadOnlyMany,
  881. },
  882. StorageClassName: classWait,
  883. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value4"),
  884. VolumeMode: &fs,
  885. },
  886. Status: v1.PersistentVolumeStatus{
  887. Phase: v1.VolumeReleased,
  888. },
  889. },
  890. {
  891. ObjectMeta: metav1.ObjectMeta{
  892. UID: "affinity-pv4-empty",
  893. Name: "affinity004-empty",
  894. },
  895. Spec: v1.PersistentVolumeSpec{
  896. Capacity: v1.ResourceList{
  897. v1.ResourceName(v1.ResourceStorage): resource.MustParse("200G"),
  898. },
  899. PersistentVolumeSource: v1.PersistentVolumeSource{
  900. Local: &v1.LocalVolumeSource{},
  901. },
  902. AccessModes: []v1.PersistentVolumeAccessMode{
  903. v1.ReadWriteOnce,
  904. v1.ReadOnlyMany,
  905. },
  906. StorageClassName: classWait,
  907. NodeAffinity: pvutil.GetVolumeNodeAffinity("key1", "value4"),
  908. VolumeMode: &fs,
  909. },
  910. },
  911. }
  912. }
  913. func testVolume(name, size string) *v1.PersistentVolume {
  914. fs := v1.PersistentVolumeFilesystem
  915. return &v1.PersistentVolume{
  916. ObjectMeta: metav1.ObjectMeta{
  917. Name: name,
  918. Annotations: map[string]string{},
  919. },
  920. Spec: v1.PersistentVolumeSpec{
  921. Capacity: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse(size)},
  922. PersistentVolumeSource: v1.PersistentVolumeSource{HostPath: &v1.HostPathVolumeSource{}},
  923. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  924. VolumeMode: &fs,
  925. },
  926. Status: v1.PersistentVolumeStatus{
  927. Phase: v1.VolumeAvailable,
  928. },
  929. }
  930. }
  931. func createVolumeModeBlockTestVolume() *v1.PersistentVolume {
  932. blockMode := v1.PersistentVolumeBlock
  933. return &v1.PersistentVolume{
  934. ObjectMeta: metav1.ObjectMeta{
  935. UID: "local-1",
  936. Name: "block",
  937. },
  938. Spec: v1.PersistentVolumeSpec{
  939. Capacity: v1.ResourceList{
  940. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G"),
  941. },
  942. PersistentVolumeSource: v1.PersistentVolumeSource{
  943. Local: &v1.LocalVolumeSource{},
  944. },
  945. AccessModes: []v1.PersistentVolumeAccessMode{
  946. v1.ReadWriteOnce,
  947. },
  948. VolumeMode: &blockMode,
  949. },
  950. Status: v1.PersistentVolumeStatus{
  951. Phase: v1.VolumeAvailable,
  952. },
  953. }
  954. }
  955. func createVolumeModeFilesystemTestVolume() *v1.PersistentVolume {
  956. filesystemMode := v1.PersistentVolumeFilesystem
  957. return &v1.PersistentVolume{
  958. ObjectMeta: metav1.ObjectMeta{
  959. UID: "local-1",
  960. Name: "block",
  961. },
  962. Spec: v1.PersistentVolumeSpec{
  963. Capacity: v1.ResourceList{
  964. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G"),
  965. },
  966. PersistentVolumeSource: v1.PersistentVolumeSource{
  967. Local: &v1.LocalVolumeSource{},
  968. },
  969. AccessModes: []v1.PersistentVolumeAccessMode{
  970. v1.ReadWriteOnce,
  971. },
  972. VolumeMode: &filesystemMode,
  973. },
  974. Status: v1.PersistentVolumeStatus{
  975. Phase: v1.VolumeAvailable,
  976. },
  977. }
  978. }
  979. func createVolumeModeNilTestVolume() *v1.PersistentVolume {
  980. return &v1.PersistentVolume{
  981. ObjectMeta: metav1.ObjectMeta{
  982. UID: "local-1",
  983. Name: "nil-mode",
  984. },
  985. Spec: v1.PersistentVolumeSpec{
  986. Capacity: v1.ResourceList{
  987. v1.ResourceName(v1.ResourceStorage): resource.MustParse("10G"),
  988. },
  989. PersistentVolumeSource: v1.PersistentVolumeSource{
  990. Local: &v1.LocalVolumeSource{},
  991. },
  992. AccessModes: []v1.PersistentVolumeAccessMode{
  993. v1.ReadWriteOnce,
  994. },
  995. },
  996. Status: v1.PersistentVolumeStatus{
  997. Phase: v1.VolumeAvailable,
  998. },
  999. }
  1000. }
  1001. func createTestVolOrderedIndex(pv *v1.PersistentVolume) persistentVolumeOrderedIndex {
  1002. volFile := newPersistentVolumeOrderedIndex()
  1003. volFile.store.Add(pv)
  1004. return volFile
  1005. }
  1006. func TestVolumeModeCheck(t *testing.T) {
  1007. blockMode := v1.PersistentVolumeBlock
  1008. filesystemMode := v1.PersistentVolumeFilesystem
  1009. // If feature gate is enabled, VolumeMode will always be defaulted
  1010. // If feature gate is disabled, VolumeMode is dropped by API and ignored
  1011. scenarios := map[string]struct {
  1012. isExpectedMismatch bool
  1013. vol *v1.PersistentVolume
  1014. pvc *v1.PersistentVolumeClaim
  1015. enableBlock bool
  1016. }{
  1017. "feature enabled - pvc block and pv filesystem": {
  1018. isExpectedMismatch: true,
  1019. vol: createVolumeModeFilesystemTestVolume(),
  1020. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1021. enableBlock: true,
  1022. },
  1023. "feature enabled - pvc filesystem and pv block": {
  1024. isExpectedMismatch: true,
  1025. vol: createVolumeModeBlockTestVolume(),
  1026. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1027. enableBlock: true,
  1028. },
  1029. "feature enabled - pvc block and pv block": {
  1030. isExpectedMismatch: false,
  1031. vol: createVolumeModeBlockTestVolume(),
  1032. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1033. enableBlock: true,
  1034. },
  1035. "feature enabled - pvc filesystem and pv filesystem": {
  1036. isExpectedMismatch: false,
  1037. vol: createVolumeModeFilesystemTestVolume(),
  1038. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1039. enableBlock: true,
  1040. },
  1041. "feature enabled - pvc filesystem and pv nil": {
  1042. isExpectedMismatch: false,
  1043. vol: createVolumeModeNilTestVolume(),
  1044. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1045. enableBlock: true,
  1046. },
  1047. "feature enabled - pvc nil and pv filesystem": {
  1048. isExpectedMismatch: false,
  1049. vol: createVolumeModeFilesystemTestVolume(),
  1050. pvc: makeVolumeModePVC("8G", nil, nil),
  1051. enableBlock: true,
  1052. },
  1053. "feature enabled - pvc nil and pv nil": {
  1054. isExpectedMismatch: false,
  1055. vol: createVolumeModeNilTestVolume(),
  1056. pvc: makeVolumeModePVC("8G", nil, nil),
  1057. enableBlock: true,
  1058. },
  1059. "feature enabled - pvc nil and pv block": {
  1060. isExpectedMismatch: true,
  1061. vol: createVolumeModeBlockTestVolume(),
  1062. pvc: makeVolumeModePVC("8G", nil, nil),
  1063. enableBlock: true,
  1064. },
  1065. "feature enabled - pvc block and pv nil": {
  1066. isExpectedMismatch: true,
  1067. vol: createVolumeModeNilTestVolume(),
  1068. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1069. enableBlock: true,
  1070. },
  1071. "feature disabled - pvc block and pv filesystem": {
  1072. isExpectedMismatch: true,
  1073. vol: createVolumeModeFilesystemTestVolume(),
  1074. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1075. enableBlock: false,
  1076. },
  1077. "feature disabled - pvc filesystem and pv block": {
  1078. isExpectedMismatch: true,
  1079. vol: createVolumeModeBlockTestVolume(),
  1080. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1081. enableBlock: false,
  1082. },
  1083. "feature disabled - pvc block and pv block": {
  1084. isExpectedMismatch: true,
  1085. vol: createVolumeModeBlockTestVolume(),
  1086. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1087. enableBlock: false,
  1088. },
  1089. "feature disabled - pvc filesystem and pv filesystem": {
  1090. isExpectedMismatch: false,
  1091. vol: createVolumeModeFilesystemTestVolume(),
  1092. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1093. enableBlock: false,
  1094. },
  1095. }
  1096. for name, scenario := range scenarios {
  1097. t.Run(name, func(t *testing.T) {
  1098. defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.BlockVolume, scenario.enableBlock)()
  1099. expectedMismatch := pvutil.CheckVolumeModeMismatches(&scenario.pvc.Spec, &scenario.vol.Spec)
  1100. // expected to match but either got an error or no returned pvmatch
  1101. if expectedMismatch && !scenario.isExpectedMismatch {
  1102. t.Errorf("Unexpected failure for scenario, expected not to mismatch on modes but did: %s", name)
  1103. }
  1104. if !expectedMismatch && scenario.isExpectedMismatch {
  1105. t.Errorf("Unexpected failure for scenario, did not mismatch on mode when expected to mismatch: %s", name)
  1106. }
  1107. })
  1108. }
  1109. }
  1110. func TestFilteringVolumeModes(t *testing.T) {
  1111. blockMode := v1.PersistentVolumeBlock
  1112. filesystemMode := v1.PersistentVolumeFilesystem
  1113. // If feature gate is enabled, VolumeMode will always be defaulted
  1114. // If feature gate is disabled, VolumeMode is dropped by API and ignored
  1115. scenarios := map[string]struct {
  1116. isExpectedMatch bool
  1117. vol persistentVolumeOrderedIndex
  1118. pvc *v1.PersistentVolumeClaim
  1119. enableBlock bool
  1120. }{
  1121. "1-1 feature enabled - pvc block and pv filesystem": {
  1122. isExpectedMatch: false,
  1123. vol: createTestVolOrderedIndex(createVolumeModeFilesystemTestVolume()),
  1124. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1125. enableBlock: true,
  1126. },
  1127. "1-2 feature enabled - pvc filesystem and pv block": {
  1128. isExpectedMatch: false,
  1129. vol: createTestVolOrderedIndex(createVolumeModeBlockTestVolume()),
  1130. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1131. enableBlock: true,
  1132. },
  1133. "1-3 feature enabled - pvc block and pv no mode with default filesystem": {
  1134. isExpectedMatch: false,
  1135. vol: createTestVolOrderedIndex(createVolumeModeFilesystemTestVolume()),
  1136. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1137. enableBlock: true,
  1138. },
  1139. "1-4 feature enabled - pvc no mode defaulted to filesystem and pv block": {
  1140. isExpectedMatch: false,
  1141. vol: createTestVolOrderedIndex(createVolumeModeBlockTestVolume()),
  1142. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1143. enableBlock: true,
  1144. },
  1145. "1-5 feature enabled - pvc block and pv block": {
  1146. isExpectedMatch: true,
  1147. vol: createTestVolOrderedIndex(createVolumeModeBlockTestVolume()),
  1148. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1149. enableBlock: true,
  1150. },
  1151. "1-6 feature enabled - pvc filesystem and pv filesystem": {
  1152. isExpectedMatch: true,
  1153. vol: createTestVolOrderedIndex(createVolumeModeFilesystemTestVolume()),
  1154. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1155. enableBlock: true,
  1156. },
  1157. "1-7 feature enabled - pvc mode is nil and defaulted and pv mode is nil and defaulted": {
  1158. isExpectedMatch: true,
  1159. vol: createTestVolOrderedIndex(createVolumeModeFilesystemTestVolume()),
  1160. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1161. enableBlock: true,
  1162. },
  1163. "2-1 feature disabled - pvc mode is nil and pv mode is nil": {
  1164. isExpectedMatch: true,
  1165. vol: createTestVolOrderedIndex(testVolume("nomode-1", "8G")),
  1166. pvc: makeVolumeModePVC("8G", nil, nil),
  1167. enableBlock: false,
  1168. },
  1169. "2-2 feature disabled - pvc mode is block and pv mode is block - fields should be dropped by api and not analyzed with gate disabled": {
  1170. isExpectedMatch: false,
  1171. vol: createTestVolOrderedIndex(createVolumeModeBlockTestVolume()),
  1172. pvc: makeVolumeModePVC("8G", &blockMode, nil),
  1173. enableBlock: false,
  1174. },
  1175. "2-3 feature disabled - pvc mode is filesystem and pv mode is filesystem - fields should be dropped by api and not analyzed with gate disabled": {
  1176. isExpectedMatch: true,
  1177. vol: createTestVolOrderedIndex(createVolumeModeFilesystemTestVolume()),
  1178. pvc: makeVolumeModePVC("8G", &filesystemMode, nil),
  1179. enableBlock: false,
  1180. },
  1181. }
  1182. for name, scenario := range scenarios {
  1183. t.Run(name, func(t *testing.T) {
  1184. defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.BlockVolume, scenario.enableBlock)()
  1185. pvmatch, err := scenario.vol.findBestMatchForClaim(scenario.pvc, false)
  1186. // expected to match but either got an error or no returned pvmatch
  1187. if pvmatch == nil && scenario.isExpectedMatch {
  1188. t.Errorf("Unexpected failure for scenario, no matching volume: %s", name)
  1189. }
  1190. if err != nil && scenario.isExpectedMatch {
  1191. t.Errorf("Unexpected failure for scenario: %s - %+v", name, err)
  1192. }
  1193. // expected to not match but either got an error or a returned pvmatch
  1194. if pvmatch != nil && !scenario.isExpectedMatch {
  1195. t.Errorf("Unexpected failure for scenario, expected no matching volume: %s", name)
  1196. }
  1197. if err != nil && !scenario.isExpectedMatch {
  1198. t.Errorf("Unexpected failure for scenario: %s - %+v", name, err)
  1199. }
  1200. })
  1201. }
  1202. }
  1203. func TestStorageObjectInUseProtectionFiltering(t *testing.T) {
  1204. fs := v1.PersistentVolumeFilesystem
  1205. pv := &v1.PersistentVolume{
  1206. ObjectMeta: metav1.ObjectMeta{
  1207. Name: "pv1",
  1208. Annotations: map[string]string{},
  1209. },
  1210. Spec: v1.PersistentVolumeSpec{
  1211. Capacity: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G")},
  1212. PersistentVolumeSource: v1.PersistentVolumeSource{HostPath: &v1.HostPathVolumeSource{}},
  1213. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  1214. VolumeMode: &fs,
  1215. },
  1216. Status: v1.PersistentVolumeStatus{
  1217. Phase: v1.VolumeAvailable,
  1218. },
  1219. }
  1220. pvToDelete := pv.DeepCopy()
  1221. now := metav1.Now()
  1222. pvToDelete.ObjectMeta.DeletionTimestamp = &now
  1223. pvc := &v1.PersistentVolumeClaim{
  1224. ObjectMeta: metav1.ObjectMeta{
  1225. Name: "pvc1",
  1226. Namespace: "myns",
  1227. },
  1228. Spec: v1.PersistentVolumeClaimSpec{
  1229. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  1230. Resources: v1.ResourceRequirements{Requests: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("1G")}},
  1231. VolumeMode: &fs,
  1232. },
  1233. }
  1234. satisfyingTestCases := map[string]struct {
  1235. isExpectedMatch bool
  1236. vol *v1.PersistentVolume
  1237. pvc *v1.PersistentVolumeClaim
  1238. enableStorageObjectInUseProtection bool
  1239. }{
  1240. "feature enabled - pv deletionTimeStamp not set": {
  1241. isExpectedMatch: true,
  1242. vol: pv,
  1243. pvc: pvc,
  1244. enableStorageObjectInUseProtection: true,
  1245. },
  1246. "feature enabled - pv deletionTimeStamp set": {
  1247. isExpectedMatch: false,
  1248. vol: pvToDelete,
  1249. pvc: pvc,
  1250. enableStorageObjectInUseProtection: true,
  1251. },
  1252. "feature disabled - pv deletionTimeStamp not set": {
  1253. isExpectedMatch: true,
  1254. vol: pv,
  1255. pvc: pvc,
  1256. enableStorageObjectInUseProtection: false,
  1257. },
  1258. "feature disabled - pv deletionTimeStamp set": {
  1259. isExpectedMatch: true,
  1260. vol: pvToDelete,
  1261. pvc: pvc,
  1262. enableStorageObjectInUseProtection: false,
  1263. },
  1264. }
  1265. for name, testCase := range satisfyingTestCases {
  1266. t.Run(name, func(t *testing.T) {
  1267. defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.StorageObjectInUseProtection, testCase.enableStorageObjectInUseProtection)()
  1268. err := checkVolumeSatisfyClaim(testCase.vol, testCase.pvc)
  1269. // expected to match but got an error
  1270. if err != nil && testCase.isExpectedMatch {
  1271. t.Errorf("%s: expected to match but got an error: %v", name, err)
  1272. }
  1273. // not expected to match but did
  1274. if err == nil && !testCase.isExpectedMatch {
  1275. t.Errorf("%s: not expected to match but did", name)
  1276. }
  1277. })
  1278. }
  1279. filteringTestCases := map[string]struct {
  1280. isExpectedMatch bool
  1281. vol persistentVolumeOrderedIndex
  1282. pvc *v1.PersistentVolumeClaim
  1283. enableStorageObjectInUseProtection bool
  1284. }{
  1285. "feature enabled - pv deletionTimeStamp not set": {
  1286. isExpectedMatch: true,
  1287. vol: createTestVolOrderedIndex(pv),
  1288. pvc: pvc,
  1289. enableStorageObjectInUseProtection: true,
  1290. },
  1291. "feature enabled - pv deletionTimeStamp set": {
  1292. isExpectedMatch: false,
  1293. vol: createTestVolOrderedIndex(pvToDelete),
  1294. pvc: pvc,
  1295. enableStorageObjectInUseProtection: true,
  1296. },
  1297. "feature disabled - pv deletionTimeStamp not set": {
  1298. isExpectedMatch: true,
  1299. vol: createTestVolOrderedIndex(pv),
  1300. pvc: pvc,
  1301. enableStorageObjectInUseProtection: false,
  1302. },
  1303. "feature disabled - pv deletionTimeStamp set": {
  1304. isExpectedMatch: true,
  1305. vol: createTestVolOrderedIndex(pvToDelete),
  1306. pvc: pvc,
  1307. enableStorageObjectInUseProtection: false,
  1308. },
  1309. }
  1310. for name, testCase := range filteringTestCases {
  1311. t.Run(name, func(t *testing.T) {
  1312. defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.StorageObjectInUseProtection, testCase.enableStorageObjectInUseProtection)()
  1313. pvmatch, err := testCase.vol.findBestMatchForClaim(testCase.pvc, false)
  1314. // expected to match but either got an error or no returned pvmatch
  1315. if pvmatch == nil && testCase.isExpectedMatch {
  1316. t.Errorf("Unexpected failure for testcase, no matching volume: %s", name)
  1317. }
  1318. if err != nil && testCase.isExpectedMatch {
  1319. t.Errorf("Unexpected failure for testcase: %s - %+v", name, err)
  1320. }
  1321. // expected to not match but either got an error or a returned pvmatch
  1322. if pvmatch != nil && !testCase.isExpectedMatch {
  1323. t.Errorf("Unexpected failure for testcase, expected no matching volume: %s", name)
  1324. }
  1325. if err != nil && !testCase.isExpectedMatch {
  1326. t.Errorf("Unexpected failure for testcase: %s - %+v", name, err)
  1327. }
  1328. })
  1329. }
  1330. }
  1331. func TestFindingPreboundVolumes(t *testing.T) {
  1332. fs := v1.PersistentVolumeFilesystem
  1333. claim := &v1.PersistentVolumeClaim{
  1334. ObjectMeta: metav1.ObjectMeta{
  1335. Name: "claim01",
  1336. Namespace: "myns",
  1337. SelfLink: "/api/v1/namespaces/myns/persistentvolumeclaims/claim01",
  1338. },
  1339. Spec: v1.PersistentVolumeClaimSpec{
  1340. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce},
  1341. Resources: v1.ResourceRequirements{Requests: v1.ResourceList{v1.ResourceName(v1.ResourceStorage): resource.MustParse("1Gi")}},
  1342. VolumeMode: &fs,
  1343. },
  1344. }
  1345. claimRef, err := ref.GetReference(scheme.Scheme, claim)
  1346. if err != nil {
  1347. t.Errorf("error getting claimRef: %v", err)
  1348. }
  1349. pv1 := testVolume("pv1", "1Gi")
  1350. pv5 := testVolume("pv5", "5Gi")
  1351. pv8 := testVolume("pv8", "8Gi")
  1352. pvBadSize := testVolume("pvBadSize", "1Mi")
  1353. pvBadMode := testVolume("pvBadMode", "1Gi")
  1354. pvBadMode.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany}
  1355. index := newPersistentVolumeOrderedIndex()
  1356. index.store.Add(pv1)
  1357. index.store.Add(pv5)
  1358. index.store.Add(pv8)
  1359. index.store.Add(pvBadSize)
  1360. index.store.Add(pvBadMode)
  1361. // expected exact match on size
  1362. volume, _ := index.findBestMatchForClaim(claim, false)
  1363. if volume.Name != pv1.Name {
  1364. t.Errorf("Expected %s but got volume %s instead", pv1.Name, volume.Name)
  1365. }
  1366. // pretend the exact match is pre-bound. should get the next size up.
  1367. pv1.Spec.ClaimRef = &v1.ObjectReference{Name: "foo", Namespace: "bar"}
  1368. volume, _ = index.findBestMatchForClaim(claim, false)
  1369. if volume.Name != pv5.Name {
  1370. t.Errorf("Expected %s but got volume %s instead", pv5.Name, volume.Name)
  1371. }
  1372. // pretend the exact match is available but the largest volume is pre-bound to the claim.
  1373. pv1.Spec.ClaimRef = nil
  1374. pv8.Spec.ClaimRef = claimRef
  1375. volume, _ = index.findBestMatchForClaim(claim, false)
  1376. if volume.Name != pv8.Name {
  1377. t.Errorf("Expected %s but got volume %s instead", pv8.Name, volume.Name)
  1378. }
  1379. // pretend the volume with too small a size is pre-bound to the claim. should get the exact match.
  1380. pv8.Spec.ClaimRef = nil
  1381. pvBadSize.Spec.ClaimRef = claimRef
  1382. volume, _ = index.findBestMatchForClaim(claim, false)
  1383. if volume.Name != pv1.Name {
  1384. t.Errorf("Expected %s but got volume %s instead", pv1.Name, volume.Name)
  1385. }
  1386. // pretend the volume without the right access mode is pre-bound to the claim. should get the exact match.
  1387. pvBadSize.Spec.ClaimRef = nil
  1388. pvBadMode.Spec.ClaimRef = claimRef
  1389. volume, _ = index.findBestMatchForClaim(claim, false)
  1390. if volume.Name != pv1.Name {
  1391. t.Errorf("Expected %s but got volume %s instead", pv1.Name, volume.Name)
  1392. }
  1393. }
  1394. func TestBestMatchDelayed(t *testing.T) {
  1395. volList := newPersistentVolumeOrderedIndex()
  1396. for _, pv := range createTestVolumes() {
  1397. volList.store.Add(pv)
  1398. }
  1399. // binding through PV controller should be delayed
  1400. claim := makePVC("8G", nil)
  1401. volume, err := volList.findBestMatchForClaim(claim, true)
  1402. if err != nil {
  1403. t.Errorf("Unexpected error matching volume by claim: %v", err)
  1404. }
  1405. if volume != nil {
  1406. t.Errorf("Unexpected match with %q", volume.UID)
  1407. }
  1408. }
  1409. func TestFindMatchVolumeWithNode(t *testing.T) {
  1410. volumes := createTestVolumes()
  1411. node1 := &v1.Node{
  1412. ObjectMeta: metav1.ObjectMeta{
  1413. Labels: map[string]string{"key1": "value1"},
  1414. },
  1415. }
  1416. node2 := &v1.Node{
  1417. ObjectMeta: metav1.ObjectMeta{
  1418. Labels: map[string]string{"key1": "value2"},
  1419. },
  1420. }
  1421. node3 := &v1.Node{
  1422. ObjectMeta: metav1.ObjectMeta{
  1423. Labels: map[string]string{"key1": "value3"},
  1424. },
  1425. }
  1426. node4 := &v1.Node{
  1427. ObjectMeta: metav1.ObjectMeta{
  1428. Labels: map[string]string{"key1": "value4"},
  1429. },
  1430. }
  1431. scenarios := map[string]struct {
  1432. expectedMatch string
  1433. claim *v1.PersistentVolumeClaim
  1434. node *v1.Node
  1435. excludedVolumes map[string]*v1.PersistentVolume
  1436. }{
  1437. "success-match": {
  1438. expectedMatch: "affinity-pv",
  1439. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1440. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1441. pvc.Spec.StorageClassName = &classWait
  1442. }),
  1443. node: node1,
  1444. },
  1445. "success-prebound": {
  1446. expectedMatch: "affinity-prebound",
  1447. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1448. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1449. pvc.Spec.StorageClassName = &classWait
  1450. pvc.Name = "claim02"
  1451. }),
  1452. node: node1,
  1453. },
  1454. "success-exclusion": {
  1455. expectedMatch: "affinity-pv2",
  1456. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1457. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1458. pvc.Spec.StorageClassName = &classWait
  1459. }),
  1460. node: node1,
  1461. excludedVolumes: map[string]*v1.PersistentVolume{"affinity001": nil},
  1462. },
  1463. "fail-exclusion": {
  1464. expectedMatch: "",
  1465. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1466. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1467. pvc.Spec.StorageClassName = &classWait
  1468. }),
  1469. node: node1,
  1470. excludedVolumes: map[string]*v1.PersistentVolume{"affinity001": nil, "affinity002": nil},
  1471. },
  1472. "fail-accessmode": {
  1473. expectedMatch: "",
  1474. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1475. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany}
  1476. pvc.Spec.StorageClassName = &classWait
  1477. }),
  1478. node: node1,
  1479. },
  1480. "fail-nodeaffinity": {
  1481. expectedMatch: "",
  1482. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1483. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1484. pvc.Spec.StorageClassName = &classWait
  1485. }),
  1486. node: node2,
  1487. },
  1488. "fail-prebound-node-affinity": {
  1489. expectedMatch: "",
  1490. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1491. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1492. pvc.Spec.StorageClassName = &classWait
  1493. pvc.Name = "claim02"
  1494. }),
  1495. node: node3,
  1496. },
  1497. "fail-nonavaiable": {
  1498. expectedMatch: "",
  1499. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1500. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1501. pvc.Spec.StorageClassName = &classWait
  1502. pvc.Name = "claim04"
  1503. }),
  1504. node: node4,
  1505. },
  1506. "success-bad-and-good-node-affinity": {
  1507. expectedMatch: "affinity-pv3",
  1508. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1509. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce}
  1510. pvc.Spec.StorageClassName = &classWait
  1511. pvc.Name = "claim03"
  1512. }),
  1513. node: node3,
  1514. },
  1515. }
  1516. for name, scenario := range scenarios {
  1517. volume, err := pvutil.FindMatchingVolume(scenario.claim, volumes, scenario.node, scenario.excludedVolumes, true)
  1518. if err != nil {
  1519. t.Errorf("Unexpected error matching volume by claim: %v", err)
  1520. }
  1521. if len(scenario.expectedMatch) != 0 && volume == nil {
  1522. t.Errorf("Expected match but received nil volume for scenario: %s", name)
  1523. }
  1524. if len(scenario.expectedMatch) != 0 && volume != nil && string(volume.UID) != scenario.expectedMatch {
  1525. t.Errorf("Expected %s but got volume %s in scenario %s", scenario.expectedMatch, volume.UID, name)
  1526. }
  1527. if len(scenario.expectedMatch) == 0 && volume != nil {
  1528. t.Errorf("Unexpected match for scenario: %s, matched with %s instead", name, volume.UID)
  1529. }
  1530. }
  1531. }
  1532. func TestCheckAccessModes(t *testing.T) {
  1533. volume := &v1.PersistentVolume{
  1534. Spec: v1.PersistentVolumeSpec{
  1535. AccessModes: []v1.PersistentVolumeAccessMode{v1.ReadWriteOnce, v1.ReadWriteMany},
  1536. },
  1537. }
  1538. scenarios := map[string]struct {
  1539. shouldSucceed bool
  1540. claim *v1.PersistentVolumeClaim
  1541. }{
  1542. "success-single-mode": {
  1543. shouldSucceed: true,
  1544. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1545. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany}
  1546. }),
  1547. },
  1548. "success-many-modes": {
  1549. shouldSucceed: true,
  1550. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1551. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany, v1.ReadWriteOnce}
  1552. }),
  1553. },
  1554. "fail-single-mode": {
  1555. shouldSucceed: false,
  1556. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1557. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadOnlyMany}
  1558. }),
  1559. },
  1560. "fail-many-modes": {
  1561. shouldSucceed: false,
  1562. claim: makePVC("100G", func(pvc *v1.PersistentVolumeClaim) {
  1563. pvc.Spec.AccessModes = []v1.PersistentVolumeAccessMode{v1.ReadWriteMany, v1.ReadOnlyMany}
  1564. }),
  1565. },
  1566. }
  1567. for name, scenario := range scenarios {
  1568. result := pvutil.CheckAccessModes(scenario.claim, volume)
  1569. if result != scenario.shouldSucceed {
  1570. t.Errorf("Test %q failed: Expected %v, got %v", name, scenario.shouldSucceed, result)
  1571. }
  1572. }
  1573. }
  1574. // byCapacity is used to order volumes by ascending storage size
  1575. type byCapacity struct {
  1576. volumes []*v1.PersistentVolume
  1577. }
  1578. func (c byCapacity) Less(i, j int) bool {
  1579. return matchStorageCapacity(c.volumes[i], c.volumes[j])
  1580. }
  1581. func (c byCapacity) Swap(i, j int) {
  1582. c.volumes[i], c.volumes[j] = c.volumes[j], c.volumes[i]
  1583. }
  1584. func (c byCapacity) Len() int {
  1585. return len(c.volumes)
  1586. }
  1587. // matchStorageCapacity is a matchPredicate used to sort and find volumes
  1588. func matchStorageCapacity(pvA, pvB *v1.PersistentVolume) bool {
  1589. aQty := pvA.Spec.Capacity[v1.ResourceStorage]
  1590. bQty := pvB.Spec.Capacity[v1.ResourceStorage]
  1591. return aQty.Cmp(bQty) <= 0
  1592. }