mirror of
https://github.com/zardus/ctf-tools
synced 2025-01-18 23:43:59 +00:00
495 lines
11 KiB
Bash
Executable file
495 lines
11 KiB
Bash
Executable file
#!/bin/bash
|
|
set -eu -o pipefail
|
|
# set -x
|
|
|
|
CTF_TOOLS_ROOT="$(dirname "${BASH_SOURCE[0]}")/.."
|
|
|
|
|
|
function usage()
|
|
{
|
|
cat <<END
|
|
Usage: $(basename $0) [-sv] (list|setup|install|uninstall|bin|search) tool
|
|
|
|
Where:
|
|
-s allow running things with sudo (i.e., to install debs)
|
|
-n "nice" the installer to reduce background load
|
|
-v verbose mode. print log while installing
|
|
-f force certain actions (such as installing over an installed tool)
|
|
tool the name of the tool. if "all", does the action on all (installed) tools
|
|
|
|
Actions:
|
|
setup set up the environment (adds ctf-tools/bin to your \$PATH in .bashrc)
|
|
list list all tools (-i: only installed, -u: only uninstalled)
|
|
install installs a tool
|
|
uninstall uninstalls a tool
|
|
reinstall reinstalls a tool
|
|
upgrade upgrades a tool (use "all" to do a full upgrade of installed tools)
|
|
bin re-links tool binaries into ctf-tools/bin
|
|
search search description and name of tools
|
|
|
|
END
|
|
}
|
|
|
|
function tool_log()
|
|
{
|
|
echo "$(tput setaf 4 2>/dev/null)TOOLS | $TOOL |$(tput sgr0 2>/dev/null) $@"
|
|
}
|
|
|
|
|
|
function detect_distribution()
|
|
{
|
|
if which pacman >/dev/null 2>&1; then
|
|
echo "archlinux"
|
|
elif which apt-get >/dev/null 2>&1; then
|
|
if lsb_release -a 2>/dev/null | grep -i ubuntu >/dev/null 2>&1; then
|
|
echo "ubuntu"
|
|
else
|
|
echo "debian"
|
|
fi
|
|
elif which dnf >/dev/null 2>&1; then
|
|
echo "fedora"
|
|
else
|
|
echo ""
|
|
fi
|
|
}
|
|
|
|
|
|
function base_build_setup_debian()
|
|
{
|
|
PACKAGE_REQS="build-essential libtool g++ gcc texinfo curl wget automake autoconf python python-dev git subversion unzip virtualenvwrapper lsb-release"
|
|
PACKAGE_COUNT=$(echo $PACKAGE_REQS | tr ' ' '\n' | wc -l)
|
|
sudo apt-get update
|
|
if [ $(dpkg -l $PACKAGE_REQS | grep "^ii" | wc -l) -ne $PACKAGE_COUNT ]
|
|
then
|
|
if [ "$ALLOW_SUDO" -eq 1 ]; then
|
|
sudo apt-get -y install $PACKAGE_REQS
|
|
else
|
|
TOOL=SETUP tool_log "Please install the following packages: $PACKAGE_REQS"
|
|
fi
|
|
fi
|
|
|
|
if ! dpkg --print-foreign-architectures | grep -q i386
|
|
then
|
|
if [ "$ALLOW_SUDO" -eq 1 ]
|
|
then
|
|
sudo dpkg --add-architecture i386
|
|
sudo apt-get update
|
|
else
|
|
TOOL=SETUP tool_log "Certain tools need i386 libraries (enable with 'dpkg --add-architecture i386; apt-get update')."
|
|
fi
|
|
fi
|
|
}
|
|
|
|
|
|
function base_build_setup_arch()
|
|
{
|
|
PACKAGE_REQS="curl wget python2 python3 git subversion unzip python-virtualenvwrapper"
|
|
if [ "$ALLOW_SUDO" -eq 1 ]; then
|
|
sudo pacman -Syu --noconfirm --needed $PACKAGE_REQS
|
|
sudo pacman -Syu --noconfirm --needed base-devel || true
|
|
else
|
|
TOOL=SETUP tool_log "Please install the following packages: $PACKAGE_REQS"
|
|
fi
|
|
|
|
if ! grep "^\[multilib\]$" /etc/pacman.conf >/dev/null; then
|
|
if [ "$ALLOW_SUDO" -eq 1 ]; then
|
|
sudo sh -c 'cat >> /etc/pacman.conf' <<EOF
|
|
|
|
[multilib]
|
|
Include = /etc/pacman.d/mirrorlist
|
|
EOF
|
|
else
|
|
TOOL=SETUP tool_log "Certain tools need i386 libraries (enable multilib in /etc/pacman.conf')."
|
|
return
|
|
fi
|
|
fi
|
|
|
|
if [ "$ALLOW_SUDO" -eq 1 ] \
|
|
&& grep "^\[multilib\]$" /etc/pacman.conf >/dev/null \
|
|
&& ! sudo pacman -Qk gcc-multilib >/dev/null
|
|
then
|
|
sudo pacman -Syy --noconfirm
|
|
#sudo pacman -Syu --noconfirm multilib-devel
|
|
# unfortunately we cannot do --noconfirm if we might choose to replace
|
|
# a package such as gcc with gcc-multilib, therefore this workaround
|
|
printf "\ny\ny\ny\n" | sudo pacman -Syu multilib-devel
|
|
fi
|
|
}
|
|
|
|
|
|
function base_build_setup_fedora()
|
|
{
|
|
PACKAGE_REQS="libtool gcc gcc-c++ clang cmake texinfo curl wget automake autoconf python python-devel git subversion unzip python-virtualenvwrapper redhat-rpm-config"
|
|
if [ "$ALLOW_SUDO" -eq 1 ]; then
|
|
sudo dnf -y install $PACKAGE_REQS
|
|
else
|
|
TOOL=SETUP tool_log "Please install the following packages: $PACKAGE_REQS"
|
|
fi
|
|
|
|
# TODO: check whether we have to explicitly enable i386 package support
|
|
}
|
|
|
|
|
|
function base_build_setup()
|
|
{
|
|
case "$1" in
|
|
"ubuntu")
|
|
;& # fallthrough
|
|
"debian")
|
|
base_build_setup_debian
|
|
;;
|
|
"archlinux")
|
|
base_build_setup_arch
|
|
export VIRTUALENVWRAPPER_PYTHON=/usr/bin/python3
|
|
;;
|
|
"fedora")
|
|
base_build_setup_fedora
|
|
;;
|
|
*)
|
|
TOOL=SETUP tool_log "Cannot detect or unsupported distribution"
|
|
esac
|
|
|
|
## setup PATH for several shells
|
|
|
|
MAGIC="# ctf-tools: PATH setup"
|
|
# make sure at least bashrc exists in case of plain VM setup
|
|
touch ~/.bashrc
|
|
|
|
for f in ~/.bashrc ~/.zshrc; do
|
|
if [[ -e "$f" ]]; then
|
|
if ! grep "$MAGIC" "$f" >/dev/null 2>&1; then
|
|
cat >> "$f" << EOF
|
|
$MAGIC
|
|
export PATH=$PWD/bin:\$PATH
|
|
EOF
|
|
fi
|
|
fi
|
|
done
|
|
|
|
f=~/.config/fish/config.fish
|
|
if [[ -e "$f" ]]; then
|
|
if ! grep "$MAGIC" "$f" >/dev/null 2>&1; then
|
|
cat >> "$f" << EOF
|
|
$MAGIC
|
|
set -x PATH $PWD/bin \$PATH
|
|
EOF
|
|
fi
|
|
fi
|
|
|
|
if [[ ! -e "$PWD/bin/ctf-tools-pip3" ]]; then
|
|
ln -s "$PWD/bin/ctf-tools-pip" "$PWD/bin/ctf-tools-pip3"
|
|
fi
|
|
|
|
# create the py2 virtualenv
|
|
"$PWD/bin/ctf-tools-pip" freeze 2>&1 >/dev/null
|
|
|
|
# create the py3 virtualenv
|
|
"$PWD/bin/ctf-tools-pip3" freeze 2>&1 >/dev/null
|
|
}
|
|
|
|
|
|
function is_tool_installed() {
|
|
git status --ignored "$1" | egrep -q 'Untracked|Ignored'
|
|
}
|
|
|
|
|
|
function full_upgrade() {
|
|
TOOL="FULL-UPGRADE" tool_log "Upgrading all installed tools!"
|
|
succ=0
|
|
fail=0
|
|
declare -a failed
|
|
installed=$($0 list -i)
|
|
TOOL="FULL-UPGRADE" tool_log "Upgrading tools: $installed"
|
|
for t in $installed
|
|
do
|
|
TOOL="FULL-UPGRADE" tool_log "Upgrading tool $t"
|
|
if $0 upgrade $t; then
|
|
succ=$((succ+1))
|
|
else
|
|
fail=$((fail+1))
|
|
failed[${#failed[@]}]="$t"
|
|
fi
|
|
done
|
|
TOOL="FULL-UPGRADE" tool_log "failed to upgrade ${failed[@]}"
|
|
TOOL="FULL-UPGRADE" tool_log "tool full-upgrade stats - sucess=$succ failed=$fail"
|
|
}
|
|
|
|
function verify_tool_sanity() {
|
|
if [[ -z "$TOOL" ]]; then
|
|
TOOL="$ACTION" tool_log "must provide at least one tool for $ACTION"
|
|
usage
|
|
exit 1
|
|
fi
|
|
|
|
if [[ "$TOOL" == "all" ]]; then
|
|
TOOL="$ACTION" tool_log "can't handle \"all\" magic tool directly!" \
|
|
"This is probably a bug you should report."
|
|
usage
|
|
exit 1
|
|
fi
|
|
|
|
if [[ ! -d "$CTF_TOOLS_ROOT/$TOOL" ]]; then
|
|
TOOL="$ACTION" tool_log "invalid tool $TOOL"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
|
|
if [[ $# -eq 0 ]]; then
|
|
usage
|
|
exit 1
|
|
fi
|
|
|
|
DISTRI=$(detect_distribution)
|
|
|
|
while [[ $1 == -* ]]
|
|
do
|
|
case $1 in
|
|
-s)
|
|
export ALLOW_SUDO=1
|
|
;;
|
|
-n)
|
|
export NICE_LEVEL=10
|
|
;;
|
|
-f)
|
|
export FORCE=1
|
|
;;
|
|
-v)
|
|
export VERBOSE_OUTPUT=1
|
|
;;
|
|
*)
|
|
usage
|
|
exit
|
|
;;
|
|
esac
|
|
shift
|
|
done
|
|
|
|
[[ -z ${ALLOW_SUDO+x} ]] && export ALLOW_SUDO=0
|
|
[[ -z ${FORCE+x} ]] && export FORCE=0
|
|
[[ -z ${VERBOSE_OUTPUT+x} ]] && export VERBOSE_OUTPUT=0
|
|
[[ -z ${NICE_LEVEL+x} ]] && export NICE_LEVEL=0
|
|
export EXPECTFAIL=${EXPECTFAIL:-0}
|
|
|
|
if [[ $# -ge 1 ]]; then
|
|
ACTION="$1"
|
|
fi
|
|
if [[ $# -eq 2 ]]; then
|
|
TOOL="$2"
|
|
else
|
|
TOOL=""
|
|
fi
|
|
|
|
# handle the special all tool
|
|
if [[ "$TOOL" == "all" ]]
|
|
then
|
|
case $ACTION in
|
|
install)
|
|
for t in $($0 list)
|
|
do
|
|
$0 $ACTION $t
|
|
done
|
|
exit 0
|
|
;;
|
|
bin | uninstall | reinstall)
|
|
for t in $($0 list -i)
|
|
do
|
|
$0 $ACTION $t
|
|
done
|
|
exit 0
|
|
;;
|
|
upgrade)
|
|
full_upgrade
|
|
exit 0
|
|
;;
|
|
*)
|
|
TOOL="" tool_log "action $ACTION cannot handle the special \"all\" tool"
|
|
usage
|
|
exit 1
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
|
|
|
|
cd $(dirname "${BASH_SOURCE[0]}")/..
|
|
|
|
case $ACTION in
|
|
setup)
|
|
|
|
base_build_setup "$DISTRI"
|
|
;;
|
|
list)
|
|
for t in *
|
|
do
|
|
[ ! -e "$t/install" ] && continue
|
|
|
|
if [[ "${2:-}" == "" ]]; then
|
|
echo "$t"
|
|
else
|
|
if is_tool_installed "$t"; then
|
|
if [[ "$2" == "-i" ]]; then
|
|
echo "$t"
|
|
fi
|
|
else
|
|
if [[ "$2" == "-u" ]]; then
|
|
echo "$t"
|
|
fi
|
|
fi
|
|
fi
|
|
done
|
|
;;
|
|
bin)
|
|
verify_tool_sanity
|
|
cd bin
|
|
if [ -d ../$TOOL/bin ]; then
|
|
ln -sf ../$TOOL/bin/* .
|
|
tool_log "bin symlinks updated"
|
|
fi
|
|
cd ..
|
|
;;
|
|
install)
|
|
verify_tool_sanity
|
|
cd $TOOL
|
|
if [ "$FORCE" -eq 0 ] && is_tool_installed "."
|
|
then
|
|
tool_log "appears to already be installed. Uninstall first?"
|
|
exit 0
|
|
fi
|
|
|
|
# the first line in all install and uninstall scripts should have the -e flag, otherwise fail
|
|
if [ $(for i in install* uninstall test; do if [ -e "$i" ]; then head -1 "$i"; fi; done | sort | uniq | grep -v '^#!/bin/bash -ex$' | wc -l) -ne 0 ];
|
|
then
|
|
tool_log "not all install/uninstall/test scripts start with '#!/bin/bash -ex', which is a must for accurate testing."
|
|
exit 1
|
|
fi
|
|
|
|
tool_log "starting install, logging to $PWD/install.log"
|
|
rm -f install.log
|
|
|
|
# first get distri specific dependencies
|
|
if [[ $(find . -name 'install-root*' | wc -l) -ge 1 ]]; then
|
|
INSTALL_ROOT_SCRIPT="./install-root-$DISTRI"
|
|
# use debian install script if we are on ubuntu and no ubuntu
|
|
# specific install script exists
|
|
if [[ "$DISTRI" == "ubuntu" \
|
|
&& ! -x "$INSTALL_ROOT_SCRIPT" \
|
|
&& -x "./install-root-debian" ]]
|
|
then
|
|
INSTALL_ROOT_SCRIPT="./install-root-debian"
|
|
fi
|
|
if [[ -x "$INSTALL_ROOT_SCRIPT" && "$ALLOW_SUDO" -eq 1 ]]; then
|
|
set +e
|
|
if [[ "$VERBOSE_OUTPUT" -eq 1 ]]; then
|
|
sudo env DISTRI=$DISTRI "$INSTALL_ROOT_SCRIPT" 2>&1 | tee -a install.log
|
|
else
|
|
sudo env DISTRI=$DISTRI "$INSTALL_ROOT_SCRIPT" >> install.log 2>&1
|
|
fi
|
|
INSTALL_FAILED=$?
|
|
set -e
|
|
|
|
if [[ "$INSTALL_FAILED" -eq 0 ]]; then
|
|
tool_log "system dependencies installed"
|
|
else
|
|
tool_log "INSTALL FAILED: $INSTALL_ROOT_SCRIPT failed to install dependencies"
|
|
cat install.log >&2
|
|
exit 1
|
|
fi
|
|
else
|
|
tool_log "Warning: make sure build dependencies are installed!"
|
|
fi
|
|
fi
|
|
|
|
# execute install script
|
|
set +e
|
|
if [ "$VERBOSE_OUTPUT" -eq 1 ]; then
|
|
DISTRI=$DISTRI PATH=$CTF_TOOLS_ROOT/bin/:$PATH nice -n$NICE_LEVEL ./install 2>&1 | tee -a install.log
|
|
else
|
|
DISTRI=$DISTRI PATH=$CTF_TOOLS_ROOT/bin/:$PATH nice -n$NICE_LEVEL ./install >>install.log 2>&1
|
|
fi
|
|
INSTALL_FAILED=$?
|
|
set -e
|
|
|
|
if [ "$INSTALL_FAILED" -eq 0 ]; then
|
|
tool_log "install finished"
|
|
else
|
|
tool_log "INSTALL FAILED"
|
|
cat install.log >&2
|
|
exit 1
|
|
fi
|
|
|
|
cd ..
|
|
$0 bin $TOOL
|
|
;;
|
|
uninstall)
|
|
verify_tool_sanity
|
|
cd $TOOL
|
|
|
|
tool_log "starting uninstall, logging to $PWD/uninstall.log"
|
|
[ -x ./uninstall ] && ./uninstall >> uninstall.log 2>&1
|
|
git clean -dffx . >/dev/null 2>&1
|
|
tool_log "uninstall finished"
|
|
|
|
cd ..
|
|
;;
|
|
upgrade)
|
|
if [[ "$TOOL" == "" ]]; then
|
|
full_upgrade
|
|
exit 0
|
|
fi
|
|
verify_tool_sanity
|
|
cd $TOOL
|
|
if [ -x ./upgrade ]
|
|
then
|
|
./upgrade
|
|
tool_log "upgrade complete!"
|
|
else
|
|
tool_log "no upgrade script -- reinstalling"
|
|
$0 uninstall $TOOL
|
|
$0 install $TOOL
|
|
fi
|
|
;;
|
|
reinstall)
|
|
$0 uninstall $TOOL
|
|
$0 install $TOOL
|
|
;;
|
|
search)
|
|
cat README.md | grep "<\!--tool-->" | sed "s/<\!--[^-]*-->//g" | grep -i "$TOOL"
|
|
;;
|
|
test)
|
|
if [ "$FORCE" -eq 0 ] && ! cat README.md | grep "<\!--tool-->" | grep "| \[$TOOL\](" | grep -q -- "--test--"
|
|
then
|
|
tool_log "Tests not enabled."
|
|
if [ "$EXPECTFAIL" -eq "1" ]; then exit 1; fi
|
|
else
|
|
if (
|
|
if ! $0 install $TOOL; then exit 1; fi
|
|
|
|
cd $TOOL || exit 1
|
|
if [ -f ./test ]
|
|
then
|
|
tool_log "Running test script."
|
|
if ! ./test
|
|
then
|
|
tool_log "$TOOL test failed!"
|
|
exit 1
|
|
fi
|
|
tool_log "test script succeeded!"
|
|
else
|
|
tool_log "Install succeeded. No test script!"
|
|
fi
|
|
exit 0
|
|
);
|
|
then
|
|
if [ "$EXPECTFAIL" -eq "1" ]; then exit 1; else exit 0; fi
|
|
else
|
|
if [ "$EXPECTFAIL" -eq "1" ]; then exit 0; else exit 1; fi
|
|
fi
|
|
fi
|
|
;;
|
|
*)
|
|
echo "TOOLS | ERROR | unknown action $ACTION"
|
|
usage
|
|
exit 1
|
|
;;
|
|
esac
|