update installer, select multiple packages at the same time
This commit is contained in:
207
install.sh
207
install.sh
@@ -3,8 +3,8 @@
|
|||||||
# --- Main Script ---
|
# --- Main Script ---
|
||||||
|
|
||||||
# This script presents a menu of software packages defined in a remote
|
# This script presents a menu of software packages defined in a remote
|
||||||
# configuration file. The user can select a package, and the script
|
# configuration file. The user can select one or more packages, and the
|
||||||
# will download the corresponding files. It includes a feature to show
|
# script will download the corresponding files. It includes a feature to show
|
||||||
# a diff and ask for confirmation before overwriting existing config files.
|
# a diff and ask for confirmation before overwriting existing config files.
|
||||||
# It can also check for updates to already-installed scripts.
|
# It can also check for updates to already-installed scripts.
|
||||||
|
|
||||||
@@ -15,7 +15,7 @@ log() {
|
|||||||
echo "[$1] $2"
|
echo "[$1] $2"
|
||||||
}
|
}
|
||||||
|
|
||||||
# New function to get the version from a local script file.
|
# Get the version from a local script file.
|
||||||
# It reads the first 5 lines and extracts the version number.
|
# It reads the first 5 lines and extracts the version number.
|
||||||
get_local_version() {
|
get_local_version() {
|
||||||
local file_path="$1"
|
local file_path="$1"
|
||||||
@@ -27,7 +27,7 @@ get_local_version() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# New function to compare two version strings (e.g., "1.2.0" vs "1.10.0").
|
# Compare two version strings (e.g., "1.2.0" vs "1.10.0").
|
||||||
# Returns 0 if v1 is newer, 1 if they are the same or v2 is newer.
|
# Returns 0 if v1 is newer, 1 if they are the same or v2 is newer.
|
||||||
is_version_greater() {
|
is_version_greater() {
|
||||||
local v1=$1
|
local v1=$1
|
||||||
@@ -41,82 +41,15 @@ is_version_greater() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# --- Main Logic ---
|
# New function to process a single selected package.
|
||||||
|
process_package() {
|
||||||
# Generate a unique temporary filename with a timestamp.
|
local choice="$1"
|
||||||
conf_file="packages.conf.$(date +%Y%m%d%H%M%S)"
|
# Check if the choice is a valid package name.
|
||||||
|
if [[ -z "${SCRIPT_PACKAGES[$choice]}" ]]; then
|
||||||
# Set up a trap to delete the temporary file on exit.
|
log "❌" "Invalid package name provided: '${choice}'"
|
||||||
trap 'rm -f "${conf_file}"' EXIT
|
return
|
||||||
|
|
||||||
# Download the configuration file.
|
|
||||||
log "🔄" "Downloading configuration file..."
|
|
||||||
if ! curl -fsSL -o "${conf_file}" "https://git.technopunk.space/tomi/Scripts/raw/branch/main/packages.conf"; then
|
|
||||||
log "❌" "Error: Failed to download packages.conf. Exiting."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
log "✅" "Configuration file downloaded successfully."
|
|
||||||
|
|
||||||
# Source the configuration file to load the SCRIPT_PACKAGES associative array.
|
|
||||||
source "${conf_file}"
|
|
||||||
|
|
||||||
# --- Update Check & User Interface ---
|
|
||||||
|
|
||||||
echo "-------------------------------------"
|
|
||||||
echo " Script Downloader "
|
|
||||||
echo "-------------------------------------"
|
|
||||||
|
|
||||||
# Create an array of options from the package names.
|
|
||||||
# We will modify this array to show installation and update status.
|
|
||||||
declare -a options
|
|
||||||
package_keys=("${!SCRIPT_PACKAGES[@]}")
|
|
||||||
|
|
||||||
log "🔎" "Checking for updates..."
|
|
||||||
for key in "${package_keys[@]}"; do
|
|
||||||
# The config format is now "VERSION|URL1 URL2..."
|
|
||||||
config_value="${SCRIPT_PACKAGES[$key]}"
|
|
||||||
remote_version=$(echo "${config_value}" | cut -d'|' -f1)
|
|
||||||
|
|
||||||
# Get just the URLs and assume the first URL is the main script to check.
|
|
||||||
urls=$(echo "${config_value}" | cut -d'|' -f2-)
|
|
||||||
read -r -a url_array <<< "$urls"
|
|
||||||
main_script_filename=$(basename "${url_array[0]}")
|
|
||||||
|
|
||||||
# Get the local version of the main script file.
|
|
||||||
local_version=$(get_local_version "${main_script_filename}")
|
|
||||||
|
|
||||||
status=""
|
|
||||||
if [[ -f "${main_script_filename}" ]]; then
|
|
||||||
status=" (Installed: v${local_version})"
|
|
||||||
# Compare versions
|
|
||||||
if is_version_greater "$remote_version" "$local_version"; then
|
|
||||||
status+=" [Update available: v${remote_version}]"
|
|
||||||
fi
|
fi
|
||||||
fi
|
|
||||||
options+=("${key}${status}")
|
|
||||||
done
|
|
||||||
echo
|
|
||||||
|
|
||||||
options+=("Quit") # Add a Quit option to the menu.
|
|
||||||
|
|
||||||
# Set the prompt for the select menu.
|
|
||||||
PS3="Please enter your choice: "
|
|
||||||
|
|
||||||
# Display the menu and handle user input.
|
|
||||||
# Setting COLUMNS=1 forces the select menu to display vertically.
|
|
||||||
COLUMNS=1
|
|
||||||
select choice_with_status in "${options[@]}"; do
|
|
||||||
# Strip the status message from the choice for key lookup
|
|
||||||
choice=$(echo "${choice_with_status}" | sed 's/ (.*//')
|
|
||||||
|
|
||||||
case "${choice}" in
|
|
||||||
"Quit")
|
|
||||||
log "👋" "Exiting."
|
|
||||||
break # Exit the select loop.
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
# Check if the user's choice is a valid package name.
|
|
||||||
if [[ -n "${SCRIPT_PACKAGES[$choice]}" ]]; then
|
|
||||||
echo
|
echo
|
||||||
log "⬇️" "Processing package: '${choice}'..."
|
log "⬇️" "Processing package: '${choice}'..."
|
||||||
|
|
||||||
@@ -164,7 +97,7 @@ select choice_with_status in "${options[@]}"; do
|
|||||||
log "->" "Downloading '${filename}'..."
|
log "->" "Downloading '${filename}'..."
|
||||||
if curl -fsSL -o "${filename}" "${url}"; then
|
if curl -fsSL -o "${filename}" "${url}"; then
|
||||||
log "✅" "Successfully downloaded '${filename}'."
|
log "✅" "Successfully downloaded '${filename}'."
|
||||||
if [[ "${filename}" == *.sh ]]; then
|
if [[ "${filename}" == *.sh || "${filename}" == *.bash ]]; then
|
||||||
chmod +x "${filename}"
|
chmod +x "${filename}"
|
||||||
log "🤖" "Made '${filename}' executable."
|
log "🤖" "Made '${filename}' executable."
|
||||||
fi
|
fi
|
||||||
@@ -173,14 +106,116 @@ select choice_with_status in "${options[@]}"; do
|
|||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
echo
|
log "📦" "Package processing complete for '${choice}'."
|
||||||
log "📦" "Package processing complete."
|
}
|
||||||
break
|
|
||||||
else
|
|
||||||
echo "Invalid selection. Please try again."
|
# --- Main Logic ---
|
||||||
|
|
||||||
|
# Generate a unique temporary filename with a timestamp.
|
||||||
|
conf_file="packages.conf.$(date +%Y%m%d%H%M%S)"
|
||||||
|
|
||||||
|
# Set up a trap to delete the temporary file on exit.
|
||||||
|
trap 'rm -f "${conf_file}"' EXIT
|
||||||
|
|
||||||
|
# Download the configuration file.
|
||||||
|
log "🔄" "Downloading configuration file..."
|
||||||
|
if ! curl -fsSL -o "${conf_file}" "https://git.technopunk.space/tomi/Scripts/raw/branch/main/packages.conf"; then
|
||||||
|
log "❌" "Error: Failed to download packages.conf. Exiting."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
log "✅" "Configuration file downloaded successfully."
|
||||||
|
|
||||||
|
# Source the configuration file to load the SCRIPT_PACKAGES associative array.
|
||||||
|
source "${conf_file}"
|
||||||
|
|
||||||
|
# --- Update Check & User Interface ---
|
||||||
|
|
||||||
|
# Create an array of options from the package names.
|
||||||
|
# We will modify this array to show installation and update status.
|
||||||
|
declare -a options
|
||||||
|
package_keys=("${!SCRIPT_PACKAGES[@]}")
|
||||||
|
|
||||||
|
log "🔎" "Checking for updates..."
|
||||||
|
for key in "${package_keys[@]}"; do
|
||||||
|
# The config format is now "VERSION|URL1 URL2..."
|
||||||
|
config_value="${SCRIPT_PACKAGES[$key]}"
|
||||||
|
remote_version=$(echo "${config_value}" | cut -d'|' -f1)
|
||||||
|
|
||||||
|
# Get just the URLs and assume the first URL is the main script to check.
|
||||||
|
urls=$(echo "${config_value}" | cut -d'|' -f2-)
|
||||||
|
read -r -a url_array <<< "$urls"
|
||||||
|
main_script_filename=$(basename "${url_array[0]}")
|
||||||
|
|
||||||
|
# Get the local version of the main script file.
|
||||||
|
local_version=$(get_local_version "${main_script_filename}")
|
||||||
|
|
||||||
|
status=""
|
||||||
|
if [[ -f "${main_script_filename}" ]]; then
|
||||||
|
status=" (Installed: v${local_version})"
|
||||||
|
# Compare versions
|
||||||
|
if is_version_greater "$remote_version" "$local_version"; then
|
||||||
|
status+=" [Update available: v${remote_version}]"
|
||||||
fi
|
fi
|
||||||
;;
|
fi
|
||||||
esac
|
options+=("${key}${status}")
|
||||||
done
|
done
|
||||||
|
|
||||||
|
options+=("Quit") # Add a Quit option to the menu.
|
||||||
|
|
||||||
|
# --- User Interaction ---
|
||||||
|
|
||||||
|
# Manually display the options with numbers.
|
||||||
|
echo
|
||||||
|
echo "-------------------------------------"
|
||||||
|
echo " Script Downloader "
|
||||||
|
echo "-------------------------------------"
|
||||||
|
for i in "${!options[@]}"; do
|
||||||
|
printf "%d) %s\n" "$((i+1))" "${options[$i]}"
|
||||||
|
done
|
||||||
|
echo
|
||||||
|
|
||||||
|
# Prompt the user for one or more choices.
|
||||||
|
read -p "Please enter your choice(s) (e.g., 1 3 4), or press Enter to quit: " -r -a user_choices
|
||||||
|
|
||||||
|
# If no choices are made, exit gracefully.
|
||||||
|
if [ ${#user_choices[@]} -eq 0 ]; then
|
||||||
|
log "👋" "No selection made. Exiting."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Loop through the user's selections and process each one.
|
||||||
|
for choice_num in "${user_choices[@]}"; do
|
||||||
|
# Validate that the input is a number.
|
||||||
|
if ! [[ "$choice_num" =~ ^[0-9]+$ ]]; then
|
||||||
|
log "⚠️" "Skipping invalid input: '${choice_num}'. Not a number."
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Convert selection number to array index (0-based).
|
||||||
|
index=$((choice_num - 1))
|
||||||
|
|
||||||
|
# Validate that the index is within the bounds of the options array.
|
||||||
|
if [[ -z "${options[$index]}" ]]; then
|
||||||
|
log "⚠️" "Skipping invalid choice: '${choice_num}'. Out of range."
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Get the choice text from the array.
|
||||||
|
choice_with_status="${options[$index]}"
|
||||||
|
|
||||||
|
# Strip the status message to get the package key.
|
||||||
|
choice=$(echo "${choice_with_status}" | sed 's/ (.*//')
|
||||||
|
|
||||||
|
# Handle the "Quit" option.
|
||||||
|
if [[ "${choice}" == "Quit" ]]; then
|
||||||
|
log "👋" "Quit selected. Exiting now."
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Process the selected package.
|
||||||
|
process_package "${choice}"
|
||||||
|
done
|
||||||
|
|
||||||
|
echo
|
||||||
|
log "🏁" "All selected packages have been processed."
|
||||||
|
|||||||
Reference in New Issue
Block a user