2016-11-21 14:56:10 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 The Kubernetes Authors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
2018-08-14 14:07:26 +00:00
|
|
|
"errors"
|
2016-11-21 14:56:10 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
2018-07-02 00:41:37 +00:00
|
|
|
"time"
|
2017-10-13 14:14:23 +00:00
|
|
|
|
2018-08-14 14:07:26 +00:00
|
|
|
"github.com/golang/glog"
|
2017-10-13 14:14:23 +00:00
|
|
|
netutil "k8s.io/apimachinery/pkg/util/net"
|
2018-09-21 05:06:51 +00:00
|
|
|
versionutil "k8s.io/apimachinery/pkg/util/version"
|
2018-08-14 14:07:26 +00:00
|
|
|
pkgversion "k8s.io/kubernetes/pkg/version"
|
2016-11-21 14:56:10 +00:00
|
|
|
)
|
|
|
|
|
2018-07-02 00:41:37 +00:00
|
|
|
const (
|
|
|
|
getReleaseVersionTimeout = time.Duration(10 * time.Second)
|
|
|
|
)
|
|
|
|
|
2016-11-21 14:56:10 +00:00
|
|
|
var (
|
2017-08-14 17:19:36 +00:00
|
|
|
kubeReleaseBucketURL = "https://dl.k8s.io"
|
2017-07-05 16:41:57 +00:00
|
|
|
kubeReleaseRegex = regexp.MustCompile(`^v?(0|[1-9][0-9]*)\.(0|[1-9][0-9]*)\.(0|[1-9][0-9]*)([-0-9a-zA-Z_\.+]*)?$`)
|
2016-11-21 14:56:10 +00:00
|
|
|
kubeReleaseLabelRegex = regexp.MustCompile(`^[[:lower:]]+(-[-\w_\.]+)?$`)
|
2017-08-14 17:19:36 +00:00
|
|
|
kubeBucketPrefixes = regexp.MustCompile(`^((release|ci|ci-cross)/)?([-\w_\.+]+)$`)
|
2016-11-21 14:56:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// KubernetesReleaseVersion is helper function that can fetch
|
|
|
|
// available version information from release servers based on
|
|
|
|
// label names, like "stable" or "latest".
|
|
|
|
//
|
|
|
|
// If argument is already semantic version string, it
|
|
|
|
// will return same string.
|
|
|
|
//
|
|
|
|
// In case of labels, it tries to fetch from release
|
|
|
|
// servers and then return actual semantic version.
|
|
|
|
//
|
|
|
|
// Available names on release servers:
|
|
|
|
// stable (latest stable release)
|
|
|
|
// stable-1 (latest stable release in 1.x)
|
|
|
|
// stable-1.0 (and similarly 1.1, 1.2, 1.3, ...)
|
|
|
|
// latest (latest release, including alpha/beta)
|
|
|
|
// latest-1 (latest release in 1.x, including alpha/beta)
|
|
|
|
// latest-1.0 (and similarly 1.1, 1.2, 1.3, ...)
|
|
|
|
func KubernetesReleaseVersion(version string) (string, error) {
|
2017-09-25 15:08:17 +00:00
|
|
|
ver := normalizedBuildVersion(version)
|
|
|
|
if len(ver) != 0 {
|
|
|
|
return ver, nil
|
2017-08-14 17:19:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bucketURL, versionLabel, err := splitVersion(version)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2017-09-25 15:08:17 +00:00
|
|
|
|
|
|
|
// revalidate, if exact build from e.g. CI bucket requested.
|
|
|
|
ver = normalizedBuildVersion(versionLabel)
|
|
|
|
if len(ver) != 0 {
|
|
|
|
return ver, nil
|
|
|
|
}
|
|
|
|
|
2018-08-14 14:07:26 +00:00
|
|
|
// kubeReleaseLabelRegex matches labels such as: latest, latest-1, latest-1.10
|
2017-08-14 17:19:36 +00:00
|
|
|
if kubeReleaseLabelRegex.MatchString(versionLabel) {
|
|
|
|
url := fmt.Sprintf("%s/%s.txt", bucketURL, versionLabel)
|
2018-07-02 00:41:37 +00:00
|
|
|
body, err := fetchFromURL(url, getReleaseVersionTimeout)
|
2016-11-21 14:56:10 +00:00
|
|
|
if err != nil {
|
2018-08-14 14:07:26 +00:00
|
|
|
// If the network operaton was successful but the server did not reply with StatusOK
|
|
|
|
if body != "" {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
// Handle air-gapped environments by falling back to the client version.
|
|
|
|
glog.Infof("could not fetch a Kubernetes version from the internet: %v", err)
|
|
|
|
body, err = kubeadmVersion(pkgversion.Get().String())
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
glog.Infof("falling back to the local client version: %s", body)
|
2016-11-21 14:56:10 +00:00
|
|
|
}
|
|
|
|
// Re-validate received version and return.
|
2017-08-14 17:19:36 +00:00
|
|
|
return KubernetesReleaseVersion(body)
|
2016-11-21 14:56:10 +00:00
|
|
|
}
|
2016-12-09 20:16:37 +00:00
|
|
|
return "", fmt.Errorf("version %q doesn't match patterns for neither semantic version nor labels (stable, latest, ...)", version)
|
2016-11-21 14:56:10 +00:00
|
|
|
}
|
2017-07-12 11:43:29 +00:00
|
|
|
|
|
|
|
// KubernetesVersionToImageTag is helper function that replaces all
|
|
|
|
// non-allowed symbols in tag strings with underscores.
|
|
|
|
// Image tag can only contain lowercase and uppercase letters, digits,
|
|
|
|
// underscores, periods and dashes.
|
|
|
|
// Current usage is for CI images where all of symbols except '+' are valid,
|
|
|
|
// but function is for generic usage where input can't be always pre-validated.
|
|
|
|
func KubernetesVersionToImageTag(version string) string {
|
|
|
|
allowed := regexp.MustCompile(`[^-a-zA-Z0-9_\.]`)
|
|
|
|
return allowed.ReplaceAllString(version, "_")
|
|
|
|
}
|
2017-08-14 17:19:36 +00:00
|
|
|
|
|
|
|
// KubernetesIsCIVersion checks if user requested CI version
|
|
|
|
func KubernetesIsCIVersion(version string) bool {
|
|
|
|
subs := kubeBucketPrefixes.FindAllStringSubmatch(version, 1)
|
|
|
|
if len(subs) == 1 && len(subs[0]) == 4 && strings.HasPrefix(subs[0][2], "ci") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-09-25 15:08:17 +00:00
|
|
|
// Internal helper: returns normalized build version (with "v" prefix if needed)
|
|
|
|
// If input doesn't match known version pattern, returns empty string.
|
|
|
|
func normalizedBuildVersion(version string) string {
|
|
|
|
if kubeReleaseRegex.MatchString(version) {
|
|
|
|
if strings.HasPrefix(version, "v") {
|
|
|
|
return version
|
|
|
|
}
|
|
|
|
return "v" + version
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2017-08-14 17:19:36 +00:00
|
|
|
// Internal helper: split version parts,
|
|
|
|
// Return base URL and cleaned-up version
|
|
|
|
func splitVersion(version string) (string, string, error) {
|
|
|
|
var urlSuffix string
|
|
|
|
subs := kubeBucketPrefixes.FindAllStringSubmatch(version, 1)
|
|
|
|
if len(subs) != 1 || len(subs[0]) != 4 {
|
|
|
|
return "", "", fmt.Errorf("invalid version %q", version)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case strings.HasPrefix(subs[0][2], "ci"):
|
2018-05-07 19:56:07 +00:00
|
|
|
// Just use whichever the user specified
|
|
|
|
urlSuffix = subs[0][2]
|
2017-08-14 17:19:36 +00:00
|
|
|
default:
|
|
|
|
urlSuffix = "release"
|
|
|
|
}
|
|
|
|
url := fmt.Sprintf("%s/%s", kubeReleaseBucketURL, urlSuffix)
|
|
|
|
return url, subs[0][3], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Internal helper: return content of URL
|
2018-07-02 00:41:37 +00:00
|
|
|
func fetchFromURL(url string, timeout time.Duration) (string, error) {
|
2018-08-14 14:07:26 +00:00
|
|
|
glog.V(2).Infof("fetching Kubernetes version from URL: %s", url)
|
2018-07-02 00:41:37 +00:00
|
|
|
client := &http.Client{Timeout: timeout, Transport: netutil.SetOldTransportDefaults(&http.Transport{})}
|
2017-10-13 14:14:23 +00:00
|
|
|
resp, err := client.Get(url)
|
2017-08-14 17:19:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("unable to get URL %q: %s", url, err.Error())
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("unable to read content of URL %q: %s", url, err.Error())
|
|
|
|
}
|
2018-08-14 14:07:26 +00:00
|
|
|
bodyString := strings.TrimSpace(string(body))
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
msg := fmt.Sprintf("unable to fetch file. URL: %q, status: %v", url, resp.Status)
|
|
|
|
return bodyString, errors.New(msg)
|
|
|
|
}
|
|
|
|
return bodyString, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// kubeadmVersion returns the version of the client without metadata.
|
|
|
|
func kubeadmVersion(info string) (string, error) {
|
|
|
|
v, err := versionutil.ParseSemantic(info)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("kubeadm version error: %v", err)
|
|
|
|
}
|
|
|
|
// There is no utility in versionutil to get the version without the metadata,
|
|
|
|
// so this needs some manual formatting.
|
|
|
|
// Discard offsets after a release label and keep the labels down to e.g. `alpha.0` instead of
|
|
|
|
// including the offset e.g. `alpha.0.206`. This is done to comply with GCR image tags.
|
|
|
|
pre := v.PreRelease()
|
|
|
|
patch := v.Patch()
|
|
|
|
if len(pre) > 0 {
|
|
|
|
if patch > 0 {
|
|
|
|
// If the patch version is more than zero, decrement it and remove the label.
|
|
|
|
// this is done to comply with the latest stable patch release.
|
|
|
|
patch = patch - 1
|
|
|
|
pre = ""
|
|
|
|
} else {
|
|
|
|
split := strings.Split(pre, ".")
|
|
|
|
if len(split) > 2 {
|
|
|
|
pre = split[0] + "." + split[1] // Exclude the third element
|
|
|
|
} else if len(split) < 2 {
|
|
|
|
pre = split[0] + ".0" // Append .0 to a partial label
|
|
|
|
}
|
|
|
|
pre = "-" + pre
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vStr := fmt.Sprintf("v%d.%d.%d%s", v.Major(), v.Minor(), patch, pre)
|
|
|
|
return vStr, nil
|
2017-08-14 17:19:36 +00:00
|
|
|
}
|