2007-03-06 00:40:45 +03:00
#!/bin/sh -efu
2002-03-25 23:37:46 +03:00
#
2003-04-22 18:24:35 +04:00
# Copyright (C) 2000,2003 Dmitry V. Levin <ldv@altlinux.org>
2007-08-28 00:41:17 +04:00
# Copyright (C) 2007 Alexey Tourbin <at@altlinux.org>
2002-03-25 23:37:46 +03:00
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
2007-03-06 00:40:45 +03:00
. @RPMCONFIGDIR@/functions
2007-08-28 00:41:17 +04:00
. @RPMCONFIGDIR@/find-package
2005-06-29 22:21:40 +04:00
2007-12-03 12:32:11 +03:00
Buildrooted()
{
local rep="$1"; shift || return
if [ -n "${RPM_BUILD_ROOT-}" ] &&
[ -n "$rep" -a -z "${rep##/*}" ] &&
[ -e "$RPM_BUILD_ROOT$rep" -o -L "$RPM_BUILD_ROOT$rep" ]; then
return 0
fi
return 1
}
2007-10-08 02:10:53 +04:00
shREQ= shPROV=
2007-08-28 00:41:17 +04:00
ShellReq()
{
local f="$1"; shift
2005-06-29 22:21:40 +04:00
2007-08-28 00:41:17 +04:00
local t sh
2007-08-28 02:12:58 +04:00
t=$(file -bL "$f") || Fatal "${t:-$f: file type not available}"
2007-08-28 00:41:17 +04:00
case "$t" in
*"Bourne-Again shell script text"*)
sh=/bin/bash ;;
*" bash script text"*)
sh=/bin/bash ;;
*)
sh=/bin/sh ;;
esac
$sh --rpm-requires </dev/null >/dev/null ||
Fatal "$sh interpreter does not support --rpm-requires feature"
2005-06-29 22:21:40 +04:00
2007-10-08 01:08:33 +04:00
local out line1
if ! out=$($sh --rpm-requires "$f"); then
2007-08-28 00:41:17 +04:00
# sh --rpm-requires failed, and stderr is already there.
# We are almost dead. The last chance to escape is to see
# if the shell is used only to re-exec another interpreter, e.g.
# exec tclsh "$0" "$@"
if line1=$(egrep -m1 -v '^[[:space:]]*(#|$)' "$f"); then
set -- $line1
if [ $# -gt 1 ] && [ "$1" = exec ]; then
Info "$f is $2 script!"
2007-09-24 02:59:25 +04:00
# We do no more than shebang.req does. If the script
# is not executable, shebang.req.files must have been
# already issued "executable not executable" warning.
if [ -x "$f" ]; then
FindPackage "$f" "$2"
fi
2007-08-28 00:41:17 +04:00
return 0
fi
fi
Fatal "$f: $sh --rpm-requires failed"
2002-03-25 23:37:46 +03:00
fi
2007-10-08 01:08:33 +04:00
[ -n "$out" ] || return 0
2007-10-03 14:31:23 +04:00
2007-10-07 18:05:35 +04:00
# This function is a "closure": parent variables are available without
# explicit argument passing. I checked that it works at least with some
# modern shells. Why I ever need a separate function is because my $EDITOR
# is not as perfect as to provide decent syntax highlight for subshells.
CleanupRpmRequires()
{
2007-10-08 01:08:33 +04:00
printf '%s\n' "$out" |
2007-10-08 00:40:15 +04:00
while read -r line; do
2007-10-07 18:05:35 +04:00
# NB: grep and sed are expensive here.
case "$line" in
# Basic sanity check for --rpm-requires output.
# Better command/path validation is performed in FindPackage.
'executable('*[A-Za-z0-9]*')' | 'function('*[A-Za-z0-9]*')' ) ;;
*) Info "$f: invalid $sh --rpm-requires output: $line"; continue ;;
esac
2007-10-08 00:40:15 +04:00
# Allow e.g. executable(\ls) and executable("ls").
set -- $(IFS="($IFS\\'\")"; echo $line)
2007-10-07 18:05:35 +04:00
case $# in
2007-10-08 00:16:18 +04:00
2) ;;
*) Info "$f: invalid $sh --rpm-requires output: $line"; continue ;;
esac
2007-11-30 13:45:03 +03:00
case "$(PATH= type -t -- "$2")" in
2007-11-20 08:20:48 +03:00
alias|keyword|builtin) continue ;;
2007-10-08 00:16:18 +04:00
*) printf '%s\t%s\n' "$1" "$2" ;;
2007-10-07 18:05:35 +04:00
esac
done
}
2007-10-08 01:08:33 +04:00
out=$(CleanupRpmRequires)
[ -n "$out" ] || return 0
2007-10-08 00:49:46 +04:00
# Now that the output is sane I can fold dups.
2007-10-08 02:10:53 +04:00
out=$(printf '%s\n' "$out" |LC_COLLATE=C sort -u)
2007-10-07 18:05:35 +04:00
2007-10-03 14:31:23 +04:00
# Self-requires elimination: first pass.
# Consider e.g. /etc/rc.d/init.d/functions has both
# executable(failure) and function(failure).
# This means that failure() is used before being defined.
# This is okay since it is actually used in another function.
# We want to keep only the function(failure).
2007-10-08 02:10:53 +04:00
GetProv()
{
printf '%s\n' "$out" |
while read -r t r; do
[ "$t" = function ] || continue
printf '%s\n' "$r"
done
}
local prov
prov=$(GetProv)
2007-10-03 14:04:25 +04:00
2007-10-08 02:10:53 +04:00
GetReq()
{
printf '%s\n' "$out" |
while read -r t r; do
[ "$t" = executable ] || continue
if [ -n "$prov" ] && printf '%s\n' "$prov" |fgrep -qs -x -e "$r"; then
2007-10-29 13:35:15 +03:00
# Still check if executable candidates are available.
local exe="$(FindPackage "$f" "$r" 2>/dev/null)"
local Verbose=Verbose
2007-12-03 12:32:11 +03:00
if [ -n "$exe" ] && ! Buildrooted "$exe"; then
2007-10-29 13:35:15 +03:00
Verbose=Info
fi
$Verbose "$f: $r() is possibly used before its definition"
if [ -n "$exe" ]; then
2007-12-03 12:32:11 +03:00
$Verbose "$f: $r() has execuatable candidate: $r -> $exe (skip)"
2007-10-29 13:35:15 +03:00
fi
2007-10-08 02:10:53 +04:00
continue
fi
printf '%s\n' "$r"
done
}
local req
req=$(GetReq)
2007-03-06 00:40:45 +03:00
2007-10-09 00:22:42 +04:00
AddReqProv()
2007-10-08 02:10:53 +04:00
{
2007-10-09 00:22:42 +04:00
[ -z "$1" ] || printf '%s\n' "$1"
[ -n "$2" ] || return 0
printf '%s\n' "$2" |
2007-10-08 02:10:53 +04:00
while read -r r; do
printf '%s\t%s\n' "$f" "$r"
done
}
2007-10-09 00:22:42 +04:00
shREQ=$(AddReqProv "$shREQ" "$req")
shPROV=$(AddReqProv "$shPROV" "$prov")
2007-10-03 14:04:25 +04:00
}
ShellReqEND()
{
2007-10-08 02:10:53 +04:00
[ -n "$shREQ" ] || return 0
Debug "shREQ=$shREQ"
2007-10-09 00:51:29 +04:00
Debug "shPROV=$shPROV"
# Self-requires elimination: second pass.
# If ANY file has function(foo), skip all dependencies on execuatable(foo).
2007-10-08 02:10:53 +04:00
printf '%s\n' "$shREQ" |
2007-10-03 14:04:25 +04:00
while IFS=$'\t' read -r f r; do
2007-10-08 02:10:53 +04:00
CheckSelfReq()
{
[ -n "$shPROV" ] || return 0
printf '%s\n' "$shPROV" |
while IFS=$'\t' read -r f2 r2; do
if [ "$r" = "$r2" ]; then
printf '%s\n' "$f2"
fi
done
}
local self_req
self_req=$(CheckSelfReq)
if [ -n "$self_req" ]; then
2007-10-09 16:50:20 +04:00
# Still check if executable candidates are available.
local exe="$(FindPackage "$f" "$r" 2>/dev/null)"
local Verbose=Verbose
2007-12-03 12:32:11 +03:00
if [ -n "$exe" ] && ! Buildrooted "$exe"; then
2007-10-09 16:50:20 +04:00
Verbose=Info
fi
2007-10-08 02:10:53 +04:00
printf '%s\n' "$self_req" |
while read -r f2; do
2007-10-09 16:50:20 +04:00
$Verbose "$f: $r() is possibly defined in $f2"
2007-10-08 02:10:53 +04:00
done
2007-10-09 16:50:20 +04:00
if [ -n "$exe" ]; then
2007-12-03 12:32:11 +03:00
$Verbose "$f: $r() has execuatable candidate: $r -> $exe (skip)"
2007-10-09 16:50:20 +04:00
fi
2007-10-08 02:10:53 +04:00
continue
fi
2007-10-08 02:27:27 +04:00
local dir
dir=${f#${RPM_BUILD_ROOT-}}
dir=${dir%/*}
CheckDirProv()
{
fgrep -qs -x -e "$r" "$1/.provides.sh" || return
}
if [ -n "${RPM_BUILD_ROOT-}" ] && CheckDirProv "$RPM_BUILD_ROOT$dir"; then
# I think that .provides.sh must provide this thingy.
# Otherwise we get an unmet depedency, which is good --
# one has to fix the .provides.sh early.
printf '%s(%s)\n' "$dir" "$r"
elif CheckDirProv "$dir"; then
printf '%s(%s)\n' "$dir" "$r"
2007-03-06 00:40:45 +03:00
else
2007-08-28 00:41:17 +04:00
FindPackage "$f" "$r"
2003-04-25 20:51:52 +04:00
fi
2007-10-08 02:10:53 +04:00
done
2002-03-25 23:37:46 +03:00
}
2007-08-28 00:41:17 +04:00
ArgvFileAction ShellReq "$@"
2007-10-03 14:04:25 +04:00
ShellReqEND