2024-01-21 20:49:34 -08:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
# vim:set ft=bash:
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
skip_questions()
|
|
|
|
{
|
|
|
|
while :; do
|
|
|
|
case $1 in
|
|
|
|
-y | --yes ) return 0 ;;
|
|
|
|
* ) break ;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
shift 1
|
|
|
|
done
|
|
|
|
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
ask_for_sudo()
|
|
|
|
{
|
|
|
|
sudo -v &> /dev/null
|
|
|
|
|
|
|
|
# Update existing 'sudo' timestamp until this script has finished.
|
|
|
|
#
|
|
|
|
# https://gist.github.com/cowboy/3118588
|
|
|
|
|
|
|
|
while true; do
|
|
|
|
sudo -n true
|
|
|
|
sleep 60
|
|
|
|
kill -0 "$$" || exit
|
|
|
|
done &> /dev/null &
|
|
|
|
}
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
print_in_color()
|
|
|
|
{
|
2024-01-25 13:58:01 -08:00
|
|
|
string=$(echo "$1" | tr -s " ")
|
|
|
|
|
2024-01-21 20:49:34 -08:00
|
|
|
printf "%b" \
|
|
|
|
"$(tput setaf "$2" 2> /dev/null)" \
|
2024-01-25 13:58:01 -08:00
|
|
|
"$string" \
|
2024-01-21 20:49:34 -08:00
|
|
|
"$(tput sgr0 2> /dev/null)"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_in_red()
|
|
|
|
{
|
|
|
|
print_in_color "$1" 1
|
|
|
|
}
|
|
|
|
|
|
|
|
print_in_yellow()
|
|
|
|
{
|
|
|
|
print_in_color "$1" 3
|
|
|
|
}
|
|
|
|
|
|
|
|
print_in_green()
|
|
|
|
{
|
|
|
|
print_in_color "$1" 2
|
|
|
|
}
|
|
|
|
|
|
|
|
print_in_purple()
|
|
|
|
{
|
|
|
|
print_in_color "$1" 5
|
|
|
|
}
|
|
|
|
|
|
|
|
print_title()
|
|
|
|
{
|
|
|
|
print_in_purple "\n • $1\n\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_success()
|
|
|
|
{
|
|
|
|
print_in_green " [✔] $1\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_warning()
|
|
|
|
{
|
|
|
|
print_in_yellow " [!] $1\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_error()
|
|
|
|
{
|
|
|
|
print_in_red " [✖] $1 $2\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_question()
|
|
|
|
{
|
|
|
|
print_in_yellow " [?] $1\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_result()
|
|
|
|
{
|
|
|
|
if [[ "$1" == 0 ]]; then
|
|
|
|
print_success "$2"
|
|
|
|
else
|
|
|
|
print_error "$2"
|
|
|
|
fi
|
|
|
|
|
|
|
|
return "$1"
|
|
|
|
}
|
|
|
|
|
|
|
|
print_error_stream()
|
|
|
|
{
|
|
|
|
while read -r line; do
|
|
|
|
print_error "↳ ERROR: $line"
|
|
|
|
done
|
|
|
|
}
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
ask()
|
|
|
|
{
|
|
|
|
print_question "$1"
|
|
|
|
read -r
|
|
|
|
}
|
|
|
|
|
|
|
|
get_answer()
|
|
|
|
{
|
|
|
|
printf "%s" "$REPLY"
|
|
|
|
}
|
|
|
|
|
|
|
|
ask_for_confirmation()
|
|
|
|
{
|
|
|
|
print_question "$1 (y/n) "
|
|
|
|
read -r -n 1
|
|
|
|
printf "\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
answer_is_yes()
|
|
|
|
{
|
|
|
|
[[ "$REPLY" =~ ^[Yy]$ ]] \
|
|
|
|
&& return 0 \
|
|
|
|
|| return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
show_spinner()
|
|
|
|
{
|
|
|
|
local -r FRAMES='/-\|'
|
|
|
|
|
|
|
|
# shellcheck disable=SC2034
|
|
|
|
local -r NUMBER_OF_FRAMES=${#FRAMES}
|
|
|
|
|
|
|
|
local -r CMDS="$2"
|
|
|
|
local -r MSG="$3"
|
|
|
|
local -r PID="$1"
|
|
|
|
|
|
|
|
local i=0
|
|
|
|
local frame_text=""
|
|
|
|
|
|
|
|
# Provide more space so that the text hopefully doesn't reach the bottom line
|
|
|
|
# of the terminal window.
|
|
|
|
#
|
|
|
|
# This is a workaround for escape sequences not tracking the buffer position
|
|
|
|
# (accounting for scrolling).
|
|
|
|
#
|
|
|
|
# See also: https://unix.stackexchange.com/a/278888
|
|
|
|
|
|
|
|
printf "\n\n\n"
|
|
|
|
tput cuu 3
|
|
|
|
tput sc
|
|
|
|
|
|
|
|
while kill -0 "$PID" &> /dev/null; do
|
2024-01-25 13:58:01 -08:00
|
|
|
frame_text=" [${FRAMES:i++%NUMBER_OF_FRAMES:1}] $MSG"
|
2024-01-21 20:49:34 -08:00
|
|
|
|
|
|
|
# Print frame text.
|
|
|
|
printf "%s" "$frame_text"
|
|
|
|
|
|
|
|
sleep 0.2
|
|
|
|
|
|
|
|
# Clear frame text.
|
|
|
|
tput rc
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
cmd_exists()
|
|
|
|
{
|
|
|
|
command -v "$1" &> /dev/null
|
|
|
|
}
|
|
|
|
|
|
|
|
is_git_repository()
|
|
|
|
{
|
|
|
|
git rev-parse &> /dev/null
|
|
|
|
}
|
|
|
|
|
|
|
|
# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
|
|
|
|
|
|
set_trap()
|
|
|
|
{
|
|
|
|
trap -p "$1" | grep "$2" &> /dev/null \
|
2024-01-25 13:58:01 -08:00
|
|
|
|| trap '$2' "$1"
|
2024-01-21 20:49:34 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
kill_all_subproccesses()
|
|
|
|
{
|
|
|
|
local i=""
|
|
|
|
|
|
|
|
for i in $(jobs -p); do
|
|
|
|
kill "$i"
|
|
|
|
wait "$i" &> /dev/null
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
execute()
|
|
|
|
{
|
|
|
|
local -r CMDS="$1"
|
|
|
|
local -r MSG="${2:-$1}"
|
|
|
|
local -r TMP_FILE="$(mktemp /tmp/XXXXX)"
|
|
|
|
|
|
|
|
local exit_code=0
|
|
|
|
local cmds_pid=""
|
|
|
|
|
|
|
|
# If the current process is ended, also end all its subproccesses.
|
|
|
|
set_trap "EXIT" "kill_all_subproccesses"
|
|
|
|
|
|
|
|
# Execute commands in background
|
|
|
|
# shellcheck disable=SC2261
|
|
|
|
eval "$CMDS" \
|
|
|
|
&> /dev/null \
|
|
|
|
2> "$TMP_FILE" &
|
|
|
|
|
|
|
|
cmds_pid=$!
|
|
|
|
|
|
|
|
# Show a spinner if the commands require more time to complete.
|
|
|
|
show_spinner "$cmds_pid" "$CMDS" "$MSG"
|
|
|
|
|
|
|
|
# Wait for the commands to no longer be executing in the background, and then
|
|
|
|
# get their exit code.
|
|
|
|
wait "$cmds_pid" &> /dev/null
|
|
|
|
exit_code=$?
|
|
|
|
|
|
|
|
# Print output based on what happened.
|
|
|
|
print_result $exit_code "$MSG"
|
|
|
|
|
2024-01-22 21:02:43 -08:00
|
|
|
if [[ $exit_code -ne 0 ]]; then
|
2024-01-21 20:49:34 -08:00
|
|
|
print_error_stream < "$TMP_FILE"
|
|
|
|
fi
|
|
|
|
|
|
|
|
rm -rf "$TMP_FILE"
|
|
|
|
|
|
|
|
return $exit_code
|
|
|
|
}
|