05b961dec7
- put variable expansions into quotes where possible; - fix couple of minor issues. Change-Id: I56576250f428aee75c3e8d7154b287a080cb596c
390 lines
10 KiB
Bash
Executable File
390 lines
10 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
locale -a | grep -q -e "^en_US$" && export LANG="en_US"
|
|
|
|
shopt -s nocasematch
|
|
|
|
SMITHY_NAME=$(readlink -f "$0")
|
|
cd "$(dirname "$0")"
|
|
|
|
VERBOSE="${VERBOSE:-0}"
|
|
PY2RPM_CMD="$PWD/tools/py2rpm"
|
|
YYOOM_CMD="$PWD/tools/yyoom"
|
|
PIPDOWNLOAD_CMD="$PWD/tools/pip-download"
|
|
|
|
YUM_OPTS="--assumeyes --nogpgcheck"
|
|
YYOOM_OPTS="--verbose"
|
|
RPM_OPTS=""
|
|
CURL_OPTS=""
|
|
|
|
# Colors supported??
|
|
COLOR_SUPPORT=$(tput colors)
|
|
|
|
if [ "$COLOR_SUPPORT" -ge 8 ]; then
|
|
ESC_SEQ="\x1b["
|
|
COL_RESET=$ESC_SEQ"39;49;00m"
|
|
COL_GREEN=$ESC_SEQ"32;01m"
|
|
COL_RED=$ESC_SEQ"31;01m"
|
|
COL_YELLOW=$ESC_SEQ"33;01m"
|
|
else
|
|
ESC_SEQ=""
|
|
COL_RESET=""
|
|
COL_GREEN=""
|
|
COL_RED=""
|
|
COL_YELLOW=""
|
|
fi
|
|
|
|
if [ "$VERBOSE" == "0" ]; then
|
|
YUM_OPTS="$YUM_OPTS -q"
|
|
YYOOM_OPTS=""
|
|
RPM_OPTS="-q"
|
|
CURL_OPTS="-s"
|
|
fi
|
|
|
|
# Source in our variables (or overrides)
|
|
source ".anvilrc"
|
|
if [ -n "$SUDO_USER" ]; then
|
|
USER_HOME=$(getent passwd "$SUDO_USER" | cut -d: -f6)
|
|
if [ -n "$USER_HOME" ]; then
|
|
HOME_RC="${USER_HOME}/.anvilrc"
|
|
if [ -f "$HOME_RC" ]; then
|
|
source "$HOME_RC"
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
if [ -z "$BOOT_FILES" ]; then
|
|
BOOT_FN=".anvil_bootstrapped"
|
|
BOOT_FILES="${PWD}/$BOOT_FN"
|
|
fi
|
|
|
|
clean_pip()
|
|
{
|
|
# https://github.com/pypa/pip/issues/982
|
|
if [ -n "$SUDO_USER" ]; then
|
|
rm -rf "/tmp/pip-build-$SUDO_USER"
|
|
fi
|
|
}
|
|
|
|
rpm_is_installed()
|
|
{
|
|
local name=$(basename "$1")
|
|
rpm $RPM_OPTS "${name%.rpm}" &>/dev/null
|
|
return $?
|
|
}
|
|
|
|
cache_and_install_rpm_url()
|
|
{
|
|
url=${1:?"Error: rpm uri is undefined!"}
|
|
cachedir=${RPM_CACHEDIR:-'/tmp'}
|
|
rpm=$(basename "$url")
|
|
if rpm_is_installed "$rpm"; then
|
|
return
|
|
fi
|
|
if [ ! -f "$cachedir/$rpm" ]; then
|
|
echo -e "Downloading ${COL_GREEN}${rpm}${COL_RESET} to ${COL_GREEN}${cachedir}${COL_RESET}"
|
|
curl $CURL_OPTS "$url" -o "$cachedir/$rpm" || return 1
|
|
fi
|
|
echo -e "Installing ${COL_GREEN}$cachedir/$rpm${COL_RESET}"
|
|
yum install $YUM_OPTS "$cachedir/$rpm"
|
|
return $?
|
|
}
|
|
|
|
bootstrap_epel()
|
|
{
|
|
[ -z "$EPEL_RPM_URL" ] && return 0
|
|
cache_and_install_rpm_url "$EPEL_RPM_URL"
|
|
return $?
|
|
}
|
|
|
|
dump_list()
|
|
{
|
|
for var in "$@"; do
|
|
for name in $var; do
|
|
echo " - $name"
|
|
done
|
|
done
|
|
}
|
|
|
|
greatest_version()
|
|
{
|
|
for arg in "$@"; do
|
|
echo "$arg"
|
|
done | sort --version-sort --reverse | head -n1
|
|
}
|
|
|
|
bootstrap_rpm_packages()
|
|
{
|
|
# NOTE(aababilov): the latter operations require some packages,
|
|
# so, begin from installation
|
|
if [ -n "$REQUIRES" ]; then
|
|
echo -e "Installing ${COL_GREEN}system${COL_RESET} packages:"
|
|
dump_list $REQUIRES
|
|
yum install $YUM_OPTS $REQUIRES
|
|
if [ "$?" != "0" ]; then
|
|
echo -e "${COL_RED}Failed installing!${COL_RESET}"
|
|
return 1
|
|
fi
|
|
fi
|
|
|
|
# Remove any known conflicting packages
|
|
CONFLICTS=$(python -c "import yaml
|
|
packages = set()
|
|
try:
|
|
for i in yaml.safe_load(open('$DISTRO_CONFIG'))['components'].itervalues():
|
|
for j in i.get('conflicts', []):
|
|
packages.add(j.get('name'))
|
|
except KeyError:
|
|
pass
|
|
for pkg in packages:
|
|
if pkg:
|
|
print pkg
|
|
")
|
|
if [ -n "$CONFLICTS" ]; then
|
|
echo -e "Removing ${COL_YELLOW}conflicting${COL_RESET} packages:"
|
|
dump_list "$CONFLICTS"
|
|
yum erase $YUM_OPTS $CONFLICTS
|
|
fi
|
|
}
|
|
|
|
bootstrap_selinux()
|
|
{
|
|
# See if selinux is on.
|
|
if [ "$(getenforce)" == "Enforcing" ]; then
|
|
# Ensure all yum api interacting binaries are ok to be used
|
|
echo "Enabling selinux for yum like binaries."
|
|
chcon -h "system_u:object_r:rpm_exec_t:s0" "$YYOOM_CMD"
|
|
fi
|
|
}
|
|
|
|
bootstrap_python_rpms()
|
|
{
|
|
echo -e "Bootstrapping ${COL_GREEN}python${COL_RESET} rpms."
|
|
local package_map=$(python -c "import yaml
|
|
try:
|
|
for k, v in yaml.safe_load(open('$DISTRO_CONFIG'))['dependency_handler']['package_map'].iteritems():
|
|
print '%s==%s' % (k, v)
|
|
except KeyError:
|
|
pass
|
|
")
|
|
local python_names=$(sed -r -e 's/#.*$//' requirements.txt | sort -u)
|
|
local bootstrap_dir=$(readlink -f ./.bootstrap/)
|
|
local transaction_cmd="transaction --skip-missing"
|
|
local install_packages=""
|
|
declare -A rpm_python_map
|
|
for python_name in $python_names; do
|
|
local specs=$(echo "$python_name" | awk 'match($0, "((=|>|<|!).*$)", res) {print res[1]}')
|
|
local rpm_name=$("$PY2RPM_CMD" --package-map $package_map --convert "$python_name" |
|
|
awk '/^Requires:/ {print $2; exit }')
|
|
rpm_python_map["$rpm_name"]="$python_name"
|
|
install_packages="$install_packages $rpm_name$specs"
|
|
transaction_cmd+=" --install $rpm_name$specs"
|
|
done
|
|
|
|
echo -e "Installing ${COL_GREEN}python${COL_RESET} requirements:"
|
|
dump_list "$install_packages"
|
|
|
|
# NOTE(imelnikov): if we declare local variable and specify its value
|
|
# at the same statement, exit code from subshell is lost.
|
|
local yyoom_res
|
|
yyoom_res=$("$YYOOM_CMD" $YYOOM_OPTS $transaction_cmd) || return 1
|
|
local missing_rpms
|
|
missing_rpms=$(echo "$yyoom_res" | python -c "import sys, json
|
|
for item in json.load(sys.stdin):
|
|
if item.get('action_type') == 'missing':
|
|
print(item['name'])
|
|
") || return 2
|
|
local missing_packages=""
|
|
for rpm in $missing_rpms; do
|
|
missing_packages="$missing_packages ${rpm_python_map[$rpm]}"
|
|
done
|
|
|
|
if [ -z "$missing_packages" ]; then
|
|
return 0
|
|
fi
|
|
echo -e "Building ${COL_YELLOW}missing${COL_RESET} python requirements:"
|
|
dump_list "$missing_packages"
|
|
local pip_tmp_dir="$bootstrap_dir/pip-download"
|
|
mkdir -p "$pip_tmp_dir"
|
|
|
|
$PIPDOWNLOAD_CMD -d "$pip_tmp_dir" $missing_packages
|
|
echo "Building RPMs..."
|
|
local rpm_names
|
|
rpm_names=$("$PY2RPM_CMD" --package-map $package_map --scripts-dir "conf/templates/packaging/scripts" --rpm-base "$bootstrap_dir/rpmbuild" -- "$pip_tmp_dir/"* 2>/dev/null |
|
|
awk '/^Wrote: /{ print $2 }' | grep -v '.src.rpm' | sort -u)
|
|
if [ -z "$rpm_names" ]; then
|
|
echo -e "${COL_RED}No binary RPMs were built!${COL_RESET}"
|
|
return 1
|
|
fi
|
|
local rpm_base_names=""
|
|
for rpm in $rpm_names; do
|
|
rpm_base_names="$rpm_base_names $(basename "$rpm")"
|
|
done
|
|
echo -e "Installing ${COL_YELLOW}missing${COL_RESET} python requirement packages:"
|
|
dump_list "$rpm_base_names"
|
|
yum install $YUM_OPTS $rpm_names
|
|
if [ "$?" != "0" ]; then
|
|
echo -e "${COL_RED}Failed installing!${COL_RESET}"
|
|
return 1
|
|
fi
|
|
rm -rf "$pip_tmp_dir"
|
|
rm -rf "$bootstrap_dir/rpmbuild/"{BUILD,SOURCES,SPECS,BUILDROOT}
|
|
return 0
|
|
}
|
|
|
|
needs_bootstrap()
|
|
{
|
|
local contents checksums
|
|
$BOOTSTRAP && return 0
|
|
checksums=$(get_checksums)
|
|
for i in $BOOT_FILES; do
|
|
if [ -f "$i" ]; then
|
|
contents=$(cat $i)
|
|
[ "$contents" = "$checksums" ] && return 1
|
|
fi
|
|
done
|
|
return 0
|
|
}
|
|
|
|
get_checksums()
|
|
{
|
|
if [ ! -f "$BSCONF_FILE" ]; then
|
|
return 1
|
|
fi
|
|
# Used to tell if the file have changed
|
|
echo $(md5sum "$BSCONF_FILE")
|
|
}
|
|
|
|
run_smithy()
|
|
{
|
|
PYTHON=$(which python)
|
|
exec "$PYTHON" anvil $ARGS
|
|
}
|
|
|
|
puke()
|
|
{
|
|
cleaned_force=$(echo "$FORCE" | sed -e 's/\([A-Z]\)/\L\1/g;s/\s//g')
|
|
if [[ "$cleaned_force" == "yes" ]]; then
|
|
run_smithy
|
|
else
|
|
echo -e "To run anyway set ${COL_YELLOW}FORCE=yes${COL_RESET} and rerun." >&2
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
## Identify which bootstrap configuration file to use: either set
|
|
## explicitly (BSCONF_FILE) or determined based on the os distribution:
|
|
BSCONF_DIR="${BSCONF_DIR:-$(dirname $(readlink -f "$0"))/tools/bootstrap}"
|
|
get_os_info(){
|
|
if [ "$(uname)" = "Linux" ] ; then
|
|
if [ -f /etc/redhat-release ] ; then
|
|
PKG="rpm"
|
|
OSNAME=`cat /etc/redhat-release`
|
|
OSDIST=`cat /etc/redhat-release | sed -e 's/release.*$//g;s/\s//g'`
|
|
PSUEDONAME=`cat /etc/redhat-release | sed s/.*\(// | sed s/\)//`
|
|
RELEASE=`cat /etc/redhat-release | sed s/.*release\ // | sed s/\ .*//`
|
|
elif [ -f /etc/debian_version ] ; then
|
|
PKG="deb"
|
|
OSDIST=`cat /etc/lsb-release | grep '^DISTRIB_ID' | awk -F= '{ print $2 }'`
|
|
PSUEDONAME=`cat /etc/lsb-release | grep '^DISTRIB_CODENAME' | awk -F= '{ print $2 }'`
|
|
RELEASE=`cat /etc/lsb-release | grep '^DISTRIB_RELEASE' | awk -F= '{ print $2 }'`
|
|
OSNAME="$OSDIST $RELEASE ($PSUEDONAME)"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
get_os_info
|
|
|
|
if [ -z "$BSCONF_FILE" ]; then
|
|
BSCONF_FILE="$BSCONF_DIR/$OSDIST"
|
|
fi
|
|
|
|
ARGS=""
|
|
BOOTSTRAP=false
|
|
|
|
# Ad-hoc getopt to handle long opts.
|
|
#
|
|
# Smithy opts are consumed while those to anvil are copied through.
|
|
while [ "$#" != 0 ]; do
|
|
case "$1" in
|
|
'--bootstrap')
|
|
BOOTSTRAP=true
|
|
shift
|
|
;;
|
|
'--force')
|
|
FORCE=yes
|
|
shift
|
|
;;
|
|
*)
|
|
ARGS="$ARGS $1"
|
|
shift
|
|
;;
|
|
esac
|
|
done
|
|
|
|
# Source immediately so that we can export the needed variables.
|
|
if [ -f "$BSCONF_FILE" ]; then
|
|
source "$BSCONF_FILE"
|
|
export REQUIRED_PACKAGES="$REQUIRES"
|
|
fi
|
|
|
|
if ! needs_bootstrap; then
|
|
clean_pip
|
|
run_smithy
|
|
clean_pip
|
|
elif ! $BOOTSTRAP; then
|
|
echo "This system needs to be updated in order to run anvil!" >&2
|
|
echo "Running 'sudo $SMITHY_NAME --bootstrap' will attempt to do so." >&2
|
|
puke
|
|
fi
|
|
|
|
## Bootstrap smithy
|
|
if [ "$(id -u)" != "0" ]; then
|
|
echo "You must run '$SMITHY_NAME --bootstrap' with root privileges!" >&2
|
|
exit 1
|
|
fi
|
|
|
|
if [ ! -f "$BSCONF_FILE" ]; then
|
|
echo "Anvil has not been tested on distribution '$OSNAME'" >&2
|
|
puke
|
|
fi
|
|
|
|
MIN_RELEASE=${MIN_RELEASE:?"Error: MIN_RELEASE is undefined!"}
|
|
SHORTNAME=${SHORTNAME:?"Error: SHORTNAME is undefined!"}
|
|
if [ "$RELEASE" != "$(greatest_version "$RELEASE" "$MIN_RELEASE")" ]; then
|
|
echo "This script must be run on $SHORTNAME $MIN_RELEASE+ and not $SHORTNAME $RELEASE." >&2
|
|
puke
|
|
fi
|
|
|
|
echo "Bootstrapping $SHORTNAME $RELEASE"
|
|
echo "Please wait..."
|
|
clean_pip
|
|
for step in ${STEPS:?"Error: STEPS is undefined!"}; do
|
|
"bootstrap_${step}"
|
|
if [ $? != 0 ]; then
|
|
echo "Bootstrapping $SHORTNAME $RELEASE failed." >&2
|
|
exit 1
|
|
fi
|
|
done
|
|
clean_pip
|
|
|
|
# Write the checksums of the bootstrap file
|
|
# which if new requirements are added will cause new checksums
|
|
# and a new dependency install...
|
|
checksum=$(get_checksums)
|
|
for i in $BOOT_FILES; do
|
|
echo -e "$checksum" > "$i"
|
|
done
|
|
|
|
mkdir -p -v /etc/anvil /usr/share/anvil
|
|
touch /var/log/anvil.log
|
|
if [ -n "$SUDO_UID" -a -n "SUDO_GID" ]; then
|
|
chown -c "$SUDO_UID:$SUDO_GID" /etc/anvil /usr/share/anvil \
|
|
/var/log/anvil.log
|
|
[ -d .bootstrap ] && chown -R "$SUDO_UID:$SUDO_GID" .bootstrap
|
|
fi
|
|
|
|
echo "Bootstrapped for $SHORTNAME $RELEASE"
|
|
ARGS="-a moo"
|
|
run_smithy
|