mirror of
https://github.com/containers/podman.git
synced 2025-06-23 18:59:30 +08:00
Merge pull request #9840 from cevich/new_get_ci_vm.sh
Cirrus: Make use of shared get_ci_vm container
This commit is contained in:
@ -30,7 +30,7 @@ env:
|
|||||||
PRIOR_UBUNTU_NAME: "ubuntu-2004"
|
PRIOR_UBUNTU_NAME: "ubuntu-2004"
|
||||||
|
|
||||||
# Google-cloud VM Images
|
# Google-cloud VM Images
|
||||||
IMAGE_SUFFIX: "c5961289315909632"
|
IMAGE_SUFFIX: "c5501386583441408"
|
||||||
FEDORA_CACHE_IMAGE_NAME: "fedora-${IMAGE_SUFFIX}"
|
FEDORA_CACHE_IMAGE_NAME: "fedora-${IMAGE_SUFFIX}"
|
||||||
PRIOR_FEDORA_CACHE_IMAGE_NAME: "prior-fedora-${IMAGE_SUFFIX}"
|
PRIOR_FEDORA_CACHE_IMAGE_NAME: "prior-fedora-${IMAGE_SUFFIX}"
|
||||||
UBUNTU_CACHE_IMAGE_NAME: "ubuntu-${IMAGE_SUFFIX}"
|
UBUNTU_CACHE_IMAGE_NAME: "ubuntu-${IMAGE_SUFFIX}"
|
||||||
|
@ -13,8 +13,12 @@ set -a
|
|||||||
_waserrexit=0
|
_waserrexit=0
|
||||||
if [[ "$SHELLOPTS" =~ errexit ]]; then _waserrexit=1; fi
|
if [[ "$SHELLOPTS" =~ errexit ]]; then _waserrexit=1; fi
|
||||||
set +e # Assumed in F33 for setting global vars
|
set +e # Assumed in F33 for setting global vars
|
||||||
source /etc/profile
|
if [[ -r "/etc/automation_environment" ]]; then
|
||||||
source /etc/environment
|
source /etc/automation_environment
|
||||||
|
else # prior to automation library v2.0, this was necessary
|
||||||
|
source /etc/profile
|
||||||
|
source /etc/environment
|
||||||
|
fi
|
||||||
if [[ -r "/etc/ci_environment" ]]; then source /etc/ci_environment; fi
|
if [[ -r "/etc/ci_environment" ]]; then source /etc/ci_environment; fi
|
||||||
USER="$(whoami)"
|
USER="$(whoami)"
|
||||||
HOME="$(getent passwd $USER | cut -d : -f 6)"
|
HOME="$(getent passwd $USER | cut -d : -f 6)"
|
||||||
|
@ -269,13 +269,18 @@ function _run_release() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
logformatter() {
|
logformatter() {
|
||||||
# Use similar format as human-friendly task name from .cirrus.yml
|
if [[ "$CI" == "true" ]]; then
|
||||||
# shellcheck disable=SC2154
|
# Use similar format as human-friendly task name from .cirrus.yml
|
||||||
output_name="$TEST_FLAVOR-$PODBIN_NAME-$DISTRO_NV-$PRIV_NAME-$TEST_ENVIRON"
|
# shellcheck disable=SC2154
|
||||||
# Requires stdin and stderr combined!
|
output_name="$TEST_FLAVOR-$PODBIN_NAME-$DISTRO_NV-$PRIV_NAME-$TEST_ENVIRON"
|
||||||
cat - \
|
# Requires stdin and stderr combined!
|
||||||
|& awk --file "${CIRRUS_WORKING_DIR}/${SCRIPT_BASE}/timestamp.awk" \
|
cat - \
|
||||||
|& "${CIRRUS_WORKING_DIR}/${SCRIPT_BASE}/logformatter" "$output_name"
|
|& awk --file "${CIRRUS_WORKING_DIR}/${SCRIPT_BASE}/timestamp.awk" \
|
||||||
|
|& "${CIRRUS_WORKING_DIR}/${SCRIPT_BASE}/logformatter" "$output_name"
|
||||||
|
else
|
||||||
|
# Assume script is run by a human, they want output immediatly
|
||||||
|
cat -
|
||||||
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Handle local|remote integration|system testing in a uniform way
|
# Handle local|remote integration|system testing in a uniform way
|
||||||
|
@ -3,270 +3,68 @@
|
|||||||
#
|
#
|
||||||
# For help and usage information, simply execute the script w/o any arguments.
|
# For help and usage information, simply execute the script w/o any arguments.
|
||||||
#
|
#
|
||||||
# This script is intended to be run by podman developers who need to debug
|
# This script is intended to be run by Red Hat podman developers who need
|
||||||
# problems specifically related to Cirrus-CI automated testing. However,
|
# to debug problems specifically related to Cirrus-CI automated testing.
|
||||||
# because it's only loosely coupled to the `.cirrus.yml` configuration, it must
|
# It requires that you have been granted prior access to create VMs in
|
||||||
# orchestrate VMs in GCP directly. This means users need to have
|
# google-cloud. For non-Red Hat contributors, VMs are available as-needed,
|
||||||
# pre-authorization (access) to manipulate google-cloud resources. Additionally,
|
# with supervision upon request.
|
||||||
# there are no guarantees it will remain in-sync with other automation-related
|
|
||||||
# scripts. Therefore it may not always function for everybody in every
|
|
||||||
# future scenario without updates/modifications/tweaks.
|
|
||||||
#
|
|
||||||
# When successful, you will end up connected to a GCP VM with with a clone of
|
|
||||||
# the upstream podman repository 'master' branch, using a remote named 'origin'.
|
|
||||||
# If you want to customize this behavior, you will want to use a "hook" script.
|
|
||||||
# Please use this example carefully, since git setups vary by person, you
|
|
||||||
# will probably need to make local edits.
|
|
||||||
#
|
|
||||||
# https://gist.github.com/cevich/626a0790c0b476d5cd2a5a76fbdae0a1
|
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
RED="\e[1;31m"
|
SCRIPT_FILEPATH=$(realpath "${BASH_SOURCE[0]}")
|
||||||
YEL="\e[1;32m"
|
SCRIPT_DIRPATH=$(dirname "$SCRIPT_FILEPATH")
|
||||||
NOR="\e[0m"
|
REPO_DIRPATH=$(realpath "$SCRIPT_DIRPATH/../")
|
||||||
USAGE_WARNING="
|
|
||||||
${YEL}WARNING: This will not work without podman,${NOR}
|
|
||||||
${YEL}and prior authorization to use the libpod GCP project.${NOR}
|
|
||||||
"
|
|
||||||
# These values come from .cirrus.yml gce_instance clause
|
|
||||||
ZONE="${ZONE:-us-central1-a}"
|
|
||||||
CPUS="2"
|
|
||||||
MEMORY="4Gb"
|
|
||||||
DISK="200"
|
|
||||||
PROJECT="libpod-218412"
|
|
||||||
GOSRC="/var/tmp/go/src/github.com/containers/podman"
|
|
||||||
GIT_REPO="https://github.com/containers/podman.git"
|
|
||||||
|
|
||||||
# Container image with necessary runtime elements
|
# Help detect if we were called by get_ci_vm container
|
||||||
GCLOUD_IMAGE="${GCLOUD_IMAGE:-docker.io/google/cloud-sdk:alpine}"
|
GET_CI_VM="${GET_CI_VM:-0}"
|
||||||
GCLOUD_CFGDIR=".config/gcloud"
|
in_get_ci_vm() {
|
||||||
|
if ((GET_CI_VM==0)); then
|
||||||
SCRIPT_FILENAME=$(basename ${BASH_SOURCE[0]})
|
echo "Error: $1 is not intended for use in this context"
|
||||||
HOOK_FILENAME="hook_${SCRIPT_FILENAME}"
|
exit 2
|
||||||
|
|
||||||
# Shared tmp directory between container and us
|
|
||||||
TMPDIR=$(mktemp -d --tmpdir ${SCRIPT_FILENAME}_tmpdir_XXXXXX)
|
|
||||||
|
|
||||||
show_usage() {
|
|
||||||
echo -e "\n${RED}ERROR: $1${NOR}"
|
|
||||||
echo -e "${YEL}Usage: $SCRIPT_FILENAME <image_name>${NOR}"
|
|
||||||
echo ""
|
|
||||||
if [[ -r ".cirrus.yml" ]]
|
|
||||||
then
|
|
||||||
echo -e "${YEL}Some possible image_name values (from .cirrus.yml):${NOR}"
|
|
||||||
image_hints
|
|
||||||
echo ""
|
|
||||||
echo -e "${YEL}Optional:${NOR} If a $HOME/$GCLOUD_CFGDIR/$HOOK_FILENAME executable exists during"
|
|
||||||
echo "VM creation, it will be executed remotely after cloning"
|
|
||||||
echo "$GIT_REPO. The"
|
|
||||||
echo "current local working branch name and commit ID, will be provided as"
|
|
||||||
echo "it's arguments."
|
|
||||||
fi
|
fi
|
||||||
exit 1
|
|
||||||
}
|
}
|
||||||
|
|
||||||
LIBPODROOT=$(realpath "$(dirname ${BASH_SOURCE[0]})/../")
|
# get_ci_vm APIv1 container entrypoint calls into this script
|
||||||
# else: Assume $PWD is the root of the libpod repository
|
# to obtain required repo. specific configuration options.
|
||||||
[[ "$LIBPODROOT" != "/" ]] || \
|
if [[ "$1" == "--config" ]]; then
|
||||||
show_usage "Must execute script from within clone of containers/podman repo."
|
in_get_ci_vm "$1"
|
||||||
|
cat <<EOF
|
||||||
[[ "$UID" -ne 0 ]] || \
|
DESTDIR="/var/tmp/go/src/github.com/containers/podman"
|
||||||
show_usage "Must execute script as a regular (non-root) user."
|
UPSTREAM_REPO="https://github.com/containers/podman.git"
|
||||||
|
CI_ENVFILE="/etc/ci_environment"
|
||||||
[[ "${LIBPODROOT#$HOME}" != "$LIBPODROOT" ]] || \
|
GCLOUD_PROJECT="libpod-218412"
|
||||||
show_usage "Clone of containers/podman must be a subdirectory of \$HOME ($HOME)"
|
GCLOUD_IMGPROJECT="libpod-218412"
|
||||||
|
GCLOUD_CFG="libpod"
|
||||||
# Disable SELinux labeling to allow read-only mounting of repository files
|
GCLOUD_ZONE="${GCLOUD_ZONE:-us-central1-a}"
|
||||||
PGCLOUD="podman run -it --rm --security-opt label=disable -v $TMPDIR:$TMPDIR -v $HOME/.config/gcloud:/root/.config/gcloud -v $HOME/.config/gcloud/ssh:/root/.ssh -v $LIBPODROOT:$LIBPODROOT:ro $GCLOUD_IMAGE gcloud --configuration=libpod --project=$PROJECT"
|
GCLOUD_CPUS="2"
|
||||||
SCP_CMD="$PGCLOUD compute scp"
|
GCLOUD_MEMORY="4Gb"
|
||||||
|
GCLOUD_DISK="200"
|
||||||
showrun() {
|
EOF
|
||||||
echo '+ '$(printf " %q" "$@") > /dev/stderr
|
elif [[ "$1" == "--setup" ]]; then
|
||||||
echo ""
|
in_get_ci_vm "$1"
|
||||||
"$@"
|
# get_ci_vm container entrypoint calls us with this option on the
|
||||||
}
|
# Cirrus-CI environment instance, to perform repo.-specific setup.
|
||||||
|
cd $REPO_DIRPATH
|
||||||
cleanup() {
|
echo "+ Loading ./contrib/cirrus/lib.sh" > /dev/stderr
|
||||||
RET=$?
|
source ./contrib/cirrus/lib.sh
|
||||||
set +e
|
echo "+ Mimicking .cirrus.yml clone_script and build_task" > /dev/stderr
|
||||||
wait
|
make install.tools
|
||||||
|
make vendor
|
||||||
# set GCLOUD_DEBUG to leave tmpdir behind for postmortem
|
make podman
|
||||||
# shellcheck disable=SC2154
|
make podman-remote
|
||||||
test -z "$GCLOUD_DEBUG" && rm -rf $TMPDIR
|
echo "+ Running environment setup" > /dev/stderr
|
||||||
|
./contrib/cirrus/setup_environment.sh
|
||||||
# Not always called from an exit handler, but should always exit when called
|
else
|
||||||
exit $RET
|
# Create and access VM for specified Cirrus-CI task
|
||||||
}
|
mkdir -p $HOME/.config/gcloud/ssh
|
||||||
trap cleanup EXIT
|
podman run -it --rm \
|
||||||
|
--tz=local \
|
||||||
delvm() {
|
-e NAME="$USER" \
|
||||||
echo -e "\n"
|
-e SRCDIR=/src \
|
||||||
echo -e "\n${YEL}Offering to Delete $VMNAME${NOR}"
|
-e GCLOUD_ZONE="$GCLOUD_ZONE" \
|
||||||
echo -e "${RED}(Deletion might take a minute or two)${NOR}"
|
-e DEBUG="${DEBUG:-0}" \
|
||||||
echo -e "${YEL}Note: It's safe to answer N, then re-run script again later.${NOR}"
|
-v $REPO_DIRPATH:/src:O \
|
||||||
showrun $CLEANUP_CMD # prompts for Yes/No
|
-v $HOME/.config/gcloud:/root/.config/gcloud:z \
|
||||||
cleanup
|
-v $HOME/.config/gcloud/ssh:/root/.ssh:z \
|
||||||
}
|
quay.io/libpod/get_ci_vm:latest "$@"
|
||||||
|
|
||||||
get_env_vars() {
|
|
||||||
# Deal with both YAML and embedded shell-like substitutions in values
|
|
||||||
# if substitution fails, fall back to printing naked env. var as-is.
|
|
||||||
python3 -c '
|
|
||||||
import sys,yaml,re
|
|
||||||
env=yaml.load(open(".cirrus.yml"), Loader=yaml.SafeLoader)["env"]
|
|
||||||
dollar_env_var=re.compile(r"\$(\w+)")
|
|
||||||
dollarcurly_env_var=re.compile(r"\$\{(\w+)\}")
|
|
||||||
class ReIterKey(dict):
|
|
||||||
def __missing__(self, key):
|
|
||||||
# Cirrus-CI provides some runtime-only env. vars. Avoid
|
|
||||||
# breaking this hack-script if/when any are present in YAML
|
|
||||||
return "${0}".format(key)
|
|
||||||
rep=r"{\1}" # Convert env vars markup to -> str.format_map(re_iter_key) markup
|
|
||||||
out=ReIterKey()
|
|
||||||
for k,v in env.items():
|
|
||||||
if "ENCRYPTED" not in str(v) and bool(v):
|
|
||||||
out[k]=dollar_env_var.sub(rep, dollarcurly_env_var.sub(rep, str(v)))
|
|
||||||
for k,v in out.items():
|
|
||||||
sys.stdout.write("{0}=\"{1}\"\n".format(k, str(v).format_map(out)))
|
|
||||||
'
|
|
||||||
}
|
|
||||||
|
|
||||||
image_hints() {
|
|
||||||
get_env_vars | fgrep '_CACHE_IMAGE_NAME' | awk -F "=" '{print $2}'
|
|
||||||
}
|
|
||||||
|
|
||||||
unset VM_IMAGE_NAME
|
|
||||||
unset VMNAME
|
|
||||||
unset CREATE_CMD
|
|
||||||
unset SSH_CMD
|
|
||||||
unset CLEANUP_CMD
|
|
||||||
declare -xa ENVS
|
|
||||||
parse_args(){
|
|
||||||
local arg
|
|
||||||
echo -e "$USAGE_WARNING"
|
|
||||||
|
|
||||||
if [[ "$USER" =~ "root" ]]
|
|
||||||
then
|
|
||||||
show_usage "This script must be run as a regular user."
|
|
||||||
fi
|
|
||||||
|
|
||||||
[[ "$#" -eq 1 ]] || \
|
|
||||||
show_usage "Must specify a VM Image name to use, and the test flavor."
|
|
||||||
|
|
||||||
VM_IMAGE_NAME="$1"
|
|
||||||
|
|
||||||
# Word-splitting is desirable in this case.
|
|
||||||
# Values are used literally (with '=') as args to future `env` command.
|
|
||||||
# get_env_vars() will take care of properly quoting it's output.
|
|
||||||
# shellcheck disable=SC2207,SC2191
|
|
||||||
ENVS=(
|
|
||||||
$(get_env_vars)
|
|
||||||
VM_IMAGE_NAME="$VM_IMAGE_NAME"
|
|
||||||
UPSTREAM_REMOTE="upstream"
|
|
||||||
)
|
|
||||||
|
|
||||||
VMNAME="${VMNAME:-${USER}-${VM_IMAGE_NAME}}"
|
|
||||||
|
|
||||||
CREATE_CMD="$PGCLOUD compute instances create --zone=$ZONE --image=${VM_IMAGE_NAME} --custom-cpu=$CPUS --custom-memory=$MEMORY --boot-disk-size=$DISK --labels=in-use-by=$USER $VMNAME"
|
|
||||||
|
|
||||||
SSH_CMD="$PGCLOUD compute ssh root@$VMNAME"
|
|
||||||
|
|
||||||
CLEANUP_CMD="$PGCLOUD compute instances delete --zone $ZONE --delete-disks=all $VMNAME"
|
|
||||||
}
|
|
||||||
|
|
||||||
# Returns true if user has run an 'init' and has a valid token for
|
|
||||||
# the specific project-id and named-configuration arguments in $PGCLOUD.
|
|
||||||
function has_valid_credentials() {
|
|
||||||
if $PGCLOUD info |& grep -Eq 'Account:.*None'; then
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# It's possible for 'gcloud info' to list expired credentials,
|
|
||||||
# e.g. 'ERROR: ... invalid grant: Bad Request'
|
|
||||||
if $PGCLOUD auth print-access-token |& grep -q 'ERROR'; then
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
##### main
|
|
||||||
|
|
||||||
[[ "${LIBPODROOT%%${LIBPODROOT##$HOME}}" == "$HOME" ]] || \
|
|
||||||
show_usage "Repo clone must be sub-dir of $HOME"
|
|
||||||
|
|
||||||
cd "$LIBPODROOT"
|
|
||||||
|
|
||||||
parse_args "$@"
|
|
||||||
mkdir -p $TMPDIR/.ssh
|
|
||||||
mkdir -p {$HOME,$TMPDIR}/.config/gcloud/ssh
|
|
||||||
chmod 700 {$HOME,$TMPDIR}/.config/gcloud/ssh $TMPDIR/.ssh
|
|
||||||
|
|
||||||
echo -e "\n${YEL}Pulling gcloud image...${NOR}"
|
|
||||||
podman pull $GCLOUD_IMAGE
|
|
||||||
|
|
||||||
if ! has_valid_credentials
|
|
||||||
then
|
|
||||||
echo -e "\n${YEL}WARNING: Can't find gcloud configuration for libpod, running init.${NOR}"
|
|
||||||
echo -e " ${RED}Please choose \"#1: Re-initialize\" and \"login\" if asked.${NOR}"
|
|
||||||
showrun $PGCLOUD init --project=$PROJECT --console-only --skip-diagnostics
|
|
||||||
|
|
||||||
# Verify it worked (account name == someone@example.com)
|
|
||||||
$PGCLOUD info > $TMPDIR/gcloud-info-after-init
|
|
||||||
if egrep -q "Account:.*None" $TMPDIR/gcloud-info-after-init
|
|
||||||
then
|
|
||||||
echo -e "${RED}ERROR: Could not initialize libpod configuration in gcloud.${NOR}"
|
|
||||||
exit 5
|
|
||||||
fi
|
|
||||||
|
|
||||||
# If this is the only config, make it the default to avoid
|
|
||||||
# persistent warnings from gcloud about there being no default.
|
|
||||||
[[ -r "$HOME/.config/gcloud/configurations/config_default" ]] || \
|
|
||||||
ln "$HOME/.config/gcloud/configurations/config_libpod" \
|
|
||||||
"$HOME/.config/gcloud/configurations/config_default"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
trap delvm EXIT # Allow deleting VM if CTRL-C during create
|
|
||||||
echo -e "\n${YEL}Trying to creating a VM named $VMNAME${NOR}\n${YEL}in GCE region/zone $ZONE${NOR}"
|
|
||||||
echo -e "For faster terminal access, export ZONE='<something-closer>'"
|
|
||||||
echo -e 'Zone-list at: https://cloud.google.com/compute/docs/regions-zones/\n'
|
|
||||||
if showrun $CREATE_CMD; then # Freshly created VM needs initial setup
|
|
||||||
|
|
||||||
echo -e "\n${YEL}Waiting up to 30s for ssh port to open${NOR}"
|
|
||||||
ATTEMPTS=10
|
|
||||||
trap "exit 1" INT
|
|
||||||
while ((ATTEMPTS)) && ! $SSH_CMD --command "true"; do
|
|
||||||
let "ATTEMPTS--"
|
|
||||||
echo -e "${RED}Nope, not yet.${NOR}"
|
|
||||||
sleep 3s
|
|
||||||
done
|
|
||||||
trap - INT
|
|
||||||
if ! ((ATTEMPTS)); then
|
|
||||||
echo -e "\n${RED}Failed${NOR}"
|
|
||||||
exit 7
|
|
||||||
fi
|
|
||||||
echo -e "${YEL}Got it. Cloning upstream repository as a starting point.${NOR}"
|
|
||||||
|
|
||||||
showrun $SSH_CMD -- "mkdir -p $GOSRC"
|
|
||||||
showrun $SSH_CMD -- "git clone --progress $GIT_REPO $GOSRC"
|
|
||||||
|
|
||||||
if [[ -x "$HOME/$GCLOUD_CFGDIR/$HOOK_FILENAME" ]]; then
|
|
||||||
echo -e "\n${YEL}Copying hook to VM and executing (ignoring errors).${NOR}"
|
|
||||||
$PGCLOUD compute scp "/root/$GCLOUD_CFGDIR/$HOOK_FILENAME" root@$VMNAME:.
|
|
||||||
if ! showrun $SSH_CMD -- "cd $GOSRC && bash /root/$HOOK_FILENAME $(git branch --show-current) $(git rev-parse HEAD)"; then
|
|
||||||
echo "-e ${RED}Hook exited: $?${NOR}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo -e "\n${YEL}Generating connection script for $VMNAME.${NOR}"
|
|
||||||
echo -e "Note: Script can be re-used in another terminal if needed."
|
|
||||||
echo -e "${RED}(option to delete VM presented upon exiting).${NOR}"
|
|
||||||
# TODO: This is fairly fragile, specifically the quoting for the remote command.
|
|
||||||
echo '#!/bin/bash' > $TMPDIR/ssh
|
|
||||||
echo "$SSH_CMD -- -t 'cd $GOSRC && exec env ${ENVS[*]} bash -il'" >> $TMPDIR/ssh
|
|
||||||
chmod +x $TMPDIR/ssh
|
|
||||||
|
|
||||||
showrun $TMPDIR/ssh
|
|
||||||
|
Reference in New Issue
Block a user