compute_test.go 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937
  1. /*
  2. Copyright 2017 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package upgrade
  14. import (
  15. "fmt"
  16. "reflect"
  17. "strings"
  18. "testing"
  19. "time"
  20. "github.com/pkg/errors"
  21. apps "k8s.io/api/apps/v1"
  22. v1 "k8s.io/api/core/v1"
  23. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  24. versionutil "k8s.io/apimachinery/pkg/util/version"
  25. clientsetfake "k8s.io/client-go/kubernetes/fake"
  26. kubeadmapi "k8s.io/kubernetes/cmd/kubeadm/app/apis/kubeadm"
  27. "k8s.io/kubernetes/cmd/kubeadm/app/constants"
  28. etcdutil "k8s.io/kubernetes/cmd/kubeadm/app/util/etcd"
  29. )
  30. type fakeVersionGetter struct {
  31. clusterVersion, kubeadmVersion, stableVersion, latestVersion, latestDevBranchVersion, stablePatchVersion, kubeletVersion string
  32. }
  33. var _ VersionGetter = &fakeVersionGetter{}
  34. // ClusterVersion gets a fake API server version
  35. func (f *fakeVersionGetter) ClusterVersion() (string, *versionutil.Version, error) {
  36. return f.clusterVersion, versionutil.MustParseSemantic(f.clusterVersion), nil
  37. }
  38. // KubeadmVersion gets a fake kubeadm version
  39. func (f *fakeVersionGetter) KubeadmVersion() (string, *versionutil.Version, error) {
  40. return f.kubeadmVersion, versionutil.MustParseSemantic(f.kubeadmVersion), nil
  41. }
  42. // VersionFromCILabel gets fake latest versions from CI
  43. func (f *fakeVersionGetter) VersionFromCILabel(ciVersionLabel, _ string) (string, *versionutil.Version, error) {
  44. if ciVersionLabel == "stable" {
  45. return f.stableVersion, versionutil.MustParseSemantic(f.stableVersion), nil
  46. }
  47. if ciVersionLabel == "latest" {
  48. return f.latestVersion, versionutil.MustParseSemantic(f.latestVersion), nil
  49. }
  50. if f.latestDevBranchVersion != "" && strings.HasPrefix(ciVersionLabel, "latest-") {
  51. return f.latestDevBranchVersion, versionutil.MustParseSemantic(f.latestDevBranchVersion), nil
  52. }
  53. return f.stablePatchVersion, versionutil.MustParseSemantic(f.stablePatchVersion), nil
  54. }
  55. // KubeletVersions gets the versions of the kubelets in the cluster
  56. func (f *fakeVersionGetter) KubeletVersions() (map[string]uint16, error) {
  57. return map[string]uint16{
  58. f.kubeletVersion: 1,
  59. }, nil
  60. }
  61. const fakeCurrentEtcdVersion = "3.1.12"
  62. type fakeEtcdClient struct {
  63. TLS bool
  64. mismatchedVersions bool
  65. }
  66. func (f fakeEtcdClient) WaitForClusterAvailable(retries int, retryInterval time.Duration) (bool, error) {
  67. return true, nil
  68. }
  69. func (f fakeEtcdClient) CheckClusterHealth() error {
  70. return nil
  71. }
  72. func (f fakeEtcdClient) GetVersion() (string, error) {
  73. versions, _ := f.GetClusterVersions()
  74. if f.mismatchedVersions {
  75. return "", errors.Errorf("etcd cluster contains endpoints with mismatched versions: %v", versions)
  76. }
  77. return fakeCurrentEtcdVersion, nil
  78. }
  79. func (f fakeEtcdClient) GetClusterVersions() (map[string]string, error) {
  80. if f.mismatchedVersions {
  81. return map[string]string{
  82. "foo": fakeCurrentEtcdVersion,
  83. "bar": "3.2.0",
  84. }, nil
  85. }
  86. return map[string]string{
  87. "foo": fakeCurrentEtcdVersion,
  88. "bar": fakeCurrentEtcdVersion,
  89. }, nil
  90. }
  91. func (f fakeEtcdClient) Sync() error { return nil }
  92. func (f fakeEtcdClient) AddMember(name string, peerAddrs string) ([]etcdutil.Member, error) {
  93. return []etcdutil.Member{}, nil
  94. }
  95. func (f fakeEtcdClient) GetMemberID(peerURL string) (uint64, error) {
  96. return 0, nil
  97. }
  98. func (f fakeEtcdClient) RemoveMember(id uint64) ([]etcdutil.Member, error) {
  99. return []etcdutil.Member{}, nil
  100. }
  101. func getEtcdVersion(v *versionutil.Version) string {
  102. return constants.SupportedEtcdVersion[uint8(v.Minor())]
  103. }
  104. const fakeCurrentCoreDNSVersion = "1.0.6"
  105. const fakeCurrentKubeDNSVersion = "1.14.7"
  106. func TestGetAvailableUpgrades(t *testing.T) {
  107. // constansts for test cases
  108. // variables are in the form v{MAJOR}{MINOR}{PATCH}, where MINOR is a variable so test are automatically uptodate to the latest MinimumControlPlaneVersion/
  109. // v1.X series, e.g. v1.14
  110. v1X0 := constants.MinimumControlPlaneVersion.WithMinor(constants.MinimumControlPlaneVersion.Minor() - 1)
  111. v1X5 := v1X0.WithPatch(5)
  112. // v1.Y series, where Y = X+1, e.g. v1.15
  113. v1Y0 := constants.MinimumControlPlaneVersion
  114. v1Y0alpha0 := v1Y0.WithPreRelease("alpha.0")
  115. v1Y0alpha1 := v1Y0.WithPreRelease("alpha.1")
  116. v1Y1 := v1Y0.WithPatch(1)
  117. v1Y2 := v1Y0.WithPatch(2)
  118. v1Y3 := v1Y0.WithPatch(3)
  119. v1Y5 := v1Y0.WithPatch(5)
  120. // v1.Z series, where Z = Y+1, e.g. v1.16
  121. v1Z0 := constants.CurrentKubernetesVersion
  122. v1Z0alpha1 := v1Z0.WithPreRelease("alpha.1")
  123. v1Z0alpha2 := v1Z0.WithPreRelease("alpha.2")
  124. v1Z0beta1 := v1Z0.WithPreRelease("beta.1")
  125. v1Z0rc1 := v1Z0.WithPreRelease("rc.1")
  126. v1Z1 := v1Z0.WithPatch(1)
  127. etcdClient := fakeEtcdClient{}
  128. tests := []struct {
  129. name string
  130. vg VersionGetter
  131. expectedUpgrades []Upgrade
  132. allowExperimental, allowRCs bool
  133. errExpected bool
  134. etcdClient etcdutil.ClusterInterrogator
  135. beforeDNSType kubeadmapi.DNSAddOnType
  136. beforeDNSVersion string
  137. dnsType kubeadmapi.DNSAddOnType
  138. }{
  139. {
  140. name: "no action needed, already up-to-date",
  141. vg: &fakeVersionGetter{
  142. clusterVersion: v1Y0.String(),
  143. kubeletVersion: v1Y0.String(),
  144. kubeadmVersion: v1Y0.String(),
  145. stablePatchVersion: v1Y0.String(),
  146. stableVersion: v1Y0.String(),
  147. },
  148. beforeDNSType: kubeadmapi.CoreDNS,
  149. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  150. dnsType: kubeadmapi.CoreDNS,
  151. expectedUpgrades: []Upgrade{},
  152. allowExperimental: false,
  153. errExpected: false,
  154. etcdClient: etcdClient,
  155. },
  156. {
  157. name: "simple patch version upgrade",
  158. vg: &fakeVersionGetter{
  159. clusterVersion: v1Y1.String(),
  160. kubeletVersion: v1Y1.String(), // the kubelet are on the same version as the control plane
  161. kubeadmVersion: v1Y2.String(),
  162. stablePatchVersion: v1Y3.String(),
  163. stableVersion: v1Y3.String(),
  164. },
  165. beforeDNSType: kubeadmapi.CoreDNS,
  166. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  167. dnsType: kubeadmapi.CoreDNS,
  168. expectedUpgrades: []Upgrade{
  169. {
  170. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  171. Before: ClusterState{
  172. KubeVersion: v1Y1.String(),
  173. KubeletVersions: map[string]uint16{
  174. v1Y1.String(): 1,
  175. },
  176. KubeadmVersion: v1Y2.String(),
  177. DNSType: kubeadmapi.CoreDNS,
  178. DNSVersion: fakeCurrentCoreDNSVersion,
  179. EtcdVersion: fakeCurrentEtcdVersion,
  180. },
  181. After: ClusterState{
  182. KubeVersion: v1Y3.String(),
  183. KubeadmVersion: v1Y3.String(),
  184. DNSType: kubeadmapi.CoreDNS,
  185. DNSVersion: constants.CoreDNSVersion,
  186. EtcdVersion: getEtcdVersion(v1Y3),
  187. },
  188. },
  189. },
  190. allowExperimental: false,
  191. errExpected: false,
  192. etcdClient: etcdClient,
  193. },
  194. {
  195. name: "no version provided to offline version getter does not change behavior",
  196. vg: NewOfflineVersionGetter(&fakeVersionGetter{
  197. clusterVersion: v1Y1.String(),
  198. kubeletVersion: v1Y1.String(), // the kubelet are on the same version as the control plane
  199. kubeadmVersion: v1Y2.String(),
  200. stablePatchVersion: v1Y3.String(),
  201. stableVersion: v1Y3.String(),
  202. }, ""),
  203. beforeDNSType: kubeadmapi.CoreDNS,
  204. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  205. dnsType: kubeadmapi.CoreDNS,
  206. expectedUpgrades: []Upgrade{
  207. {
  208. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  209. Before: ClusterState{
  210. KubeVersion: v1Y1.String(),
  211. KubeletVersions: map[string]uint16{
  212. v1Y1.String(): 1,
  213. },
  214. KubeadmVersion: v1Y2.String(),
  215. DNSType: kubeadmapi.CoreDNS,
  216. DNSVersion: fakeCurrentCoreDNSVersion,
  217. EtcdVersion: fakeCurrentEtcdVersion,
  218. },
  219. After: ClusterState{
  220. KubeVersion: v1Y3.String(),
  221. KubeadmVersion: v1Y3.String(),
  222. DNSType: kubeadmapi.CoreDNS,
  223. DNSVersion: constants.CoreDNSVersion,
  224. EtcdVersion: getEtcdVersion(v1Y3),
  225. },
  226. },
  227. },
  228. allowExperimental: false,
  229. errExpected: false,
  230. etcdClient: etcdClient,
  231. },
  232. {
  233. name: "minor version upgrade only",
  234. vg: &fakeVersionGetter{
  235. clusterVersion: v1Y1.String(),
  236. kubeletVersion: v1Y1.String(), // the kubelet are on the same version as the control plane
  237. kubeadmVersion: v1Z0.String(),
  238. stablePatchVersion: v1Y1.String(),
  239. stableVersion: v1Z0.String(),
  240. },
  241. beforeDNSType: kubeadmapi.CoreDNS,
  242. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  243. dnsType: kubeadmapi.CoreDNS,
  244. expectedUpgrades: []Upgrade{
  245. {
  246. Description: "stable version",
  247. Before: ClusterState{
  248. KubeVersion: v1Y1.String(),
  249. KubeletVersions: map[string]uint16{
  250. v1Y1.String(): 1,
  251. },
  252. KubeadmVersion: v1Z0.String(),
  253. DNSType: kubeadmapi.CoreDNS,
  254. DNSVersion: fakeCurrentCoreDNSVersion,
  255. EtcdVersion: fakeCurrentEtcdVersion,
  256. },
  257. After: ClusterState{
  258. KubeVersion: v1Z0.String(),
  259. KubeadmVersion: v1Z0.String(),
  260. DNSType: kubeadmapi.CoreDNS,
  261. DNSVersion: constants.CoreDNSVersion,
  262. EtcdVersion: getEtcdVersion(v1Z0),
  263. },
  264. },
  265. },
  266. allowExperimental: false,
  267. errExpected: false,
  268. etcdClient: etcdClient,
  269. },
  270. {
  271. name: "both minor version upgrade and patch version upgrade available",
  272. vg: &fakeVersionGetter{
  273. clusterVersion: v1Y3.String(),
  274. kubeletVersion: v1Y3.String(), // the kubelet are on the same version as the control plane
  275. kubeadmVersion: v1Y5.String(),
  276. stablePatchVersion: v1Y5.String(),
  277. stableVersion: v1Z1.String(),
  278. },
  279. beforeDNSType: kubeadmapi.CoreDNS,
  280. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  281. dnsType: kubeadmapi.CoreDNS,
  282. expectedUpgrades: []Upgrade{
  283. {
  284. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  285. Before: ClusterState{
  286. KubeVersion: v1Y3.String(),
  287. KubeletVersions: map[string]uint16{
  288. v1Y3.String(): 1,
  289. },
  290. KubeadmVersion: v1Y5.String(),
  291. DNSType: kubeadmapi.CoreDNS,
  292. DNSVersion: fakeCurrentCoreDNSVersion,
  293. EtcdVersion: fakeCurrentEtcdVersion,
  294. },
  295. After: ClusterState{
  296. KubeVersion: v1Y5.String(),
  297. KubeadmVersion: v1Y5.String(), // Note: The kubeadm version mustn't be "downgraded" here
  298. DNSType: kubeadmapi.CoreDNS,
  299. DNSVersion: constants.CoreDNSVersion,
  300. EtcdVersion: getEtcdVersion(v1Y5),
  301. },
  302. },
  303. {
  304. Description: "stable version",
  305. Before: ClusterState{
  306. KubeVersion: v1Y3.String(),
  307. KubeletVersions: map[string]uint16{
  308. v1Y3.String(): 1,
  309. },
  310. KubeadmVersion: v1Y5.String(),
  311. DNSType: kubeadmapi.CoreDNS,
  312. DNSVersion: fakeCurrentCoreDNSVersion,
  313. EtcdVersion: fakeCurrentEtcdVersion,
  314. },
  315. After: ClusterState{
  316. KubeVersion: v1Z1.String(),
  317. KubeadmVersion: v1Z1.String(),
  318. DNSType: kubeadmapi.CoreDNS,
  319. DNSVersion: constants.CoreDNSVersion,
  320. EtcdVersion: getEtcdVersion(v1Z1),
  321. },
  322. },
  323. },
  324. allowExperimental: false,
  325. errExpected: false,
  326. etcdClient: etcdClient,
  327. },
  328. {
  329. name: "allow experimental upgrades, but no upgrade available",
  330. vg: &fakeVersionGetter{
  331. clusterVersion: v1Z0alpha2.String(),
  332. kubeletVersion: v1Y5.String(),
  333. kubeadmVersion: v1Y5.String(),
  334. stablePatchVersion: v1Y5.String(),
  335. stableVersion: v1Y5.String(),
  336. latestVersion: v1Z0alpha2.String(),
  337. },
  338. beforeDNSType: kubeadmapi.CoreDNS,
  339. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  340. dnsType: kubeadmapi.CoreDNS,
  341. expectedUpgrades: []Upgrade{},
  342. allowExperimental: true,
  343. errExpected: false,
  344. etcdClient: etcdClient,
  345. },
  346. {
  347. name: "upgrade to an unstable version should be supported",
  348. vg: &fakeVersionGetter{
  349. clusterVersion: v1Y5.String(),
  350. kubeletVersion: v1Y5.String(),
  351. kubeadmVersion: v1Y5.String(),
  352. stablePatchVersion: v1Y5.String(),
  353. stableVersion: v1Y5.String(),
  354. latestVersion: v1Z0alpha2.String(),
  355. },
  356. beforeDNSType: kubeadmapi.CoreDNS,
  357. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  358. dnsType: kubeadmapi.CoreDNS,
  359. expectedUpgrades: []Upgrade{
  360. {
  361. Description: "experimental version",
  362. Before: ClusterState{
  363. KubeVersion: v1Y5.String(),
  364. KubeletVersions: map[string]uint16{
  365. v1Y5.String(): 1,
  366. },
  367. KubeadmVersion: v1Y5.String(),
  368. DNSType: kubeadmapi.CoreDNS,
  369. DNSVersion: fakeCurrentCoreDNSVersion,
  370. EtcdVersion: fakeCurrentEtcdVersion,
  371. },
  372. After: ClusterState{
  373. KubeVersion: v1Z0alpha2.String(),
  374. KubeadmVersion: v1Z0alpha2.String(),
  375. DNSType: kubeadmapi.CoreDNS,
  376. DNSVersion: constants.CoreDNSVersion,
  377. EtcdVersion: getEtcdVersion(v1Z0alpha2),
  378. },
  379. },
  380. },
  381. allowExperimental: true,
  382. errExpected: false,
  383. etcdClient: etcdClient,
  384. },
  385. {
  386. name: "upgrade from an unstable version to an unstable version should be supported",
  387. vg: &fakeVersionGetter{
  388. clusterVersion: v1Z0alpha1.String(),
  389. kubeletVersion: v1Y5.String(),
  390. kubeadmVersion: v1Y5.String(),
  391. stablePatchVersion: v1Y5.String(),
  392. stableVersion: v1Y5.String(),
  393. latestVersion: v1Z0alpha2.String(),
  394. },
  395. beforeDNSType: kubeadmapi.CoreDNS,
  396. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  397. dnsType: kubeadmapi.CoreDNS,
  398. expectedUpgrades: []Upgrade{
  399. {
  400. Description: "experimental version",
  401. Before: ClusterState{
  402. KubeVersion: v1Z0alpha1.String(),
  403. KubeletVersions: map[string]uint16{
  404. v1Y5.String(): 1,
  405. },
  406. KubeadmVersion: v1Y5.String(),
  407. DNSType: kubeadmapi.CoreDNS,
  408. DNSVersion: fakeCurrentCoreDNSVersion,
  409. EtcdVersion: fakeCurrentEtcdVersion,
  410. },
  411. After: ClusterState{
  412. KubeVersion: v1Z0alpha2.String(),
  413. KubeadmVersion: v1Z0alpha2.String(),
  414. DNSType: kubeadmapi.CoreDNS,
  415. DNSVersion: constants.CoreDNSVersion,
  416. EtcdVersion: getEtcdVersion(v1Z0alpha2),
  417. },
  418. },
  419. },
  420. allowExperimental: true,
  421. errExpected: false,
  422. etcdClient: etcdClient,
  423. },
  424. {
  425. name: "v1.X.0-alpha.0 should be ignored",
  426. vg: &fakeVersionGetter{
  427. clusterVersion: v1X5.String(),
  428. kubeletVersion: v1X5.String(),
  429. kubeadmVersion: v1X5.String(),
  430. stablePatchVersion: v1X5.String(),
  431. stableVersion: v1X5.String(),
  432. latestDevBranchVersion: v1Z0beta1.String(),
  433. latestVersion: v1Y0alpha0.String(),
  434. },
  435. beforeDNSType: kubeadmapi.CoreDNS,
  436. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  437. dnsType: kubeadmapi.CoreDNS,
  438. expectedUpgrades: []Upgrade{
  439. {
  440. Description: "experimental version",
  441. Before: ClusterState{
  442. KubeVersion: v1X5.String(),
  443. KubeletVersions: map[string]uint16{
  444. v1X5.String(): 1,
  445. },
  446. KubeadmVersion: v1X5.String(),
  447. DNSType: kubeadmapi.CoreDNS,
  448. DNSVersion: fakeCurrentCoreDNSVersion,
  449. EtcdVersion: fakeCurrentEtcdVersion,
  450. },
  451. After: ClusterState{
  452. KubeVersion: v1Z0beta1.String(),
  453. KubeadmVersion: v1Z0beta1.String(),
  454. DNSType: kubeadmapi.CoreDNS,
  455. DNSVersion: constants.CoreDNSVersion,
  456. EtcdVersion: getEtcdVersion(v1Z0beta1),
  457. },
  458. },
  459. },
  460. allowExperimental: true,
  461. errExpected: false,
  462. etcdClient: etcdClient,
  463. },
  464. {
  465. name: "upgrade to an RC version should be supported",
  466. vg: &fakeVersionGetter{
  467. clusterVersion: v1X5.String(),
  468. kubeletVersion: v1X5.String(),
  469. kubeadmVersion: v1X5.String(),
  470. stablePatchVersion: v1X5.String(),
  471. stableVersion: v1X5.String(),
  472. latestDevBranchVersion: v1Z0rc1.String(),
  473. latestVersion: v1Y0alpha1.String(),
  474. },
  475. beforeDNSType: kubeadmapi.CoreDNS,
  476. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  477. dnsType: kubeadmapi.CoreDNS,
  478. expectedUpgrades: []Upgrade{
  479. {
  480. Description: "release candidate version",
  481. Before: ClusterState{
  482. KubeVersion: v1X5.String(),
  483. KubeletVersions: map[string]uint16{
  484. v1X5.String(): 1,
  485. },
  486. KubeadmVersion: v1X5.String(),
  487. DNSType: kubeadmapi.CoreDNS,
  488. DNSVersion: fakeCurrentCoreDNSVersion,
  489. EtcdVersion: fakeCurrentEtcdVersion,
  490. },
  491. After: ClusterState{
  492. KubeVersion: v1Z0rc1.String(),
  493. KubeadmVersion: v1Z0rc1.String(),
  494. DNSType: kubeadmapi.CoreDNS,
  495. DNSVersion: constants.CoreDNSVersion,
  496. EtcdVersion: getEtcdVersion(v1Z0rc1),
  497. },
  498. },
  499. },
  500. allowRCs: true,
  501. errExpected: false,
  502. etcdClient: etcdClient,
  503. },
  504. {
  505. name: "it is possible (but very uncommon) that the latest version from the previous branch is an rc and the current latest version is alpha.0. In that case, show the RC",
  506. vg: &fakeVersionGetter{
  507. clusterVersion: v1X5.String(),
  508. kubeletVersion: v1X5.String(),
  509. kubeadmVersion: v1X5.String(),
  510. stablePatchVersion: v1X5.String(),
  511. stableVersion: v1X5.String(),
  512. latestDevBranchVersion: v1Z0rc1.String(),
  513. latestVersion: v1Y0alpha0.String(),
  514. },
  515. beforeDNSType: kubeadmapi.CoreDNS,
  516. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  517. dnsType: kubeadmapi.CoreDNS,
  518. expectedUpgrades: []Upgrade{
  519. {
  520. Description: "experimental version", // Note that this is considered an experimental version in this uncommon scenario
  521. Before: ClusterState{
  522. KubeVersion: v1X5.String(),
  523. KubeletVersions: map[string]uint16{
  524. v1X5.String(): 1,
  525. },
  526. KubeadmVersion: v1X5.String(),
  527. DNSType: kubeadmapi.CoreDNS,
  528. DNSVersion: fakeCurrentCoreDNSVersion,
  529. EtcdVersion: fakeCurrentEtcdVersion,
  530. },
  531. After: ClusterState{
  532. KubeVersion: v1Z0rc1.String(),
  533. KubeadmVersion: v1Z0rc1.String(),
  534. DNSType: kubeadmapi.CoreDNS,
  535. DNSVersion: constants.CoreDNSVersion,
  536. EtcdVersion: getEtcdVersion(v1Z0rc1),
  537. },
  538. },
  539. },
  540. allowExperimental: true,
  541. errExpected: false,
  542. etcdClient: etcdClient,
  543. },
  544. {
  545. name: "upgrade to an RC version should be supported. There may also be an even newer unstable version.",
  546. vg: &fakeVersionGetter{
  547. clusterVersion: v1X5.String(),
  548. kubeletVersion: v1X5.String(),
  549. kubeadmVersion: v1X5.String(),
  550. stablePatchVersion: v1X5.String(),
  551. stableVersion: v1X5.String(),
  552. latestDevBranchVersion: v1Z0rc1.String(),
  553. latestVersion: v1Y0alpha1.String(),
  554. },
  555. beforeDNSType: kubeadmapi.CoreDNS,
  556. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  557. dnsType: kubeadmapi.CoreDNS,
  558. expectedUpgrades: []Upgrade{
  559. {
  560. Description: "release candidate version",
  561. Before: ClusterState{
  562. KubeVersion: v1X5.String(),
  563. KubeletVersions: map[string]uint16{
  564. v1X5.String(): 1,
  565. },
  566. KubeadmVersion: v1X5.String(),
  567. DNSType: kubeadmapi.CoreDNS,
  568. DNSVersion: fakeCurrentCoreDNSVersion,
  569. EtcdVersion: fakeCurrentEtcdVersion,
  570. },
  571. After: ClusterState{
  572. KubeVersion: v1Z0rc1.String(),
  573. KubeadmVersion: v1Z0rc1.String(),
  574. DNSType: kubeadmapi.CoreDNS,
  575. DNSVersion: constants.CoreDNSVersion,
  576. EtcdVersion: getEtcdVersion(v1Z0rc1),
  577. },
  578. },
  579. {
  580. Description: "experimental version",
  581. Before: ClusterState{
  582. KubeVersion: v1X5.String(),
  583. KubeletVersions: map[string]uint16{
  584. v1X5.String(): 1,
  585. },
  586. KubeadmVersion: v1X5.String(),
  587. DNSType: kubeadmapi.CoreDNS,
  588. DNSVersion: fakeCurrentCoreDNSVersion,
  589. EtcdVersion: fakeCurrentEtcdVersion,
  590. },
  591. After: ClusterState{
  592. KubeVersion: v1Y0alpha1.String(),
  593. KubeadmVersion: v1Y0alpha1.String(),
  594. DNSType: kubeadmapi.CoreDNS,
  595. DNSVersion: constants.CoreDNSVersion,
  596. EtcdVersion: getEtcdVersion(v1Y0alpha1),
  597. },
  598. },
  599. },
  600. allowRCs: true,
  601. allowExperimental: true,
  602. errExpected: false,
  603. etcdClient: etcdClient,
  604. },
  605. {
  606. name: "Upgrades with external etcd with mismatched versions should not be allowed.",
  607. vg: &fakeVersionGetter{
  608. clusterVersion: v1Y3.String(),
  609. kubeletVersion: v1Y3.String(),
  610. kubeadmVersion: v1Y3.String(),
  611. stablePatchVersion: v1Y3.String(),
  612. stableVersion: v1Y3.String(),
  613. },
  614. allowRCs: false,
  615. allowExperimental: false,
  616. etcdClient: fakeEtcdClient{mismatchedVersions: true},
  617. expectedUpgrades: []Upgrade{},
  618. errExpected: true,
  619. },
  620. {
  621. name: "offline version getter",
  622. vg: NewOfflineVersionGetter(&fakeVersionGetter{
  623. clusterVersion: v1Y1.String(),
  624. kubeletVersion: v1Y0.String(),
  625. kubeadmVersion: v1Y1.String(),
  626. }, v1Z1.String()),
  627. etcdClient: etcdClient,
  628. beforeDNSType: kubeadmapi.CoreDNS,
  629. beforeDNSVersion: fakeCurrentCoreDNSVersion,
  630. dnsType: kubeadmapi.CoreDNS,
  631. expectedUpgrades: []Upgrade{
  632. {
  633. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  634. Before: ClusterState{
  635. KubeVersion: v1Y1.String(),
  636. KubeletVersions: map[string]uint16{
  637. v1Y0.String(): 1,
  638. },
  639. KubeadmVersion: v1Y1.String(),
  640. DNSType: kubeadmapi.CoreDNS,
  641. DNSVersion: fakeCurrentCoreDNSVersion,
  642. EtcdVersion: fakeCurrentEtcdVersion,
  643. },
  644. After: ClusterState{
  645. KubeVersion: v1Z1.String(),
  646. KubeadmVersion: v1Z1.String(),
  647. DNSType: kubeadmapi.CoreDNS,
  648. DNSVersion: constants.CoreDNSVersion,
  649. EtcdVersion: getEtcdVersion(v1Z1),
  650. },
  651. },
  652. },
  653. },
  654. {
  655. name: "kubedns to coredns",
  656. vg: &fakeVersionGetter{
  657. clusterVersion: v1Y2.String(),
  658. kubeletVersion: v1Y2.String(), // the kubelet are on the same version as the control plane
  659. kubeadmVersion: v1Z0.String(),
  660. stablePatchVersion: v1Z0.String(),
  661. stableVersion: v1Z0.String(),
  662. },
  663. etcdClient: etcdClient,
  664. beforeDNSType: kubeadmapi.KubeDNS,
  665. beforeDNSVersion: fakeCurrentKubeDNSVersion,
  666. dnsType: kubeadmapi.CoreDNS,
  667. expectedUpgrades: []Upgrade{
  668. {
  669. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  670. Before: ClusterState{
  671. KubeVersion: v1Y2.String(),
  672. KubeletVersions: map[string]uint16{
  673. v1Y2.String(): 1,
  674. },
  675. KubeadmVersion: v1Z0.String(),
  676. DNSType: kubeadmapi.KubeDNS,
  677. DNSVersion: fakeCurrentKubeDNSVersion,
  678. EtcdVersion: fakeCurrentEtcdVersion,
  679. },
  680. After: ClusterState{
  681. KubeVersion: v1Z0.String(),
  682. KubeadmVersion: v1Z0.String(),
  683. DNSType: kubeadmapi.CoreDNS,
  684. DNSVersion: constants.CoreDNSVersion,
  685. EtcdVersion: getEtcdVersion(v1Z0),
  686. },
  687. },
  688. },
  689. },
  690. {
  691. name: "keep coredns",
  692. vg: &fakeVersionGetter{
  693. clusterVersion: v1Y2.String(),
  694. kubeletVersion: v1Y2.String(), // the kubelet are on the same version as the control plane
  695. kubeadmVersion: v1Z0.String(),
  696. stablePatchVersion: v1Z0.String(),
  697. stableVersion: v1Z0.String(),
  698. },
  699. etcdClient: etcdClient,
  700. beforeDNSType: kubeadmapi.KubeDNS,
  701. beforeDNSVersion: fakeCurrentKubeDNSVersion,
  702. dnsType: kubeadmapi.KubeDNS,
  703. expectedUpgrades: []Upgrade{
  704. {
  705. Description: fmt.Sprintf("version in the v%d.%d series", v1Y0.Major(), v1Y0.Minor()),
  706. Before: ClusterState{
  707. KubeVersion: v1Y2.String(),
  708. KubeletVersions: map[string]uint16{
  709. v1Y2.String(): 1,
  710. },
  711. KubeadmVersion: v1Z0.String(),
  712. DNSType: kubeadmapi.KubeDNS,
  713. DNSVersion: fakeCurrentKubeDNSVersion,
  714. EtcdVersion: fakeCurrentEtcdVersion,
  715. },
  716. After: ClusterState{
  717. KubeVersion: v1Z0.String(),
  718. KubeadmVersion: v1Z0.String(),
  719. DNSType: kubeadmapi.KubeDNS,
  720. DNSVersion: constants.KubeDNSVersion,
  721. EtcdVersion: getEtcdVersion(v1Z0),
  722. },
  723. },
  724. },
  725. },
  726. }
  727. // Instantiating a fake etcd cluster for being able to get etcd version for a corresponding
  728. // Kubernetes release.
  729. for _, rt := range tests {
  730. t.Run(rt.name, func(t *testing.T) {
  731. dnsName := constants.CoreDNSDeploymentName
  732. if rt.beforeDNSType == kubeadmapi.KubeDNS {
  733. dnsName = constants.KubeDNSDeploymentName
  734. }
  735. client := clientsetfake.NewSimpleClientset(&apps.Deployment{
  736. TypeMeta: metav1.TypeMeta{
  737. Kind: "Deployment",
  738. APIVersion: "apps/v1",
  739. },
  740. ObjectMeta: metav1.ObjectMeta{
  741. Name: dnsName,
  742. Namespace: "kube-system",
  743. Labels: map[string]string{
  744. "k8s-app": "kube-dns",
  745. },
  746. },
  747. Spec: apps.DeploymentSpec{
  748. Template: v1.PodTemplateSpec{
  749. Spec: v1.PodSpec{
  750. Containers: []v1.Container{
  751. {
  752. Image: "test:" + rt.beforeDNSVersion,
  753. },
  754. },
  755. },
  756. },
  757. },
  758. })
  759. actualUpgrades, actualErr := GetAvailableUpgrades(rt.vg, rt.allowExperimental, rt.allowRCs, rt.etcdClient, rt.dnsType, client)
  760. if !reflect.DeepEqual(actualUpgrades, rt.expectedUpgrades) {
  761. t.Errorf("failed TestGetAvailableUpgrades\n\texpected upgrades: %v\n\tgot: %v", rt.expectedUpgrades, actualUpgrades)
  762. }
  763. if (actualErr != nil) != rt.errExpected {
  764. fmt.Printf("Hello error")
  765. t.Errorf("failed TestGetAvailableUpgrades\n\texpected error: %t\n\tgot error: %t", rt.errExpected, (actualErr != nil))
  766. }
  767. if !reflect.DeepEqual(actualUpgrades, rt.expectedUpgrades) {
  768. t.Errorf("failed TestGetAvailableUpgrades\n\texpected upgrades: %v\n\tgot: %v", rt.expectedUpgrades, actualUpgrades)
  769. }
  770. })
  771. }
  772. }
  773. func TestKubeletUpgrade(t *testing.T) {
  774. tests := []struct {
  775. name string
  776. before map[string]uint16
  777. after string
  778. expected bool
  779. }{
  780. {
  781. name: "upgrade from v1.10.1 to v1.10.3 is available",
  782. before: map[string]uint16{
  783. "v1.10.1": 1,
  784. },
  785. after: "v1.10.3",
  786. expected: true,
  787. },
  788. {
  789. name: "upgrade from v1.10.1 and v1.10.3/100 to v1.10.3 is available",
  790. before: map[string]uint16{
  791. "v1.10.1": 1,
  792. "v1.10.3": 100,
  793. },
  794. after: "v1.10.3",
  795. expected: true,
  796. },
  797. {
  798. name: "upgrade from v1.10.3 to v1.10.3 is not available",
  799. before: map[string]uint16{
  800. "v1.10.3": 1,
  801. },
  802. after: "v1.10.3",
  803. expected: false,
  804. },
  805. {
  806. name: "upgrade from v1.10.3/100 to v1.10.3 is not available",
  807. before: map[string]uint16{
  808. "v1.10.3": 100,
  809. },
  810. after: "v1.10.3",
  811. expected: false,
  812. },
  813. {
  814. name: "upgrade is not available if we don't know anything about the earlier state",
  815. before: map[string]uint16{},
  816. after: "v1.10.3",
  817. expected: false,
  818. },
  819. }
  820. for _, rt := range tests {
  821. t.Run(rt.name, func(t *testing.T) {
  822. upgrade := Upgrade{
  823. Before: ClusterState{
  824. KubeletVersions: rt.before,
  825. },
  826. After: ClusterState{
  827. KubeVersion: rt.after,
  828. },
  829. }
  830. actual := upgrade.CanUpgradeKubelets()
  831. if actual != rt.expected {
  832. t.Errorf("failed TestKubeletUpgrade\n\texpected: %t\n\tgot: %t\n\ttest object: %v", rt.expected, actual, upgrade)
  833. }
  834. })
  835. }
  836. }
  837. func TestGetBranchFromVersion(t *testing.T) {
  838. testCases := []struct {
  839. version string
  840. expectedVersion string
  841. }{
  842. {
  843. version: "v1.9.5",
  844. expectedVersion: "1.9",
  845. },
  846. {
  847. version: "v1.9.0-alpha.2",
  848. expectedVersion: "1.9",
  849. },
  850. {
  851. version: "v1.9.0-beta.0",
  852. expectedVersion: "1.9",
  853. },
  854. {
  855. version: "v1.9.0-rc.1",
  856. expectedVersion: "1.9",
  857. },
  858. {
  859. version: "v1.11.0-alpha.0",
  860. expectedVersion: "1.11",
  861. },
  862. {
  863. version: "v1.11.0-beta.1",
  864. expectedVersion: "1.11",
  865. },
  866. {
  867. version: "v1.11.0-rc.0",
  868. expectedVersion: "1.11",
  869. },
  870. {
  871. version: "1.12.5",
  872. expectedVersion: "1.12",
  873. },
  874. }
  875. for _, tc := range testCases {
  876. t.Run(tc.version, func(t *testing.T) {
  877. v := getBranchFromVersion(tc.version)
  878. if v != tc.expectedVersion {
  879. t.Errorf("expected version %s, got %s", tc.expectedVersion, v)
  880. }
  881. })
  882. }
  883. }