2014-06-17 06:16:40 +00:00
|
|
|
#!/bin/bash
|
|
|
|
|
2015-05-01 16:19:44 +00:00
|
|
|
# Copyright 2014 The Kubernetes Authors All rights reserved.
|
2014-06-17 06:16:40 +00:00
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2014-10-22 23:26:59 +00:00
|
|
|
kube::util::sortable_date() {
|
|
|
|
date "+%Y%m%d-%H%M%S"
|
|
|
|
}
|
2014-06-17 06:16:40 +00:00
|
|
|
|
2014-10-22 23:26:59 +00:00
|
|
|
kube::util::wait_for_url() {
|
|
|
|
local url=$1
|
|
|
|
local prefix=${2:-}
|
2016-02-22 21:31:28 +00:00
|
|
|
local wait=${3:-1}
|
|
|
|
local times=${4:-30}
|
2014-06-17 06:16:40 +00:00
|
|
|
|
2015-02-05 08:31:02 +00:00
|
|
|
which curl >/dev/null || {
|
|
|
|
kube::log::usage "curl must be installed"
|
|
|
|
exit 1
|
|
|
|
}
|
|
|
|
|
2015-02-06 00:04:16 +00:00
|
|
|
local i
|
2014-10-22 23:26:59 +00:00
|
|
|
for i in $(seq 1 $times); do
|
|
|
|
local out
|
|
|
|
if out=$(curl -fs $url 2>/dev/null); then
|
2015-06-09 21:15:01 +00:00
|
|
|
kube::log::status "On try ${i}, ${prefix}: ${out}"
|
2014-10-22 23:26:59 +00:00
|
|
|
return 0
|
|
|
|
fi
|
2015-06-09 21:15:01 +00:00
|
|
|
sleep ${wait}
|
2014-10-22 23:26:59 +00:00
|
|
|
done
|
2015-06-09 21:15:01 +00:00
|
|
|
kube::log::error "Timed out waiting for ${prefix} to answer at ${url}; tried ${times} waiting ${wait} between each"
|
2014-10-22 23:26:59 +00:00
|
|
|
return 1
|
|
|
|
}
|
2014-11-17 00:29:58 +00:00
|
|
|
|
2015-09-30 00:02:28 +00:00
|
|
|
# returns a random port
|
|
|
|
kube::util::get_random_port() {
|
2016-01-19 23:04:12 +00:00
|
|
|
awk -v min=1024 -v max=65535 'BEGIN{srand(); print int(min+rand()*(max-min+1))}'
|
2015-09-30 00:02:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# use netcat to check if the host($1):port($2) is free (return 0 means free, 1 means used)
|
|
|
|
kube::util::test_host_port_free() {
|
|
|
|
local host=$1
|
|
|
|
local port=$2
|
|
|
|
local success=0
|
|
|
|
local fail=1
|
|
|
|
|
|
|
|
which nc >/dev/null || {
|
|
|
|
kube::log::usage "netcat isn't installed, can't verify if ${host}:${port} is free, skipping the check..."
|
|
|
|
return ${success}
|
|
|
|
}
|
|
|
|
|
2016-01-19 23:04:12 +00:00
|
|
|
if [ ! $(nc -vz "${host}" "${port}") ]; then
|
2015-09-30 00:02:28 +00:00
|
|
|
kube::log::status "${host}:${port} is free, proceeding..."
|
|
|
|
return ${success}
|
|
|
|
else
|
|
|
|
kube::log::status "${host}:${port} is already used"
|
|
|
|
return ${fail}
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2015-08-21 01:28:05 +00:00
|
|
|
# Example: kube::util::trap_add 'echo "in trap DEBUG"' DEBUG
|
|
|
|
# See: http://stackoverflow.com/questions/3338030/multiple-bash-traps-for-the-same-signal
|
|
|
|
kube::util::trap_add() {
|
|
|
|
local trap_add_cmd
|
|
|
|
trap_add_cmd=$1
|
|
|
|
shift
|
|
|
|
|
|
|
|
for trap_add_name in "$@"; do
|
|
|
|
local existing_cmd
|
|
|
|
local new_cmd
|
|
|
|
|
|
|
|
# Grab the currently defined trap commands for this trap
|
|
|
|
existing_cmd=`trap -p "${trap_add_name}" | awk -F"'" '{print $2}'`
|
|
|
|
|
|
|
|
if [[ -z "${existing_cmd}" ]]; then
|
|
|
|
new_cmd="${trap_add_cmd}"
|
|
|
|
else
|
|
|
|
new_cmd="${existing_cmd};${trap_add_cmd}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Assign the test
|
|
|
|
trap "${new_cmd}" "${trap_add_name}"
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
# Opposite of kube::util::ensure-temp-dir()
|
|
|
|
kube::util::cleanup-temp-dir() {
|
|
|
|
rm -rf "${KUBE_TEMP}"
|
|
|
|
}
|
|
|
|
|
2015-03-22 21:43:00 +00:00
|
|
|
# Create a temp dir that'll be deleted at the end of this bash session.
|
|
|
|
#
|
|
|
|
# Vars set:
|
|
|
|
# KUBE_TEMP
|
|
|
|
kube::util::ensure-temp-dir() {
|
|
|
|
if [[ -z ${KUBE_TEMP-} ]]; then
|
2015-07-30 04:41:20 +00:00
|
|
|
KUBE_TEMP=$(mktemp -d 2>/dev/null || mktemp -d -t kubernetes.XXXXXX)
|
2015-08-21 01:28:05 +00:00
|
|
|
kube::util::trap_add kube::util::cleanup-temp-dir EXIT
|
2015-03-22 21:43:00 +00:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2014-11-17 00:29:58 +00:00
|
|
|
# This figures out the host platform without relying on golang. We need this as
|
|
|
|
# we don't want a golang install to be a prerequisite to building yet we need
|
|
|
|
# this info to figure out where the final binaries are placed.
|
|
|
|
kube::util::host_platform() {
|
|
|
|
local host_os
|
|
|
|
local host_arch
|
|
|
|
case "$(uname -s)" in
|
|
|
|
Darwin)
|
|
|
|
host_os=darwin
|
|
|
|
;;
|
|
|
|
Linux)
|
|
|
|
host_os=linux
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
kube::log::error "Unsupported host OS. Must be Linux or Mac OS X."
|
|
|
|
exit 1
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
case "$(uname -m)" in
|
|
|
|
x86_64*)
|
|
|
|
host_arch=amd64
|
|
|
|
;;
|
|
|
|
i?86_64*)
|
|
|
|
host_arch=amd64
|
|
|
|
;;
|
|
|
|
amd64*)
|
|
|
|
host_arch=amd64
|
|
|
|
;;
|
2016-05-15 13:38:11 +00:00
|
|
|
aarch64*)
|
|
|
|
host_arch=arm64
|
|
|
|
;;
|
|
|
|
arm64*)
|
|
|
|
host_arch=arm64
|
|
|
|
;;
|
2014-11-17 00:29:58 +00:00
|
|
|
arm*)
|
|
|
|
host_arch=arm
|
|
|
|
;;
|
|
|
|
i?86*)
|
|
|
|
host_arch=x86
|
|
|
|
;;
|
2015-11-02 10:29:24 +00:00
|
|
|
s390x*)
|
|
|
|
host_arch=s390x
|
|
|
|
;;
|
2015-12-09 06:42:13 +00:00
|
|
|
ppc64le*)
|
|
|
|
host_arch=ppc64le
|
|
|
|
;;
|
2016-04-13 22:11:31 +00:00
|
|
|
*)
|
2016-05-15 13:38:11 +00:00
|
|
|
kube::log::error "Unsupported host arch. Must be x86_64, 386, arm, arm64, s390x or ppc64le."
|
2014-11-17 00:29:58 +00:00
|
|
|
exit 1
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
echo "${host_os}/${host_arch}"
|
|
|
|
}
|
2015-04-06 18:09:43 +00:00
|
|
|
|
2015-04-13 17:45:13 +00:00
|
|
|
kube::util::find-binary() {
|
|
|
|
local lookfor="${1}"
|
|
|
|
local host_platform="$(kube::util::host_platform)"
|
|
|
|
local locations=(
|
|
|
|
"${KUBE_ROOT}/_output/dockerized/bin/${host_platform}/${lookfor}"
|
|
|
|
"${KUBE_ROOT}/_output/local/bin/${host_platform}/${lookfor}"
|
|
|
|
"${KUBE_ROOT}/platforms/${host_platform}/${lookfor}"
|
|
|
|
)
|
|
|
|
local bin=$( (ls -t "${locations[@]}" 2>/dev/null || true) | head -1 )
|
|
|
|
echo -n "${bin}"
|
|
|
|
}
|
|
|
|
|
2015-08-21 02:17:26 +00:00
|
|
|
# Run all known doc generators (today gendocs, genman, and genbashcomp for kubectl)
|
|
|
|
# $1 is the directory to put those generated documents
|
|
|
|
kube::util::gen-docs() {
|
|
|
|
local dest="$1"
|
|
|
|
|
|
|
|
# Find binary
|
|
|
|
gendocs=$(kube::util::find-binary "gendocs")
|
2015-10-12 14:33:39 +00:00
|
|
|
genkubedocs=$(kube::util::find-binary "genkubedocs")
|
2015-08-21 02:17:26 +00:00
|
|
|
genman=$(kube::util::find-binary "genman")
|
2016-04-13 22:11:31 +00:00
|
|
|
genyaml=$(kube::util::find-binary "genyaml")
|
2015-08-21 02:17:26 +00:00
|
|
|
genbashcomp=$(kube::util::find-binary "genbashcomp")
|
2016-05-25 21:55:33 +00:00
|
|
|
genfeddocs=$(kube::util::find-binary "genfeddocs")
|
2015-08-21 02:17:26 +00:00
|
|
|
|
|
|
|
mkdir -p "${dest}/docs/user-guide/kubectl/"
|
|
|
|
"${gendocs}" "${dest}/docs/user-guide/kubectl/"
|
2015-10-12 14:33:39 +00:00
|
|
|
mkdir -p "${dest}/docs/admin/"
|
|
|
|
"${genkubedocs}" "${dest}/docs/admin/" "kube-apiserver"
|
|
|
|
"${genkubedocs}" "${dest}/docs/admin/" "kube-controller-manager"
|
|
|
|
"${genkubedocs}" "${dest}/docs/admin/" "kube-proxy"
|
|
|
|
"${genkubedocs}" "${dest}/docs/admin/" "kube-scheduler"
|
|
|
|
"${genkubedocs}" "${dest}/docs/admin/" "kubelet"
|
2016-05-25 21:55:33 +00:00
|
|
|
"${genfeddocs}" "${dest}/docs/admin/" "federation-apiserver"
|
|
|
|
"${genfeddocs}" "${dest}/docs/admin/" "federation-controller-manager"
|
2015-08-21 02:17:26 +00:00
|
|
|
mkdir -p "${dest}/docs/man/man1/"
|
|
|
|
"${genman}" "${dest}/docs/man/man1/"
|
2016-04-13 22:11:31 +00:00
|
|
|
mkdir -p "${dest}/docs/yaml/kubectl/"
|
|
|
|
"${genyaml}" "${dest}/docs/yaml/kubectl/"
|
2015-08-21 02:17:26 +00:00
|
|
|
mkdir -p "${dest}/contrib/completions/bash/"
|
|
|
|
"${genbashcomp}" "${dest}/contrib/completions/bash/"
|
|
|
|
|
2015-04-13 16:14:08 +00:00
|
|
|
# create the list of generated files
|
2015-08-21 02:17:26 +00:00
|
|
|
pushd "${dest}" > /dev/null
|
|
|
|
touch .generated_docs
|
2015-09-16 22:16:37 +00:00
|
|
|
find . -type f | cut -sd / -f 2- | LC_ALL=C sort > .generated_docs
|
2015-08-21 02:17:26 +00:00
|
|
|
popd > /dev/null
|
2015-04-23 21:21:13 +00:00
|
|
|
|
|
|
|
while read file; do
|
2015-08-21 02:17:26 +00:00
|
|
|
# Copy out of KUBE_ROOT if we didn't really change anything
|
|
|
|
if [[ -e "${dest}/${file}" && -e "${KUBE_ROOT}/${file}" ]]; then
|
2015-07-11 22:57:52 +00:00
|
|
|
# Filter all munges from original content.
|
2015-08-21 02:17:26 +00:00
|
|
|
original=$(cat "${KUBE_ROOT}/${file}" | sed '/^<!-- BEGIN MUNGE:.*/,/^<!-- END MUNGE:.*/d')
|
|
|
|
generated=$(cat "${dest}/${file}")
|
|
|
|
|
|
|
|
# Filter out meaningless lines with timestamps
|
|
|
|
original=$(echo "${original}" | grep -v "Auto generated by spf13/cobra" || :)
|
|
|
|
generated=$(echo "${generated}" | grep -v "Auto generated by spf13/cobra" || :)
|
|
|
|
|
2015-07-11 22:57:52 +00:00
|
|
|
# By now, the contents should be normalized and stripped of any
|
2015-07-17 19:49:22 +00:00
|
|
|
# auto-managed content. We also ignore whitespace here because of
|
|
|
|
# markdown strictness fixups later in the pipeline.
|
2015-08-21 02:17:26 +00:00
|
|
|
if diff -Bw >/dev/null <(echo "${original}") <(echo "${generated}"); then
|
2015-05-14 22:12:45 +00:00
|
|
|
# actual contents same, overwrite generated with original.
|
2015-08-21 02:17:26 +00:00
|
|
|
cp "${KUBE_ROOT}/${file}" "${dest}/${file}"
|
2015-04-23 21:21:13 +00:00
|
|
|
fi
|
|
|
|
fi
|
2015-08-21 02:17:26 +00:00
|
|
|
done <"${KUBE_ROOT}/.generated_docs"
|
2015-04-13 16:14:08 +00:00
|
|
|
}
|
|
|
|
|
2015-05-14 22:12:45 +00:00
|
|
|
# Takes a path $1 to traverse for md files to append the ga-beacon tracking
|
|
|
|
# link to, if needed. If $2 is set, just print files that are missing
|
|
|
|
# the link.
|
|
|
|
kube::util::gen-analytics() {
|
|
|
|
local path="$1"
|
|
|
|
local dryrun="${2:-}"
|
|
|
|
local mdfiles dir link
|
|
|
|
# find has some strange inconsistencies between darwin/linux. The
|
|
|
|
# path to search must end in '/' for linux, but darwin will put an extra
|
|
|
|
# slash in results if there is a trailing '/'.
|
|
|
|
if [[ $( uname ) == 'Linux' ]]; then
|
|
|
|
dir="${path}/"
|
|
|
|
else
|
|
|
|
dir="${path}"
|
|
|
|
fi
|
2016-04-14 06:30:15 +00:00
|
|
|
# We don't touch files in special dirs, and the kubectl docs are
|
|
|
|
# autogenerated by gendocs.
|
2015-05-14 22:12:45 +00:00
|
|
|
mdfiles=($( find "${dir}" -name "*.md" -type f \
|
2016-04-14 06:30:15 +00:00
|
|
|
-not -path "${path}/vendor/*" \
|
2015-05-14 22:12:45 +00:00
|
|
|
-not -path "${path}/third_party/*" \
|
2015-09-25 22:01:59 +00:00
|
|
|
-not -path "${path}/_gopath/*" \
|
2015-05-14 22:12:45 +00:00
|
|
|
-not -path "${path}/_output/*" \
|
2015-07-13 17:41:04 +00:00
|
|
|
-not -path "${path}/docs/user-guide/kubectl/kubectl*" ))
|
2015-05-14 22:12:45 +00:00
|
|
|
for f in "${mdfiles[@]}"; do
|
|
|
|
link=$(kube::util::analytics-link "${f#${path}/}")
|
|
|
|
if grep -q -F -x "${link}" "${f}"; then
|
|
|
|
continue
|
|
|
|
elif [[ -z "${dryrun}" ]]; then
|
|
|
|
echo -e "\n\n${link}" >> "${f}"
|
|
|
|
else
|
|
|
|
echo "$f"
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
# Prints analytics link to append to a file at path $1.
|
|
|
|
kube::util::analytics-link() {
|
|
|
|
local path="$1"
|
|
|
|
echo "[![Analytics](https://kubernetes-site.appspot.com/UA-36037335-10/GitHub/${path}?pixel)]()"
|
|
|
|
}
|
|
|
|
|
2015-09-09 23:01:08 +00:00
|
|
|
# Takes a group/version and returns the path to its location on disk, sans
|
|
|
|
# "pkg". E.g.:
|
2015-10-09 22:19:13 +00:00
|
|
|
# * default behavior: extensions/v1beta1 -> apis/extensions/v1beta1
|
2015-09-22 23:56:45 +00:00
|
|
|
# * default behavior for only a group: experimental -> apis/experimental
|
|
|
|
# * Special handling for empty group: v1 -> api/v1, unversioned -> api/unversioned
|
|
|
|
# * Very special handling for when both group and version are "": / -> api
|
2015-09-09 23:01:08 +00:00
|
|
|
kube::util::group-version-to-pkg-path() {
|
|
|
|
local group_version="$1"
|
|
|
|
# Special cases first.
|
|
|
|
# TODO(lavalamp): Simplify this by moving pkg/api/v1 and splitting pkg/api,
|
|
|
|
# moving the results to pkg/apis/api.
|
|
|
|
case "${group_version}" in
|
2015-09-17 05:15:05 +00:00
|
|
|
# both group and version are "", this occurs when we generate deep copies for internal objects of the legacy v1 API.
|
2015-12-21 05:25:12 +00:00
|
|
|
__internal)
|
2015-09-17 05:15:05 +00:00
|
|
|
echo "api"
|
|
|
|
;;
|
2015-09-09 23:01:08 +00:00
|
|
|
v1)
|
|
|
|
echo "api/v1"
|
|
|
|
;;
|
2015-09-22 23:56:45 +00:00
|
|
|
unversioned)
|
|
|
|
echo "api/unversioned"
|
2015-09-09 23:01:08 +00:00
|
|
|
;;
|
|
|
|
*)
|
2015-12-21 05:25:12 +00:00
|
|
|
echo "apis/${group_version%__internal}"
|
2015-09-09 23:01:08 +00:00
|
|
|
;;
|
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
2016-02-09 03:11:06 +00:00
|
|
|
# Takes a group/version and returns the swagger-spec file name.
|
|
|
|
# default behavior: extensions/v1beta1 -> extensions_v1beta1
|
|
|
|
# special case for v1: v1 -> v1
|
|
|
|
kube::util::gv-to-swagger-name() {
|
|
|
|
local group_version="$1"
|
|
|
|
case "${group_version}" in
|
|
|
|
v1)
|
|
|
|
echo "v1"
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
echo "${group_version%/*}_${group_version#*/}"
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
}
|
|
|
|
|
2016-03-09 23:47:16 +00:00
|
|
|
# Returns the name of the upstream remote repository name for the local git
|
|
|
|
# repo, e.g. "upstream" or "origin".
|
|
|
|
kube::util::git_upstream_remote_name() {
|
|
|
|
git remote -v | grep fetch |\
|
2016-05-14 07:48:14 +00:00
|
|
|
grep -E 'github.com[/:]kubernetes/kubernetes|k8s.io/kubernetes' |\
|
2016-03-09 23:47:16 +00:00
|
|
|
head -n 1 | awk '{print $1}'
|
|
|
|
}
|
|
|
|
|
|
|
|
# Checks whether there are any files matching pattern $2 changed between the
|
|
|
|
# current branch and upstream branch named by $1.
|
|
|
|
# Returns 1 (false) if there are no changes, 0 (true) if there are changes
|
|
|
|
# detected.
|
|
|
|
kube::util::has_changes_against_upstream_branch() {
|
|
|
|
local -r git_branch=$1
|
|
|
|
local -r pattern=$2
|
2016-05-27 23:27:42 +00:00
|
|
|
local full_branch
|
2016-03-09 23:47:16 +00:00
|
|
|
|
2016-05-27 23:27:42 +00:00
|
|
|
full_branch="$(kube::util::git_upstream_remote_name)/${git_branch}"
|
2016-03-09 23:47:16 +00:00
|
|
|
echo "Checking for '${pattern}' changes against '${full_branch}'"
|
|
|
|
# make sure the branch is valid, otherwise the check will pass erroneously.
|
|
|
|
if ! git describe "${full_branch}" >/dev/null; then
|
2016-03-14 19:56:13 +00:00
|
|
|
# abort!
|
2016-03-09 23:47:16 +00:00
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
# notice this uses ... to find the first shared ancestor
|
2016-03-14 19:56:13 +00:00
|
|
|
if git diff --name-only "${full_branch}...HEAD" | grep "${pattern}" > /dev/null; then
|
|
|
|
return 0
|
2016-03-09 23:47:16 +00:00
|
|
|
fi
|
2016-03-14 19:56:13 +00:00
|
|
|
# also check for pending changes
|
|
|
|
if git status --porcelain | grep "${pattern}" > /dev/null; then
|
|
|
|
echo "Detected '${pattern}' uncommitted changes."
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
echo "No '${pattern}' changes detected."
|
|
|
|
return 1
|
2016-03-09 23:47:16 +00:00
|
|
|
}
|
|
|
|
|
2015-04-13 16:14:08 +00:00
|
|
|
# ex: ts=2 sw=2 et filetype=sh
|