eb971b6f7c
Function generate_dep_list is creating a DEPLISTFILE with a lot of unwanted 'installing package xxx' messages when it should only contain a list of unresolved dependencies. On the next iteration we waste a lot of time trying to install nonexistant packages 'installing' and 'packages', and re-installing packages we already have. Instead of converging on an empty DEPLISTFILE, we converge on a DEPLISTFILE that just reiterates the same list of 'installing package' if we are lucky, which we usually are, but not always. Closes-Bug: 1877483 Change-Id: I971725558eb628c6d6b5d0b42d0bcab6b46e722b Signed-off-by: Scott Little <scott.little@windriver.com>
296 lines
11 KiB
Bash
Executable File
296 lines
11 KiB
Bash
Executable File
#!/bin/env bash
|
|
|
|
# Here's the score, kids. There are a few different places from which we can
|
|
# get packages. In priority order, they are:
|
|
#
|
|
# The CGTS packages we've built ourselves
|
|
# The CGTS packages that Jenkins has built (coming soon to a script near you)
|
|
# The CentOS packages in various repos
|
|
# - Base OS
|
|
# - OpenStack Repos
|
|
# EPEL (Extra Packages for Enterprise Linux)
|
|
#
|
|
# This script can function in two ways:
|
|
# If you specify a filename, it assumes the file is a list of packages you
|
|
# want to install, or dependencies you want to meet. It installs whatever
|
|
# is in the list into current directory. Failure to find a dependency
|
|
# results in a return code of 1
|
|
#
|
|
# If no file is specified, we generate a file ($DEPLISTFILE) of dependencies
|
|
# based on current directory
|
|
#
|
|
# We then continuously loop through generating new dependencies and installing
|
|
# them until either all dependencies are met, or we cannot install anymore
|
|
#
|
|
# We also log where dependencies were installed from into
|
|
# export/dist/report_deps.txt
|
|
#
|
|
|
|
# This function generates a simple file of dependencies we're trying to resolve
|
|
function generate_dep_list {
|
|
TMP_RPM_DB=$(mktemp -d $(pwd)/tmp_rpm_db_XXXXXX)
|
|
mkdir -p $TMP_RPM_DB
|
|
rpm --initdb --dbpath $TMP_RPM_DB
|
|
rpm --dbpath $TMP_RPM_DB --test -Uvh --replacefiles '*.rpm' >> $DEPDETAILLISTFILE 2>&1
|
|
rpm --dbpath $TMP_RPM_DB --test -Uvh --replacefiles '*.rpm' 2>&1 \
|
|
| grep -v "error:" \
|
|
| grep -v "warning:" \
|
|
| grep -v "Preparing..." \
|
|
| grep -v "installing package" \
|
|
| sed "s/ is needed by.*$//" | sed "s/ >=.*$//" | sort -u > $DEPLISTFILE
|
|
\rm -rf $TMP_RPM_DB
|
|
}
|
|
|
|
# Takes a list of requirements (either explcit package name, or capabilities
|
|
# to provide) and install packages to meet those dependancies
|
|
#
|
|
# We take the list of requirements and first try to look them up based on
|
|
# package name. If we can't find a package with the name of the requirement,
|
|
# we use --whatprovides to complete the lookup.
|
|
#
|
|
# The reason for this initial name-based attempt is that a couple of funky
|
|
# packages (notably -devel packages) have "Provides:" capabilities which
|
|
# conflict with named packages. So if explictly say we want "xyz" then we'll
|
|
# install the "xyz" package, rather than "something-devel" which has "xyz"
|
|
# capabilities.
|
|
function install_deps {
|
|
local DEP_LIST=""
|
|
local DEP_LIST_FILE="$1"
|
|
|
|
# Temporary files are used in a few different ways
|
|
# Here we essenitally create variable aliases to make it easier to read
|
|
# the script
|
|
local UNSORTED_PACKAGES=$TMPFILE
|
|
local SORTED_PACKAGES=$TMPFILE1
|
|
local UNRESOLVED_PACKAGES=$TMPFILE2
|
|
|
|
rm -f $UNSORTED_PACKAGES
|
|
|
|
while read DEP
|
|
do
|
|
DEP_LIST="${DEP_LIST} ${DEP}"
|
|
done < $DEP_LIST_FILE
|
|
|
|
echo "Debug: List of deps to resolve: ${DEP_LIST}"
|
|
|
|
if [ -z "${DEP_LIST}" ]; then
|
|
return 0
|
|
fi
|
|
|
|
# go through each repo and convert deps to packages based on package name
|
|
for REPOID in `grep '^[[].*[]]$' $YUM | grep -v '[[]main[]]' | awk -F '[][]' '{print $2 }'`; do
|
|
echo "TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch ${DEP_LIST} --qf='%{name}'"
|
|
TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --qf='%{name}' ${DEP_LIST} | sed "s/kernel-debug/kernel/g" >> $UNSORTED_PACKAGES
|
|
\rm -rf $TMP_DIR/yum-$USER-*
|
|
done
|
|
sort $UNSORTED_PACKAGES -u > $SORTED_PACKAGES
|
|
|
|
# figure out any dependancies which could not be resolved based on
|
|
# package name. We use --whatpovides to deal with this
|
|
#
|
|
# First, we build a new DEP_LIST based on what was NOT found in
|
|
# search-by-name attempt
|
|
sort $DEP_LIST_FILE -u > $TMPFILE
|
|
comm -2 -3 $TMPFILE $SORTED_PACKAGES > $UNRESOLVED_PACKAGES
|
|
|
|
# If there are any requirements not resolved, look up the packages with
|
|
# --whatprovides
|
|
if [ -s $UNRESOLVED_PACKAGES ]; then
|
|
DEP_LIST=""
|
|
\cp $SORTED_PACKAGES $UNSORTED_PACKAGES
|
|
while read DEP
|
|
do
|
|
DEP_LIST="${DEP_LIST} ${DEP}"
|
|
done < $UNRESOLVED_PACKAGES
|
|
|
|
DEP_LIST=$(echo "$DEP_LIST" | sed 's/^ //g')
|
|
if [ "$DEP_LIST" != "" ]; then
|
|
|
|
for REPOID in `grep '^[[].*[]]$' $YUM | grep -v '[[]main[]]' | awk -F '[][]' '{print $2 }'`; do
|
|
echo "TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --whatprovides ${DEP_LIST} --qf='%{name}'"
|
|
TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --qf='%{name}' --whatprovides ${DEP_LIST} | sed "s/kernel-debug/kernel/g" >> $UNSORTED_PACKAGES
|
|
\rm -rf $TMP_DIR/yum-$USER-*
|
|
done
|
|
fi
|
|
|
|
sort -u $UNSORTED_PACKAGES > $SORTED_PACKAGES
|
|
fi
|
|
|
|
# clean up
|
|
\rm -f $UNSORTED_PACKAGES $UNRESOLVED_PACKAGES
|
|
|
|
# We now have, in SORTED_PACKAGES, a list of all packages that we need to install
|
|
# to meet our dependancies
|
|
DEP_LIST=" "
|
|
while read DEP
|
|
do
|
|
DEP_LIST="${DEP_LIST}${DEP} "
|
|
done < $SORTED_PACKAGES
|
|
rm $SORTED_PACKAGES
|
|
|
|
# go through each repo and install packages
|
|
local TARGETS=${DEP_LIST}
|
|
echo "Debug: Resolved list of deps to install: ${TARGETS}"
|
|
local UNRESOLVED
|
|
for REPOID in `grep '^[[].*[]]$' $YUM | grep -v '[[]main[]]' | awk -F '[][]' '{print $2 }'`; do
|
|
UNRESOLVED="$TARGETS"
|
|
|
|
if [[ ! -z "${TARGETS// }" ]]; then
|
|
REPO_PATH=$(cat $YUM | sed -n "/^\[$REPOID\]\$/,\$p" | grep '^baseurl=' | head -n 1 | awk -F 'file://' '{print $2}' | sed 's:/$::')
|
|
>&2 echo "TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --resolve $TARGETS --qf='%{name} %{name}-%{version}-%{release}.%{arch}.rpm %{relativepath}'"
|
|
TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --resolve $TARGETS --qf="%{name} %{name}-%{version}-%{release}.%{arch}.rpm %{relativepath}" | sort -r -V > $TMPFILE
|
|
\rm -rf $TMP_DIR/yum-$USER-*
|
|
|
|
while read STR
|
|
do
|
|
>&2 echo "STR=$STR"
|
|
if [ "x$STR" == "x" ]; then
|
|
continue
|
|
fi
|
|
|
|
PKG=`echo $STR | cut -d " " -f 1`
|
|
PKG_FILE=`echo $STR | cut -d " " -f 2`
|
|
PKG_REL_PATH=`echo $STR | cut -d " " -f 3`
|
|
PKG_PATH="${REPO_PATH}/${PKG_REL_PATH}"
|
|
|
|
>&2 echo "Installing PKG=$PKG PKG_FILE=$PKG_FILE PKG_REL_PATH=$PKG_REL_PATH PKG_PATH=$PKG_PATH from repo $REPOID"
|
|
cp $PKG_PATH .
|
|
if [ $? -ne 0 ]; then
|
|
>&2 echo " Here's what I have to work with..."
|
|
>&2 echo " TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --resolve $PKG --qf=\"%{name} %{name}-%{version}-%{release}.%{arch}.rpm %{relativepath}\""
|
|
>&2 echo " PKG=$PKG PKG_FILE=$PKG_FILE REPO_PATH=$REPO_PATH PKG_REL_PATH=$PKG_REL_PATH PKG_PATH=$PKG_PATH"
|
|
fi
|
|
|
|
echo $UNRESOLVED | grep $PKG >> /dev/null
|
|
if [ $? -eq 0 ]; then
|
|
echo "$PKG found in $REPOID as $PKG" >> $BUILT_REPORT
|
|
echo "$PKG_PATH" >> $BUILT_REPORT
|
|
UNRESOLVED=$(echo "$UNRESOLVED" | sed "s# $PKG # #g")
|
|
else
|
|
echo "$PKG satisfies unknown target in $REPOID" >> $BUILT_REPORT
|
|
echo " but it doesn't match targets, $UNRESOLVED" >> $BUILT_REPORT
|
|
echo " path $PKG_PATH" >> $BUILT_REPORT
|
|
FOUND_UNKNOWN=1
|
|
fi
|
|
done < $TMPFILE #<<< "$(TMPDIR=$TMP_DIR repoquery -c $YUM --repoid=$REPOID --arch=x86_64,noarch --resolve $TARGETS --qf=\"%{name} %{name}-%{version}-%{release}.%{arch}.rpm %{relativepath}\" | sort -r -V)"
|
|
\rm -rf $TMP_DIR/yum-$USER-*
|
|
TARGETS="$UNRESOLVED"
|
|
fi
|
|
done
|
|
>&2 echo "Debug: Packages still unresolved: $UNRESOLVED"
|
|
echo "Debug: Packages still unresolved: $UNRESOLVED" >> $WARNINGS_REPORT
|
|
echo "Debug: Packages still unresolved: $UNRESOLVED" >> $BUILT_REPORT
|
|
>&2 echo ""
|
|
}
|
|
|
|
function check_all_explicit_deps_installed {
|
|
|
|
PKGS_TO_CHECK=" "
|
|
while read PKG_TO_ADD
|
|
do
|
|
PKGS_TO_CHECK="$PKGS_TO_CHECK ${PKG_TO_ADD}"
|
|
done < $DEPLISTFILE
|
|
rpm -qp $MY_WORKSPACE/export/dist/isolinux/Packages/*.rpm --qf="%{name}\n" --nosignature > $TMPFILE
|
|
|
|
while read INSTALLED_PACKAGE
|
|
do
|
|
echo $PKGS_TO_CHECK | grep -q "${INSTALLED_PACKAGE}"
|
|
if [ $? -eq 0 ]; then
|
|
PKGS_TO_CHECK=`echo $PKGS_TO_CHECK | sed "s/^${INSTALLED_PACKAGE} //"`
|
|
PKGS_TO_CHECK=`echo $PKGS_TO_CHECK | sed "s/ ${INSTALLED_PACKAGE} / /"`
|
|
PKGS_TO_CHECK=`echo $PKGS_TO_CHECK | sed "s/ ${INSTALLED_PACKAGE}\$//"`
|
|
PKGS_TO_CHECK=`echo $PKGS_TO_CHECK | sed "s/^${INSTALLED_PACKAGE}\$//"`
|
|
fi
|
|
done < $TMPFILE
|
|
|
|
# Strip leading spaces. Don't want isomething like ' ' to trigger a failure
|
|
PKGS_TO_CHECK=`echo $PKGS_TO_CHECK | sed "s/^[ ]*//"`
|
|
if [ -z "$PKGS_TO_CHECK" ]; then
|
|
>&2 echo "All explicitly specified packages resolved!"
|
|
else
|
|
>&2 echo "Could not resolve packages: $PKGS_TO_CHECK"
|
|
return 1
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
ATTEMPTED=0
|
|
DISCOVERED=0
|
|
OUTPUT_DIR=$MY_WORKSPACE/export
|
|
TMP_DIR=$MY_WORKSPACE/tmp
|
|
YUM=$OUTPUT_DIR/yum.conf
|
|
DEPLISTFILE=$OUTPUT_DIR/deps.txt
|
|
DEPDETAILLISTFILE=$OUTPUT_DIR/deps_detail.txt
|
|
|
|
BUILT_REPORT=$OUTPUT_DIR/local.txt
|
|
WARNINGS_REPORT=$OUTPUT_DIR/warnings.txt
|
|
LAST_TEST=$OUTPUT_DIR/last_test.txt
|
|
TMPFILE=$OUTPUT_DIR/cgts_deps_tmp.txt
|
|
TMPFILE1=$OUTPUT_DIR/cgts_deps_tmp1.txt
|
|
TMPFILE2=$OUTPUT_DIR/cgts_deps_tmp2.txt
|
|
|
|
touch "$BUILT_REPORT"
|
|
touch "$WARNINGS_REPORT"
|
|
|
|
for i in "$@"
|
|
do
|
|
case $i in
|
|
-d=*|--deps=*)
|
|
DEPS="${i#*=}"
|
|
shift # past argument=value
|
|
;;
|
|
esac
|
|
done
|
|
|
|
mkdir -p $TMP_DIR
|
|
|
|
rm -f "$DEPDETAILLISTFILE"
|
|
# FIRST PASS we are being given a list of REQUIRED dependencies
|
|
if [ "${DEPS}x" != "x" ]; then
|
|
cat $DEPS | grep -v "^#" | sed '/^\s*$/d' > $DEPLISTFILE
|
|
install_deps $DEPLISTFILE
|
|
if [ $? -ne 0 ]; then
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
# check that we resolved them all
|
|
check_all_explicit_deps_installed
|
|
if [ $? -ne 0 ]; then
|
|
>&2 echo "Error -- could not install all explicitly listed packages"
|
|
exit 1
|
|
fi
|
|
|
|
ALL_RESOLVED=0
|
|
|
|
while [ $ALL_RESOLVED -eq 0 ]; do
|
|
cp $DEPLISTFILE $DEPLISTFILE.old
|
|
generate_dep_list
|
|
if [ ! -s $DEPLISTFILE ]; then
|
|
# no more dependencies!
|
|
ALL_RESOLVED=1
|
|
else
|
|
DIFFLINES=`diff $DEPLISTFILE.old $DEPLISTFILE | wc -l`
|
|
if [ $DIFFLINES -eq 0 ]; then
|
|
>&2 echo "Warning: Infinite loop detected in dependency resolution. See $DEPLISTFILE for details -- exiting"
|
|
>&2 echo "These RPMS had problems (likely version conflicts)"
|
|
>&2 cat $DEPLISTFILE
|
|
|
|
echo "Warning: Infinite loop detected in dependency resolution See $DEPLISTFILE for details -- exiting" >> $WARNINGS_REPORT
|
|
echo "These RPMS had problems (likely version conflicts)" >> $WARNINGS_REPORT
|
|
cat $DEPLISTFILE >> $WARNINGS_REPORT
|
|
|
|
date > $LAST_TEST
|
|
|
|
rm -f $DEPLISTFILE.old
|
|
exit 1 # nothing fixed
|
|
fi
|
|
install_deps $DEPLISTFILE
|
|
if [ $? -ne 0 ]; then
|
|
exit 1
|
|
fi
|
|
fi
|
|
done
|
|
|
|
exit 0
|