1
0
mirror of https://github.com/tw93/Mole.git synced 2026-02-04 19:44:44 +00:00
Files
Mole/lib/batch_uninstall.sh
2025-10-03 13:56:53 +08:00

253 lines
9.4 KiB
Bash
Executable File

#!/bin/bash
# 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 ""
echo -e "${BLUE}📋 Analyzing selected applications...${NC}"
for selected_app in "${selected_apps[@]}"; do
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
if [[ $total_estimated_size -gt 1048576 ]]; then
local size_display=$(echo "$total_estimated_size" | awk '{printf "%.2fGB", $1/1024/1024}')
elif [[ $total_estimated_size -gt 1024 ]]; then
local size_display=$(echo "$total_estimated_size" | awk '{printf "%.1fMB", $1/1024}')
else
local size_display="${total_estimated_size}KB"
fi
# Request sudo access if needed (do this before confirmation)
if [[ ${#sudo_apps[@]} -gt 0 ]]; then
echo ""
echo -e "${YELLOW}🔐 Admin privileges required for: ${BLUE}${sudo_apps[*]}${NC}"
echo -e "${BLUE}You will be prompted for your password before proceeding...${NC}"
if ! sudo -v; then
log_error "Administrator privileges required but not granted"
return 1
fi
# Keep sudo alive during the process
(while true; do sudo -n true; sleep 60; kill -0 "$$" || exit; done 2>/dev/null) &
local sudo_keepalive_pid=$!
# Append keepalive cleanup to existing traps without overriding them
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
echo ""
echo -e "${YELLOW}📦 Will remove ${BLUE}${#selected_apps[@]}${YELLOW} applications, free ${GREEN}$size_display${NC}"
if [[ ${#running_apps[@]} -gt 0 ]]; then
echo -e "${YELLOW}⚠️ Running apps will be force-quit: ${RED}${running_apps[*]}${NC}"
fi
echo ""
printf "%b" "${BLUE}Press ENTER to confirm, or ESC/q to cancel:${NC} "
local confirm_key=""
IFS= read -r -s -n1 confirm_key || confirm_key=""
if [[ "$confirm_key" == $'\e' ]]; then
while IFS= read -r -s -n1 -t 0 rest; do
[[ -z "$rest" || "$rest" == $'\n' ]] && break
done
fi
echo ""
local cancel=false
case "$confirm_key" in
""|$'\n'|$'\r') ;;
$'\e'|"q"|"Q") cancel=true ;;
*) cancel=true ;;
esac
if [[ "$cancel" == true ]]; then
log_info "Uninstallation cancelled by user"
# Clean up sudo keepalive if it was started
if [[ -n "${sudo_keepalive_pid:-}" ]]; then
kill "$sudo_keepalive_pid" 2>/dev/null || true
fi
return 0
fi
echo -e "${PURPLE}⚡ Starting uninstallation in 3 seconds...${NC} ${YELLOW}(Press Ctrl+C to abort)${NC}"
sleep 1 && echo -e "${PURPLE}${BLUE}2${PURPLE}...${NC}"
sleep 1 && echo -e "${PURPLE}${BLUE}1${PURPLE}...${NC}"
sleep 1
echo -e "${GREEN}✨ Let's go!${NC}"
# Force quit running apps first (batch)
if [[ ${#running_apps[@]} -gt 0 ]]; then
echo ""
log_info "Force quitting running applications..."
for app_name in "${running_apps[@]}"; do
echo " • Quitting $app_name..."
pkill -f "$app_name" 2>/dev/null || true
done
echo " • Waiting 3 seconds for apps to close..."
sleep 3
fi
# Perform uninstallations without individual confirmations
echo ""
log_info "Starting batch uninstallation..."
local success_count=0
local failed_count=0
for detail in "${app_details[@]}"; do
IFS='|' read -r app_name app_path bundle_id total_kb encoded_files <<< "$detail"
# Decode the related files list
local related_files=$(echo "$encoded_files" | base64 -d)
echo -e "${YELLOW}🗑️ Uninstalling: ${BLUE}$app_name${NC}"
# Check if app is still running (even after force quit)
if pgrep -f "$app_name" >/dev/null 2>&1; then
echo -e " ${YELLOW}⚠️${NC} App is still running, attempting force kill..."
pkill -9 -f "$app_name" 2>/dev/null || true
sleep 2
if pgrep -f "$app_name" >/dev/null 2>&1; then
echo -e " ${RED}${NC} Failed to remove $app_name"
echo -e " ${YELLOW}Reason: Application is still running and cannot be terminated${NC}"
((failed_count++))
continue
fi
fi
# Check if app requires admin privileges to delete
local needs_sudo=false
if [[ ! -w "$(dirname "$app_path")" ]] || [[ "$(stat -f%Su "$app_path" 2>/dev/null)" == "root" ]]; then
needs_sudo=true
fi
# Remove the application with appropriate permissions
local removal_success=false
local error_msg=""
if [[ "$needs_sudo" == "true" ]]; then
if sudo rm -rf "$app_path" 2>/dev/null; then
removal_success=true
echo -e " ${GREEN}${NC} Removed application"
else
error_msg="Failed to remove with sudo (check permissions or SIP protection)"
fi
else
if rm -rf "$app_path" 2>/dev/null; then
removal_success=true
echo -e " ${GREEN}${NC} Removed application"
else
error_msg="Failed to remove (check if app is running or protected)"
fi
fi
if [[ "$removal_success" == "true" ]]; then
# Remove related files
local files_removed=0
while IFS= read -r file; do
if [[ -n "$file" && -e "$file" ]]; then
if rm -rf "$file" 2>/dev/null; then
((files_removed++))
fi
fi
done <<< "$related_files"
if [[ $files_removed -gt 0 ]]; then
echo -e " ${GREEN}${NC} Cleaned $files_removed related files"
fi
((total_size_freed += total_kb))
((success_count++))
((files_cleaned++))
((total_items++))
else
echo -e " ${RED}${NC} Failed to remove $app_name"
if [[ -n "$error_msg" ]]; then
echo -e " ${YELLOW}Reason: $error_msg${NC}"
fi
((failed_count++))
fi
done
# Show final summary
echo ""
echo "===================================================================="
echo "🎉 UNINSTALLATION COMPLETE!"
if [[ $success_count -gt 0 ]]; then
if [[ $total_size_freed -gt 1048576 ]]; then
local freed_display=$(echo "$total_size_freed" | awk '{printf "%.2fGB", $1/1024/1024}')
elif [[ $total_size_freed -gt 1024 ]]; then
local freed_display=$(echo "$total_size_freed" | awk '{printf "%.1fMB", $1/1024}')
else
local freed_display="${total_size_freed}KB"
fi
echo "🗑️ Apps uninstalled: $success_count | Space freed: ${GREEN}${freed_display}${NC}"
else
echo "🗑️ No applications were uninstalled"
fi
if [[ $failed_count -gt 0 ]]; then
echo -e "${RED}⚠️ Failed to uninstall: $failed_count${NC}"
fi
echo "===================================================================="
if [[ $failed_count -gt 0 ]]; then
log_warning "$failed_count applications failed to uninstall"
fi
# 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))
}