mirror of
https://github.com/tw93/Mole.git
synced 2026-02-04 19:44:44 +00:00
191 lines
7.5 KiB
Bash
Executable File
191 lines
7.5 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
set -euo pipefail
|
|
|
|
# Ensure common.sh is loaded
|
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)"
|
|
[[ -z "${MOLE_COMMON_LOADED:-}" ]] && source "$SCRIPT_DIR/lib/common.sh"
|
|
|
|
# Batch uninstall functionality with minimal confirmations
|
|
# Replaces the overly verbose individual confirmation approach
|
|
# Note: find_app_files() and calculate_total_size() functions now in lib/common.sh
|
|
|
|
# Batch uninstall with single confirmation
|
|
batch_uninstall_applications() {
|
|
local total_size_freed=0
|
|
|
|
if [[ ${#selected_apps[@]} -eq 0 ]]; then
|
|
log_warning "No applications selected for uninstallation"
|
|
return 0
|
|
fi
|
|
|
|
# Pre-process: Check for running apps and calculate total impact
|
|
local -a running_apps=()
|
|
local -a sudo_apps=()
|
|
local total_estimated_size=0
|
|
local -a app_details=()
|
|
|
|
echo ""
|
|
# Silent analysis without spinner output (avoid visual flicker)
|
|
for selected_app in "${selected_apps[@]}"; do
|
|
[[ -z "$selected_app" ]] && continue
|
|
IFS='|' read -r epoch app_path app_name bundle_id size last_used <<< "$selected_app"
|
|
|
|
# Check if app is running
|
|
if pgrep -f "$app_name" >/dev/null 2>&1; then
|
|
running_apps+=("$app_name")
|
|
fi
|
|
|
|
# Check if app requires sudo to delete
|
|
if [[ ! -w "$(dirname "$app_path")" ]] || [[ "$(stat -f%Su "$app_path" 2>/dev/null)" == "root" ]]; then
|
|
sudo_apps+=("$app_name")
|
|
fi
|
|
|
|
# Calculate size for summary
|
|
local app_size_kb=$(du -sk "$app_path" 2>/dev/null | awk '{print $1}' || echo "0")
|
|
local related_files=$(find_app_files "$bundle_id" "$app_name")
|
|
local related_size_kb=$(calculate_total_size "$related_files")
|
|
local total_kb=$((app_size_kb + related_size_kb))
|
|
((total_estimated_size += total_kb))
|
|
|
|
# Store details for later use
|
|
# Base64 encode related_files to handle multi-line data safely
|
|
local encoded_files=$(echo "$related_files" | base64)
|
|
app_details+=("$app_name|$app_path|$bundle_id|$total_kb|$encoded_files")
|
|
done
|
|
|
|
# Format size display (convert KB to bytes for bytes_to_human())
|
|
local size_display=$(bytes_to_human "$((total_estimated_size * 1024))")
|
|
|
|
# Request sudo access if needed (do this before confirmation)
|
|
if [[ ${#sudo_apps[@]} -gt 0 ]]; then
|
|
# Check if sudo is already cached
|
|
if sudo -n true 2>/dev/null; then
|
|
echo "◎ Admin access confirmed for: ${sudo_apps[*]}"
|
|
else
|
|
echo -n "◎ Admin required for: ${sudo_apps[*]}. "
|
|
if ! sudo -v; then
|
|
echo ""
|
|
log_error "Admin access denied"
|
|
return 1
|
|
fi
|
|
echo "✓ Granted"
|
|
fi
|
|
echo "◎ Gathering targets..."
|
|
(while true; do sudo -n true; sleep 60; kill -0 "$$" || exit; done 2>/dev/null) &
|
|
local sudo_keepalive_pid=$!
|
|
local _trap_cleanup_cmd="kill $sudo_keepalive_pid 2>/dev/null || true; wait $sudo_keepalive_pid 2>/dev/null || true"
|
|
for signal in EXIT INT TERM; do
|
|
local existing_trap; existing_trap=$(trap -p "$signal" | awk -F"'" '{print $2}')
|
|
if [[ -n "$existing_trap" ]]; then
|
|
trap "$existing_trap; $_trap_cleanup_cmd" "$signal"
|
|
else
|
|
trap "$_trap_cleanup_cmd" "$signal"
|
|
fi
|
|
done
|
|
fi
|
|
|
|
# Show summary and get batch confirmation
|
|
local app_total=${#selected_apps[@]}
|
|
if [[ ${#running_apps[@]} -gt 0 ]]; then
|
|
echo -n "${BLUE}◎ Remove ${app_total} app(s) (${size_display}) | Quit: ${running_apps[*]} | Enter=go / ESC=q:${NC} "
|
|
else
|
|
echo -n "${BLUE}◎ Remove ${app_total} app(s) (${size_display}) | Enter=go / ESC=q:${NC} "
|
|
fi
|
|
IFS= read -r -s -n1 key || key=""
|
|
case "$key" in
|
|
$'\e'|q|Q) echo ""; return 0 ;;
|
|
""|$'\n'|$'\r'|y|Y) echo "" ;;
|
|
*) echo ""; return 0 ;;
|
|
esac
|
|
|
|
echo -n "◎ Starting in 3s... 3"; sleep 1; echo -ne "\r◎ Starting in 3s... 2"; sleep 1; echo -ne "\r◎ Starting in 3s... 1"; sleep 1
|
|
echo -ne "\r\033[K"
|
|
if [[ -t 1 ]]; then start_inline_spinner "Uninstalling apps..."; fi
|
|
|
|
# Force quit running apps first (batch)
|
|
if [[ ${#running_apps[@]} -gt 0 ]]; then
|
|
pkill -f "${running_apps[0]}" 2>/dev/null || true
|
|
for app_name in "${running_apps[@]:1}"; do pkill -f "$app_name" 2>/dev/null || true; done
|
|
sleep 2
|
|
if pgrep -f "${running_apps[0]}" >/dev/null 2>&1; then sleep 1; fi
|
|
fi
|
|
|
|
# Perform uninstallations (compact output)
|
|
if [[ -t 1 ]]; then stop_inline_spinner; fi
|
|
echo ""
|
|
local success_count=0 failed_count=0
|
|
local -a failed_items=()
|
|
for detail in "${app_details[@]}"; do
|
|
IFS='|' read -r app_name app_path bundle_id total_kb encoded_files <<< "$detail"
|
|
local related_files=$(echo "$encoded_files" | base64 -d)
|
|
local reason=""
|
|
local needs_sudo=false
|
|
[[ ! -w "$(dirname "$app_path")" || "$(stat -f%Su "$app_path" 2>/dev/null)" == "root" ]] && needs_sudo=true
|
|
if ! force_kill_app "$app_name"; then
|
|
reason="still running"
|
|
fi
|
|
if [[ -z "$reason" ]]; then
|
|
if [[ "$needs_sudo" == true ]]; then
|
|
sudo rm -rf "$app_path" 2>/dev/null || reason="remove failed"
|
|
else
|
|
rm -rf "$app_path" 2>/dev/null || reason="remove failed"
|
|
fi
|
|
fi
|
|
if [[ -z "$reason" ]]; then
|
|
local files_removed=0
|
|
while IFS= read -r file; do
|
|
[[ -n "$file" && -e "$file" ]] || continue
|
|
rm -rf "$file" 2>/dev/null && ((files_removed++)) || true
|
|
done <<< "$related_files"
|
|
((total_size_freed += total_kb))
|
|
((success_count++))
|
|
((files_cleaned++))
|
|
((total_items++))
|
|
printf " ${GREEN}OK${NC} %-20s%s\n" "$app_name" $([[ $files_removed -gt 0 ]] && echo "+$files_removed" )
|
|
else
|
|
((failed_count++))
|
|
failed_items+=("$app_name:$reason")
|
|
fi
|
|
done
|
|
|
|
# Summary
|
|
local freed_display="0B"
|
|
if [[ $total_size_freed -gt 0 ]]; then
|
|
local freed_kb=$total_size_freed
|
|
if [[ $freed_kb -ge 1048576 ]]; then
|
|
freed_display=$(echo "$freed_kb" | awk '{printf "%.2fGB", $1/1024/1024}')
|
|
elif [[ $freed_kb -ge 1024 ]]; then
|
|
freed_display=$(echo "$freed_kb" | awk '{printf "%.1fMB", $1/1024}')
|
|
else
|
|
freed_display="${freed_kb}KB"
|
|
fi
|
|
fi
|
|
local bar="================================================================================"
|
|
echo ""
|
|
echo "$bar"
|
|
if [[ $failed_count -gt 0 ]]; then
|
|
echo -e "🚀 Removed: ${GREEN}$success_count${NC} | Failed: ${RED}$failed_count${NC} | Freed: ${GREEN}$freed_display${NC}"
|
|
if [[ $failed_count -eq 1 ]]; then
|
|
local first="${failed_items[0]}"
|
|
local name=${first%%:*}
|
|
local reason=${first#*:}
|
|
echo "😉 ${name} $(map_uninstall_reason "$reason")"
|
|
else
|
|
local joined="${failed_items[*]}"; echo "😉 Failures: $joined"
|
|
fi
|
|
else
|
|
echo -e "🚀 Removed: ${GREEN}$success_count${NC} | Failed: ${RED}$failed_count${NC} | Freed: ${GREEN}$freed_display${NC}"
|
|
fi
|
|
echo "$bar"
|
|
|
|
# Clean up sudo keepalive if it was started
|
|
if [[ -n "${sudo_keepalive_pid:-}" ]]; then
|
|
kill "$sudo_keepalive_pid" 2>/dev/null || true
|
|
wait "$sudo_keepalive_pid" 2>/dev/null || true
|
|
fi
|
|
|
|
((total_size_cleaned += total_size_freed))
|
|
unset failed_items
|
|
}
|