2bdac9ac3b
Current build process can't handle python2-based projects because we
only build wheels based on Ussury (python3). This patch fixes this.
- build-wheels/docker/stable-wheels-py2.cfg
- build-wheels/docker/dev-wheels-py2.cfg
Resurrected original cfg files from commit
aa31732371
(before upgrade to Ussuri)
and saved with -py2 suffix
- build-wheels/docker/centos-dockerfile
Install python2 packages in addition to python3
- build-wheels/docker/docker-build-wheel.sh
* Build with python3 or python2 depending on the environment
* Suppress status message at the end because its better handled by the
parent script (build-base-wheels.sh)
- build-wheels/build-base-wheels.sh
* Generate 2 wheel directories, "base" and "base-py2" at each
invocation
* Use the same builder image and docker script (with different
environment) to generate wheels based on different cfg files.
Save python3 wheels in "base" and python2 wheels in "base-py2"
- build-wheels/build-wheel-tarball.sh
* New command-line parameter --python2: generate *-wheels-py2.tar
from "base-py2"
* Use ussuri or train constraints depending on the presence of
--python2
- build-docker-images/build-stx-images.sh
* Use *-wheels-py2.tar for python2/loci projects
* Fail early on missing --wheels/--wheels-py2
* Fail if multiple projects use the same LABEL
* Remove support for config files
- build-docker-images/docker-image-build-centos-dev.cfg
- build-docker-images/docker-image-build-centos-stable.cfg
Removed
Change-Id: I2ca444f258a537ed2ba6f68206d32cf59e1802b4
Partial-Bug: 1891416
Signed-off-by: Davlet Panech <davlet.panech@windriver.com>
409 lines
11 KiB
Bash
Executable File
409 lines
11 KiB
Bash
Executable File
#!/bin/bash
|
|
#
|
|
# Copyright (c) 2018-2019 Wind River Systems, Inc.
|
|
#
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
#
|
|
# This utility sets up a docker image to build wheels
|
|
# for a set of upstream python modules.
|
|
#
|
|
|
|
MY_SCRIPT_DIR=$(dirname $(readlink -f $0))
|
|
|
|
source ${MY_SCRIPT_DIR}/utils.sh
|
|
|
|
# Required env vars
|
|
if [ -z "${MY_WORKSPACE}" -o -z "${MY_REPO}" ]; then
|
|
echo "Environment not setup for builds" >&2
|
|
exit 1
|
|
fi
|
|
|
|
DOCKER_PATH=${MY_REPO}/build-tools/build-wheels/docker
|
|
KEEP_IMAGE=no
|
|
KEEP_CONTAINER=no
|
|
OS=centos
|
|
OS_VERSION=7.5.1804
|
|
BUILD_STREAM=stable
|
|
HTTP_PROXY=""
|
|
HTTPS_PROXY=""
|
|
NO_PROXY=""
|
|
declare -i MAX_ATTEMPTS=1
|
|
|
|
function usage {
|
|
cat >&2 <<EOF
|
|
Usage:
|
|
$(basename $0) [ --os <os> ] [ --keep-image ] [ --keep-container ] [ --stream <stable|dev> ]
|
|
|
|
Options:
|
|
--os: Specify base OS (eg. centos)
|
|
--os-version: Specify OS version
|
|
--keep-image: Skip deletion of the wheel build image in docker
|
|
--keep-container: Skip deletion of container used for the build
|
|
--http_proxy: Set http proxy <URL>:<PORT>, urls splitted by ","
|
|
--https_proxy: Set https proxy <URL>:<PORT>, urls splitted by ","
|
|
--no_proxy: Set bypass list for proxy <URL>, urls splitted by ","
|
|
--stream: Build stream, stable or dev (default: stable)
|
|
--attempts: Max attempts, in case of failure (default: 1)
|
|
|
|
EOF
|
|
}
|
|
|
|
OPTS=$(getopt -o h -l help,os:,os-version:,keep-image,keep-container,release:,stream:,http_proxy:,https_proxy:,no_proxy:,attempts: -- "$@")
|
|
if [ $? -ne 0 ]; then
|
|
usage
|
|
exit 1
|
|
fi
|
|
|
|
eval set -- "${OPTS}"
|
|
|
|
while true; do
|
|
case $1 in
|
|
--)
|
|
# End of getopt arguments
|
|
shift
|
|
break
|
|
;;
|
|
--os)
|
|
OS=$2
|
|
shift 2
|
|
;;
|
|
--os-version)
|
|
OS_VERSION=$2
|
|
shift 2
|
|
;;
|
|
--keep-image)
|
|
KEEP_IMAGE=yes
|
|
shift
|
|
;;
|
|
--keep-container)
|
|
KEEP_CONTAINER=yes
|
|
shift
|
|
;;
|
|
--stream)
|
|
BUILD_STREAM=$2
|
|
shift 2
|
|
;;
|
|
--release) # Temporarily keep --release support as an alias for --stream
|
|
BUILD_STREAM=$2
|
|
shift 2
|
|
;;
|
|
--http_proxy)
|
|
HTTP_PROXY=$2
|
|
shift 2
|
|
;;
|
|
--https_proxy)
|
|
HTTPS_PROXY=$2
|
|
shift 2
|
|
;;
|
|
--no_proxy)
|
|
NO_PROXY=$2
|
|
shift 2
|
|
;;
|
|
--attempts)
|
|
MAX_ATTEMPTS=$2
|
|
shift 2
|
|
;;
|
|
-h | --help )
|
|
usage
|
|
exit 1
|
|
;;
|
|
*)
|
|
usage
|
|
exit 1
|
|
;;
|
|
esac
|
|
done
|
|
|
|
BUILD_IMAGE_NAME="${USER}-$(basename ${MY_WORKSPACE})-wheelbuilder:${OS}-${BUILD_STREAM}"
|
|
|
|
# BUILD_IMAGE_NAME can't have caps if it's passed to docker build -t $BUILD_IMAGE_NAME.
|
|
# The following will substitute caps with lower case.
|
|
BUILD_IMAGE_NAME="${BUILD_IMAGE_NAME,,}"
|
|
|
|
DOCKER_FILE=${DOCKER_PATH}/${OS}-dockerfile
|
|
|
|
function supported_os_list {
|
|
for f in ${DOCKER_PATH}/*-dockerfile; do
|
|
echo $(basename ${f%-dockerfile})
|
|
done | xargs echo
|
|
}
|
|
|
|
if [ ! -f ${DOCKER_FILE} ]; then
|
|
echo "Unsupported OS specified: ${OS}" >&2
|
|
echo "Supported OS options: $(supported_os_list)" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Print a loud message
|
|
function notice {
|
|
(
|
|
set +x
|
|
echo
|
|
echo ======================================
|
|
for s in "$@" ; do
|
|
echo "$s"
|
|
done
|
|
echo ======================================
|
|
echo
|
|
) 2>&1
|
|
}
|
|
|
|
# prefix each line of a command's output
|
|
# also redirects command's STDERR to STDOUT
|
|
log_prefix() {
|
|
local prefix="$1" ; shift
|
|
"$@" 2>&1 | awk -v prefix="$prefix" '{print prefix $0}'
|
|
# return false if the command (rather than awk) failed
|
|
[ ${PIPESTATUS[0]} -eq 0 ]
|
|
}
|
|
|
|
|
|
# Make sure a file exists, exit otherwise
|
|
function require_file {
|
|
if [ ! -f "${1}" ]; then
|
|
echo "Required file does not exist: ${1}" >&2
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# Check build output directory for unexpected files,
|
|
# ie. wheels from old builds that are no longer in wheels.cfg
|
|
function prepare_output_dir {
|
|
local output_dir="$1"
|
|
local wheels_cfg="$2"
|
|
if [ -d ${output_dir} ]; then
|
|
local f
|
|
for f in ${output_dir}/*; do
|
|
if [ -f $f ] ; then
|
|
grep -q "^$(basename $f)|" ${wheels_cfg}
|
|
if [ $? -ne 0 ]; then
|
|
echo "Deleting stale file: $f"
|
|
rm -f $f
|
|
fi
|
|
fi
|
|
done
|
|
else
|
|
mkdir -p ${output_dir}
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to create directory: ${output_dir}" >&2
|
|
exit 1
|
|
fi
|
|
fi
|
|
}
|
|
|
|
BUILD_OUTPUT_PATH=${MY_WORKSPACE}/std/build-wheels-${OS}-${BUILD_STREAM}/base
|
|
BUILD_OUTPUT_PATH_PY2=${MY_WORKSPACE}/std/build-wheels-${OS}-${BUILD_STREAM}/base-py2
|
|
WHEELS_CFG=${DOCKER_PATH}/${BUILD_STREAM}-wheels.cfg
|
|
WHEELS_CFG_PY2=${DOCKER_PATH}/${BUILD_STREAM}-wheels-py2.cfg
|
|
|
|
# make sure .cfg files exist
|
|
require_file "${WHEELS_CFG}"
|
|
require_file "${WHEELS_CFG_PY2}"
|
|
|
|
# prepare output directories
|
|
prepare_output_dir "${BUILD_OUTPUT_PATH}" "${WHEELS_CFG}"
|
|
prepare_output_dir "${BUILD_OUTPUT_PATH_PY2}" "${WHEELS_CFG_PY2}"
|
|
|
|
if [ "${BUILD_STREAM}" = "dev" -o "${BUILD_STREAM}" = "master" ]; then
|
|
# Download the master wheel from loci, so we're only building pieces not covered by it
|
|
MASTER_WHEELS_IMAGE="loci/requirements:master-${OS}"
|
|
|
|
# Check to see if the wheels are already present.
|
|
# If so, we'll still pull to ensure the image is updated,
|
|
# but we won't delete it after
|
|
docker images --format '{{.Repository}}:{{.Tag}}' ${MASTER_WHEELS_IMAGE} | grep -q "^${MASTER_WHEELS_IMAGE}$"
|
|
MASTER_WHEELS_PRESENT=$?
|
|
|
|
docker pull ${MASTER_WHEELS_IMAGE}
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to pull ${MASTER_WHEELS_IMAGE}" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Export the image to a tarball.
|
|
# The "docker run" will always fail, due to the construct of the wheels image,
|
|
# so just ignore it
|
|
docker run --name ${USER}_inspect_wheels ${MASTER_WHEELS_IMAGE} noop 2>/dev/null
|
|
|
|
echo "Extracting wheels from ${MASTER_WHEELS_IMAGE}"
|
|
rm -rf "${BUILD_OUTPUT_PATH}-loci"
|
|
mkdir -p "$BUILD_OUTPUT_PATH-loci"
|
|
docker export ${USER}_inspect_wheels | tar x -C "${BUILD_OUTPUT_PATH}-loci" '*.whl'
|
|
if [ ${PIPESTATUS[0]} -ne 0 -o ${PIPESTATUS[1]} -ne 0 ]; then
|
|
echo "Failed to extract wheels from ${MASTER_WHEELS_IMAGE}" >&2
|
|
docker rm ${USER}_inspect_wheels
|
|
if [ ${MASTER_WHEELS_PRESENT} -ne 0 ]; then
|
|
docker image rm ${MASTER_WHEELS_IMAGE}
|
|
fi
|
|
rm -rf "${BUILD_OUTPUT_PATH}-loci"
|
|
exit 1
|
|
fi
|
|
|
|
# copy loci wheels in base and base-py2 directories
|
|
if ! cp "${BUILD_OUTPUT_PATH}-loci"/*.whl "${BUILD_OUTPUT_PATH}"/ ; then
|
|
echo "Failed to copy wheels to ${BUILD_OPUTPUT_PATH}" >&2
|
|
exit 1
|
|
fi
|
|
if ! cp "${BUILD_OUTPUT_PATH}-loci"/*.whl "${BUILD_OUTPUT_PATH_PY2}"/ ; then
|
|
echo "Failed to copy wheels to ${BUILD_OPUTPUT_PATH_PY2}" >&2
|
|
exit 1
|
|
fi
|
|
rm -rf "${BUILD_OUTPUT_PATH}-loci"
|
|
|
|
docker rm ${USER}_inspect_wheels
|
|
|
|
if [ ${MASTER_WHEELS_PRESENT} -ne 0 ]; then
|
|
docker image rm ${MASTER_WHEELS_IMAGE}
|
|
fi
|
|
fi
|
|
|
|
# check if there are any wheels missing
|
|
function all_wheels_exist {
|
|
local output_dir="$1"
|
|
local wheels_cfg="$2"
|
|
local wheel
|
|
for wheel in $(cat "${wheels_cfg}" | sed 's/#.*//' | awk -F '|' '{print $1}'); do
|
|
if [[ "${wheel}" =~ \* || ! -f ${output_dir}/${wheel} ]]; then
|
|
return 1
|
|
fi
|
|
done
|
|
return 0
|
|
}
|
|
|
|
if all_wheels_exist "${BUILD_OUTPUT_PATH}" "${WHEELS_CFG}" && \
|
|
all_wheels_exist "${BUILD_OUTPUT_PATH_PY2}" "${WHEELS_CFG_PY2}" ; then
|
|
echo "All base wheels are already present. Skipping build."
|
|
exit 0
|
|
fi
|
|
|
|
# Check to see if the OS image is already pulled
|
|
docker images --format '{{.Repository}}:{{.Tag}}' ${OS}:${OS_VERSION} | grep -q "^${OS}:${OS_VERSION}$"
|
|
BASE_IMAGE_PRESENT=$?
|
|
|
|
# Create the builder image
|
|
declare -a BUILD_ARGS
|
|
BUILD_ARGS+=(--build-arg RELEASE=${OS_VERSION})
|
|
BUILD_ARGS+=(--build-arg BUILD_STREAM=${BUILD_STREAM})
|
|
if [ ! -z "$HTTP_PROXY" ]; then
|
|
BUILD_ARGS+=(--build-arg http_proxy=$HTTP_PROXY)
|
|
fi
|
|
|
|
if [ ! -z "$HTTPS_PROXY" ]; then
|
|
BUILD_ARGS+=(--build-arg https_proxy=$HTTPS_PROXY)
|
|
fi
|
|
|
|
if [ ! -z "$NO_PROXY" ]; then
|
|
BUILD_ARGS+=(--build-arg no_proxy=$NO_PROXY)
|
|
fi
|
|
|
|
BUILD_ARGS+=(-t ${BUILD_IMAGE_NAME})
|
|
BUILD_ARGS+=(-f ${DOCKER_PATH}/${OS}-dockerfile ${DOCKER_PATH})
|
|
|
|
# Build image
|
|
with_retries ${MAX_ATTEMPTS} docker build "${BUILD_ARGS[@]}"
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to create build image in docker" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Run the image, executing the build-wheel.sh script
|
|
declare -a RUN_ARGS
|
|
if [ "${KEEP_CONTAINER}" = "no" ]; then
|
|
RUN_ARGS+=(--rm)
|
|
fi
|
|
if [ ! -z "$HTTP_PROXY" ]; then
|
|
RUN_ARGS+=(--env http_proxy=$HTTP_PROXY)
|
|
fi
|
|
if [ ! -z "$HTTPS_PROXY" ]; then
|
|
RUN_ARGS+=(--env https_proxy=$HTTPS_PROXY)
|
|
fi
|
|
if [ ! -z "$NO_PROXY" ]; then
|
|
RUN_ARGS+=(--env no_proxy=$NO_PROXY)
|
|
fi
|
|
RUN_ARGS+=(--env DISPLAY_RESULT=no)
|
|
|
|
# Run container to build wheels
|
|
rm -f ${BUILD_OUTPUT_PATH}/failed.lst
|
|
rm -f ${BUILD_OUTPUT_PATH_PY2}/failed.lst
|
|
|
|
notice "building python3 wheels"
|
|
log_prefix "[python3] " \
|
|
with_retries ${MAX_ATTEMPTS} \
|
|
docker run ${RUN_ARGS[@]} -v ${BUILD_OUTPUT_PATH}:/wheels ${BUILD_IMAGE_NAME} /docker-build-wheel.sh
|
|
BUILD_STATUS=$?
|
|
|
|
notice "building python2 wheels"
|
|
log_prefix "[python2] " \
|
|
with_retries ${MAX_ATTEMPTS} \
|
|
docker run ${RUN_ARGS[@]} -v ${BUILD_OUTPUT_PATH_PY2}:/wheels --env PYTHON=python2 ${BUILD_IMAGE_NAME} /docker-build-wheel.sh
|
|
BUILD_STATUS_PY2=$?
|
|
|
|
if [ "${KEEP_IMAGE}" = "no" ]; then
|
|
# Delete the builder image
|
|
echo "Removing docker image ${BUILD_IMAGE_NAME}"
|
|
docker image rm ${BUILD_IMAGE_NAME}
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to delete build image from docker" >&2
|
|
fi
|
|
|
|
if [ ${BASE_IMAGE_PRESENT} -ne 0 ]; then
|
|
# The base image was not already present, so delete it
|
|
echo "Removing docker image ${OS}:${OS_VERSION}"
|
|
docker image rm ${OS}:${OS_VERSION}
|
|
if [ $? -ne 0 ]; then
|
|
echo "Failed to delete base image from docker" >&2
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
# Check for failures
|
|
check_result() {
|
|
local python="$1"
|
|
local status="$2"
|
|
local dir="$3"
|
|
|
|
# There's a failed images list
|
|
if [ -f "${dir}/failed.lst" ]; then
|
|
let failures=$(cat "${dir}/failed.lst" | wc -l)
|
|
|
|
cat <<EOF
|
|
|
|
############################################################
|
|
The following ${python} module(s) failed to build:
|
|
$(cat ${dir}/failed.lst)
|
|
|
|
Summary:
|
|
${failures} build failure(s).
|
|
|
|
EOF
|
|
return 1
|
|
fi
|
|
|
|
# No failed images list, but build script failed nonetheless
|
|
if [ "${status}" != 0 ] ; then
|
|
cat <<EOF
|
|
|
|
############################################################
|
|
Build script failed for ${python}
|
|
|
|
EOF
|
|
return 1
|
|
fi
|
|
|
|
# success
|
|
cat <<EOF
|
|
|
|
############################################################
|
|
All ${python} wheels have been successfully built.
|
|
|
|
EOF
|
|
return 0
|
|
}
|
|
|
|
if ! check_result "python3" "${BUILD_STATUS}" "${BUILD_OUTPUT_PATH}" || \
|
|
! check_result "python2" "${BUILD_STATUS_PY2}" "${BUILD_OUTPUT_PATH_PY3}" ; then
|
|
exit 1
|
|
fi
|
|
exit 0
|
|
|