versiongetter.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  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. "io"
  17. "github.com/pkg/errors"
  18. "k8s.io/api/core/v1"
  19. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  20. versionutil "k8s.io/apimachinery/pkg/util/version"
  21. clientset "k8s.io/client-go/kubernetes"
  22. kubeadmutil "k8s.io/kubernetes/cmd/kubeadm/app/util"
  23. "k8s.io/kubernetes/pkg/version"
  24. )
  25. // VersionGetter defines an interface for fetching different versions.
  26. // Easy to implement a fake variant of this interface for unit testing
  27. type VersionGetter interface {
  28. // ClusterVersion should return the version of the cluster i.e. the API Server version
  29. ClusterVersion() (string, *versionutil.Version, error)
  30. // KubeadmVersion should return the version of the kubeadm CLI
  31. KubeadmVersion() (string, *versionutil.Version, error)
  32. // VersionFromCILabel should resolve CI labels like `latest`, `stable`, `stable-1.8`, etc. to real versions
  33. VersionFromCILabel(string, string) (string, *versionutil.Version, error)
  34. // KubeletVersions should return a map with a version and a number that describes how many kubelets there are for that version
  35. KubeletVersions() (map[string]uint16, error)
  36. }
  37. // KubeVersionGetter handles the version-fetching mechanism from external sources
  38. type KubeVersionGetter struct {
  39. client clientset.Interface
  40. w io.Writer
  41. }
  42. // NewKubeVersionGetter returns a new instance of KubeVersionGetter
  43. func NewKubeVersionGetter(client clientset.Interface, writer io.Writer) VersionGetter {
  44. return &KubeVersionGetter{
  45. client: client,
  46. w: writer,
  47. }
  48. }
  49. // ClusterVersion gets API server version
  50. func (g *KubeVersionGetter) ClusterVersion() (string, *versionutil.Version, error) {
  51. clusterVersionInfo, err := g.client.Discovery().ServerVersion()
  52. if err != nil {
  53. return "", nil, errors.Wrap(err, "Couldn't fetch cluster version from the API Server")
  54. }
  55. fmt.Fprintf(g.w, "[upgrade/versions] Cluster version: %s\n", clusterVersionInfo.String())
  56. clusterVersion, err := versionutil.ParseSemantic(clusterVersionInfo.String())
  57. if err != nil {
  58. return "", nil, errors.Wrap(err, "Couldn't parse cluster version")
  59. }
  60. return clusterVersionInfo.String(), clusterVersion, nil
  61. }
  62. // KubeadmVersion gets kubeadm version
  63. func (g *KubeVersionGetter) KubeadmVersion() (string, *versionutil.Version, error) {
  64. kubeadmVersionInfo := version.Get()
  65. fmt.Fprintf(g.w, "[upgrade/versions] kubeadm version: %s\n", kubeadmVersionInfo.String())
  66. kubeadmVersion, err := versionutil.ParseSemantic(kubeadmVersionInfo.String())
  67. if err != nil {
  68. return "", nil, errors.Wrap(err, "Couldn't parse kubeadm version")
  69. }
  70. return kubeadmVersionInfo.String(), kubeadmVersion, nil
  71. }
  72. // VersionFromCILabel resolves a version label like "latest" or "stable" to an actual version using the public Kubernetes CI uploads
  73. func (g *KubeVersionGetter) VersionFromCILabel(ciVersionLabel, description string) (string, *versionutil.Version, error) {
  74. versionStr, err := kubeadmutil.KubernetesReleaseVersion(ciVersionLabel)
  75. if err != nil {
  76. return "", nil, errors.Wrapf(err, "Couldn't fetch latest %s from the internet", description)
  77. }
  78. if description != "" {
  79. fmt.Fprintf(g.w, "[upgrade/versions] Latest %s: %s\n", description, versionStr)
  80. }
  81. ver, err := versionutil.ParseSemantic(versionStr)
  82. if err != nil {
  83. return "", nil, errors.Wrapf(err, "Couldn't parse latest %s", description)
  84. }
  85. return versionStr, ver, nil
  86. }
  87. // KubeletVersions gets the versions of the kubelets in the cluster
  88. func (g *KubeVersionGetter) KubeletVersions() (map[string]uint16, error) {
  89. nodes, err := g.client.CoreV1().Nodes().List(metav1.ListOptions{})
  90. if err != nil {
  91. return nil, errors.New("couldn't list all nodes in cluster")
  92. }
  93. return computeKubeletVersions(nodes.Items), nil
  94. }
  95. // computeKubeletVersions returns a string-int map that describes how many nodes are of a specific version
  96. func computeKubeletVersions(nodes []v1.Node) map[string]uint16 {
  97. kubeletVersions := map[string]uint16{}
  98. for _, node := range nodes {
  99. kver := node.Status.NodeInfo.KubeletVersion
  100. if _, found := kubeletVersions[kver]; !found {
  101. kubeletVersions[kver] = 1
  102. continue
  103. }
  104. kubeletVersions[kver]++
  105. }
  106. return kubeletVersions
  107. }
  108. // OfflineVersionGetter will use the version provided or
  109. type OfflineVersionGetter struct {
  110. VersionGetter
  111. version string
  112. }
  113. // NewOfflineVersionGetter wraps a VersionGetter and skips online communication if default information is supplied.
  114. // Version can be "" and the behavior will be identical to the versionGetter passed in.
  115. func NewOfflineVersionGetter(versionGetter VersionGetter, version string) VersionGetter {
  116. return &OfflineVersionGetter{
  117. VersionGetter: versionGetter,
  118. version: version,
  119. }
  120. }
  121. // VersionFromCILabel will return the version that was passed into the struct
  122. func (o *OfflineVersionGetter) VersionFromCILabel(ciVersionLabel, description string) (string, *versionutil.Version, error) {
  123. if o.version == "" {
  124. return o.VersionGetter.VersionFromCILabel(ciVersionLabel, description)
  125. }
  126. ver, err := versionutil.ParseSemantic(o.version)
  127. if err != nil {
  128. return "", nil, errors.Wrapf(err, "Couldn't parse version %s", description)
  129. }
  130. return o.version, ver, nil
  131. }