Files
Linux-Server-Management-Too…/modules/performance/php-optimizer.sh
T
cschantz 13d7054aa1 Fix critical bugs and add domain-by-domain batch analyzer
- Fix line 63 in php-analyzer.sh: Add default value for count variable (integer comparison error)
- Fix line 655 in php-analyzer.sh: Add default value for memory_error_count (integer comparison error)
- Fix line 396 in php-scanner.sh: Replace unsafe eval with safe getent passwd lookup
- Add php-ui.sh: User interface and menu system (18KB, 25+ functions)
- Add php-scanner.sh: Server enumeration system (17KB, 18 functions)
- Add php-action-executor.sh: Optimization execution system (17KB, 20 functions)
- Add php-server-manager.sh: Orchestration framework (21KB, 7 functions)
- Add php-fpm-batch-analyzer.sh: One-shot diagnostic script showing current vs recommended max_children, memory impact, and optimization potential
- Add comprehensive test suite (24 tests)

These fixes resolve "integer expression expected" errors during domain analysis.
Batch analyzer enables users to see domain-by-domain optimization opportunities before applying changes.

Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
2026-02-17 22:43:49 -05:00

1875 lines
72 KiB
Bash
Executable File

#!/bin/bash
# PHP & Server Performance Optimizer
# Interactive tool for analyzing and optimizing PHP-FPM configurations
# Part of Server Toolkit - Phase 3: Main Interactive Script
# Source required libraries
PHP_TOOLKIT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && cd ../.. && pwd)"
source "$PHP_TOOLKIT_DIR/lib/common-functions.sh" || { echo "ERROR: common-functions.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/system-detect.sh" || { echo "ERROR: system-detect.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/user-manager.sh" || { echo "ERROR: user-manager.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/php-detector.sh" || { echo "ERROR: php-detector.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/php-analyzer.sh" || { echo "ERROR: php-analyzer.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/php-config-manager.sh" || { echo "ERROR: php-config-manager.sh not found"; exit 1; }
source "$PHP_TOOLKIT_DIR/lib/php-calculator-improved.sh" || { echo "ERROR: php-calculator-improved.sh not found"; exit 1; }
# Phase 3 Modular Architecture - NEW (optional but recommended for batch operations)
source "$PHP_TOOLKIT_DIR/lib/php-ui.sh" 2>/dev/null || true
source "$PHP_TOOLKIT_DIR/lib/php-scanner.sh" 2>/dev/null || true
source "$PHP_TOOLKIT_DIR/lib/php-action-executor.sh" 2>/dev/null || true
source "$PHP_TOOLKIT_DIR/lib/php-server-manager.sh" 2>/dev/null || true
# Color codes (using safe echo -e)
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
MAGENTA='\033[0;35m'
CYAN='\033[0;36m'
WHITE='\033[1;37m'
BOLD='\033[1m'
NC='\033[0m' # No Color
# Safe color echo function
cecho() {
echo -e "$@"
}
# ============================================================================
# BANNER & DISPLAY FUNCTIONS
# ============================================================================
show_banner() {
clear
cecho "${CYAN}╔══════════════════════════════════════════════════════════════════════╗${NC}"
cecho "${CYAN}${WHITE} PHP & SERVER PERFORMANCE OPTIMIZER ${CYAN}${NC}"
cecho "${CYAN}╚══════════════════════════════════════════════════════════════════════╝${NC}"
echo ""
}
show_main_menu() {
cecho "${WHITE}${BOLD}MAIN MENU${NC}"
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
cecho " ${GREEN}1${NC}) Analyze Single Domain"
cecho " ${GREEN}2${NC}) Analyze All Domains (Server-Wide)"
cecho " ${GREEN}3${NC}) Quick Health Check (All Domains)"
cecho " ${GREEN}4${NC}) Optimize Domain PHP Settings"
cecho " ${GREEN}5${NC}) Optimize Server-Wide PHP Settings"
cecho " ${GREEN}6${NC}) View OPcache Statistics"
cecho " ${GREEN}7${NC}) View PHP-FPM Process Stats"
cecho " ${GREEN}8${NC}) Check for Configuration Issues"
cecho " ${GREEN}9${NC}) Check Server Memory Capacity (OOM Risk)"
echo ""
cecho " ${YELLOW}b${NC}) Backup Current Configurations"
cecho " ${YELLOW}r${NC}) Restore from Backup"
echo ""
cecho " ${RED}0${NC}) Exit"
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
}
# ============================================================================
# DOMAIN SELECTION
# ============================================================================
select_domain() {
local action="${1:-analyze}"
cecho "${WHITE}${BOLD}SELECT DOMAIN${NC}"
echo ""
# Get all users with domains
local users
users=$(list_all_users)
if [ -z "$users" ]; then
cecho "${RED}ERROR: No users found on system${NC}"
read -p "Press Enter to continue..."
return 1
fi
# Build domain list
declare -a domains
declare -A domain_to_user
while IFS= read -r username; do
local user_domains
user_domains=$(get_user_domains "$username")
while IFS= read -r domain; do
[ -z "$domain" ] && continue
domains+=("$domain")
domain_to_user["$domain"]="$username"
done <<< "$user_domains"
done <<< "$users"
if [ ${#domains[@]} -eq 0 ]; then
cecho "${RED}ERROR: No domains found on system${NC}"
read -p "Press Enter to continue..."
return 1
fi
# Display numbered list
cecho "${CYAN}Available domains:${NC}"
echo ""
local index=1
for domain in "${domains[@]}"; do
local username="${domain_to_user[$domain]}"
local php_version
php_version=$(detect_php_version_for_domain "$username" "$domain" 2>/dev/null || echo "unknown")
printf " ${GREEN}%-3d${NC}) %-40s ${CYAN}[${username}]${NC} ${YELLOW}(${php_version})${NC}\n" "$index" "$domain"
index=$((index + 1))
done
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
# Validate domain selection with retry loop
while true; do
read -p "Select domain number (or 'q' to cancel): " selection
if [[ "$selection" == "q" ]]; then
return 1
fi
if ! [[ "$selection" =~ ^[0-9]+$ ]] || [ "$selection" -lt 1 ] || [ "$selection" -gt ${#domains[@]} ]; then
echo ""
cecho "${RED}Invalid selection. Please enter a number 1-${#domains[@]}${NC}"
echo ""
continue
fi
break
done
# Return selected domain and username
local selected_domain="${domains[$((selection - 1))]}"
local selected_user="${domain_to_user[$selected_domain]}"
echo "$selected_domain|$selected_user"
return 0
}
# ============================================================================
# OPTION 1: ANALYZE SINGLE DOMAIN
# ============================================================================
analyze_single_domain() {
show_banner
local selection
selection=$(select_domain "analyze")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}Analyzing: ${GREEN}$domain${WHITE} (user: ${CYAN}$username${WHITE})${NC}"
echo ""
# Run comprehensive analysis
analyze_domain_php "$username" "$domain"
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 2: ANALYZE ALL DOMAINS
# ============================================================================
analyze_all_domains() {
show_banner
cecho "${WHITE}${BOLD}SERVER-WIDE ANALYSIS${NC}"
echo ""
cecho "${YELLOW}This will analyze PHP configuration for ALL domains...${NC}"
echo ""
if ! confirm "Continue?"; then
return
fi
show_banner
cecho "${WHITE}${BOLD}Analyzing all domains...${NC}"
echo ""
# Get all users
local users
users=$(list_all_users)
local total_domains=0
local domains_with_issues=0
while IFS= read -r username; do
local user_domains
user_domains=$(get_user_domains "$username")
while IFS= read -r domain; do
[ -z "$domain" ] && continue
total_domains=$((total_domains + 1))
cecho "${CYAN}[$total_domains] Analyzing: ${WHITE}$domain${NC}"
# Detect issues
local issues
issues=$(detect_php_config_issues "$username" "$domain")
# Count issues by severity
local critical_count high_count medium_count low_count
critical_count=$(echo "$issues" | grep -c "^[^|]*|CRITICAL|" || true)
high_count=$(echo "$issues" | grep -c "^[^|]*|HIGH|" || true)
medium_count=$(echo "$issues" | grep -c "^[^|]*|MEDIUM|" || true)
low_count=$(echo "$issues" | grep -c "^[^|]*|LOW|" || true)
# Default to 0 if empty
critical_count=${critical_count:-0}
high_count=${high_count:-0}
medium_count=${medium_count:-0}
low_count=${low_count:-0}
local total_issues=$((critical_count + high_count + medium_count + low_count))
if [ "$total_issues" -gt 0 ]; then
domains_with_issues=$((domains_with_issues + 1))
# Build issue summary
local issue_summary=""
[ "$critical_count" -gt 0 ] && issue_summary+="${RED}$critical_count CRITICAL${NC}, "
[ "$high_count" -gt 0 ] && issue_summary+="${YELLOW}$high_count HIGH${NC}, "
[ "$medium_count" -gt 0 ] && issue_summary+="$medium_count MEDIUM, "
[ "$low_count" -gt 0 ] && issue_summary+="$low_count LOW, "
issue_summary=${issue_summary%, } # Remove trailing comma
cecho " ${RED}${NC} Issues found: $issue_summary"
# Show critical/high issues
while IFS='|' read -r issue_type severity message recommendation; do
[ -z "$issue_type" ] && continue
if [[ "$severity" == "CRITICAL" ]] || [[ "$severity" == "HIGH" ]]; then
cecho " ${RED}[$severity]${NC} $issue_type: $message"
fi
done <<< "$issues"
# Show what passed
local checks_passed=""
echo "$issues" | grep -q "max_children" || checks_passed+="max_children OK, "
echo "$issues" | grep -q "memory_exhausted" || checks_passed+="memory OK, "
echo "$issues" | grep -q "execution timeout" || checks_passed+="timeouts OK, "
if [ -n "$checks_passed" ]; then
checks_passed=${checks_passed%, }
cecho " ${GREEN}${NC} Checks passed: $checks_passed"
fi
else
cecho " ${GREEN}✓ All checks passed${NC} (max_children, memory, timeouts, config)"
fi
echo ""
done <<< "$user_domains"
done <<< "$users"
# Summary
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}SUMMARY${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
cecho " Total domains analyzed: ${WHITE}$total_domains${NC}"
cecho " Domains with issues: ${RED}$domains_with_issues${NC}"
cecho " Domains healthy: ${GREEN}$((total_domains - domains_with_issues))${NC}"
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 3: QUICK HEALTH CHECK
# ============================================================================
quick_health_check() {
show_banner
cecho "${WHITE}${BOLD}QUICK HEALTH CHECK${NC}"
echo ""
# Get all users
local users
users=$(list_all_users)
declare -A issue_counts
issue_counts["CRITICAL"]=0
issue_counts["HIGH"]=0
issue_counts["MEDIUM"]=0
issue_counts["LOW"]=0
local total_domains=0
while IFS= read -r username; do
local user_domains
user_domains=$(get_user_domains "$username")
while IFS= read -r domain; do
[ -z "$domain" ] && continue
total_domains=$((total_domains + 1))
# Detect issues
local issues
issues=$(detect_php_config_issues "$username" "$domain")
# Count by severity
while IFS='|' read -r issue_type severity message recommendation; do
[ -z "$severity" ] && continue
issue_counts["$severity"]=$((issue_counts["$severity"] + 1))
done <<< "$issues"
done <<< "$user_domains"
done <<< "$users"
# Display results
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}HEALTH CHECK RESULTS${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
cecho " Total Domains: ${WHITE}$total_domains${NC}"
echo ""
cecho " ${RED}CRITICAL${NC} issues: ${issue_counts["CRITICAL"]}"
cecho " ${YELLOW}HIGH${NC} issues: ${issue_counts["HIGH"]}"
cecho " ${BLUE}MEDIUM${NC} issues: ${issue_counts["MEDIUM"]}"
cecho " ${GREEN}LOW${NC} issues: ${issue_counts["LOW"]}"
echo ""
# Overall health score
local health_score
health_score=$((100 - (issue_counts["CRITICAL"] * 20) - (issue_counts["HIGH"] * 10) - (issue_counts["MEDIUM"] * 5) - (issue_counts["LOW"] * 2)))
[ "$health_score" -lt 0 ] && health_score=0
if [ "$health_score" -ge 90 ]; then
cecho " Overall Health: ${GREEN}${BOLD}$health_score/100 - EXCELLENT${NC}"
elif [ "$health_score" -ge 70 ]; then
cecho " Overall Health: ${YELLOW}${BOLD}$health_score/100 - GOOD${NC}"
elif [ "$health_score" -ge 50 ]; then
cecho " Overall Health: ${YELLOW}${BOLD}$health_score/100 - FAIR${NC}"
else
cecho " Overall Health: ${RED}${BOLD}$health_score/100 - POOR${NC}"
fi
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 4: OPTIMIZE DOMAIN
# ============================================================================
optimize_domain() {
show_banner
local selection
selection=$(select_domain "optimize")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}OPTIMIZE: ${GREEN}$domain${NC}"
echo ""
# Detect issues
cecho "${YELLOW}Detecting issues...${NC}"
local issues
issues=$(detect_php_config_issues "$username" "$domain")
# Display issues
local has_issues=false
while IFS='|' read -r issue_type severity message recommendation; do
[ -z "$issue_type" ] && continue
if [ "$issue_type" != "NONE" ]; then
has_issues=true
case "$severity" in
CRITICAL)
cecho "${RED}[CRITICAL]${NC} $message"
;;
HIGH)
cecho "${YELLOW}[HIGH]${NC} $message"
;;
MEDIUM)
cecho "${BLUE}[MEDIUM]${NC} $message"
;;
LOW)
cecho "${GREEN}[LOW]${NC} $message"
;;
esac
cecho " ${CYAN}${NC} $recommendation"
echo ""
fi
done <<< "$issues"
if [ "$has_issues" = false ]; then
cecho "${GREEN}${BOLD}✓ No issues detected - configuration is optimal!${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
# Check server-wide memory capacity first
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
cecho "${WHITE}${BOLD}SERVER MEMORY ANALYSIS${NC}"
echo ""
local capacity_result
capacity_result=$(calculate_server_memory_capacity 2>&1)
local total_required_mb total_ram_mb percentage status
total_required_mb=$(echo "$capacity_result" | head -1 | cut -d'|' -f1)
total_ram_mb=$(echo "$capacity_result" | head -1 | cut -d'|' -f2)
percentage=$(echo "$capacity_result" | head -1 | cut -d'|' -f3)
status=$(echo "$capacity_result" | head -1 | cut -d'|' -f4)
cecho " Total Server RAM: ${WHITE}${total_ram_mb}MB${NC}"
cecho " Current FPM Capacity: ${WHITE}${total_required_mb}MB${NC} (${percentage}% of RAM)"
case "$status" in
CRITICAL)
cecho " Server Status: ${RED}${BOLD}CRITICAL${NC} - Server at risk of OOM!"
;;
WARNING)
cecho " Server Status: ${YELLOW}${BOLD}WARNING${NC} - High memory pressure"
;;
CAUTION)
cecho " Server Status: ${YELLOW}CAUTION${NC} - Approaching limits"
;;
HEALTHY)
cecho " Server Status: ${GREEN}HEALTHY${NC} - Sufficient headroom"
;;
esac
echo ""
# Check for max_children errors
cecho "${WHITE}${BOLD}MAX_CHILDREN ERROR ANALYSIS${NC}"
echo ""
local max_children_errors
max_children_errors=$(analyze_max_children_errors "$username" 7)
local error_count last_error_time
error_count=$(echo "$max_children_errors" | cut -d'|' -f1)
last_error_time=$(echo "$max_children_errors" | cut -d'|' -f2)
if [ "$error_count" -gt 0 ]; then
cecho " ${RED}${NC} Found ${RED}${BOLD}$error_count${NC} max_children errors in last 7 days"
cecho " ${YELLOW}Last error: $last_error_time${NC}"
cecho " ${CYAN}${NC} This domain is hitting process limits and rejecting requests!"
else
cecho " ${GREEN}${NC} No max_children errors in last 7 days"
fi
echo ""
# Get optimization recommendations
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
cecho "${WHITE}${BOLD}RECOMMENDED OPTIMIZATIONS${NC}"
echo ""
# Get total system memory for improved calculation
local total_ram_mb
total_ram_mb=$(free -m | awk '/^Mem:/ {print $2}')
# IMPROVED: Calculate using new algorithm
local improved_result
improved_result=$(calculate_optimal_php_settings "$username" "$total_ram_mb")
local improved_max_children improved_pm_mode improved_min_spare improved_max_spare improved_reason
improved_max_children=$(get_field "$improved_result" 1)
improved_pm_mode=$(get_field "$improved_result" 2)
improved_min_spare=$(get_field "$improved_result" 3)
improved_max_spare=$(get_field "$improved_result" 4)
improved_reason=$(get_field "$improved_result" 5)
# OLD: Calculate using legacy algorithm (for comparison)
local optimal_result
optimal_result=$(calculate_optimal_max_children "$username" 1024)
local legacy_max_children legacy_reason
legacy_max_children=$(echo "$optimal_result" | cut -d'|' -f1)
legacy_reason=$(echo "$optimal_result" | cut -d'|' -f2)
# Get current max_children
local pool_config
pool_config=$(find_fpm_pool_config "$username" "$domain")
# Track available optimizations
declare -A opt_available
declare -A opt_description
local opt_count=0
local current_max_children current_pm_mode
if [ -n "$pool_config" ] && [ -f "$pool_config" ]; then
current_max_children=$(grep "^pm.max_children" "$pool_config" | awk -F'=' '{print $2}' | tr -d ' ')
current_pm_mode=$(grep "^pm =" "$pool_config" | awk -F'=' '{print $2}' | tr -d ' ')
if [ -n "$current_max_children" ] && [ "$improved_max_children" -ne "$current_max_children" ]; then
opt_count=$((opt_count + 1))
opt_available["max_children"]="true"
opt_description["max_children"]="Adjust pm.max_children from $current_max_children to $improved_max_children"
# Display comprehensive recommendation
cecho "${GREEN}$opt_count.${NC} Adjust ${BOLD}pm.max_children${NC} from ${RED}$current_max_children${NC} to ${GREEN}$improved_max_children${NC}"
cecho " ${CYAN}Improved Algorithm:${NC} $improved_max_children (${improved_reason})"
cecho " ${YELLOW}Legacy Algorithm:${NC} $legacy_max_children (${legacy_reason})"
# Show comparison if different
if [ "$improved_max_children" -ne "$legacy_max_children" ]; then
local diff=$((improved_max_children - legacy_max_children))
if [ "$diff" -gt 0 ]; then
cecho " ${GREEN}✓ Improved: +$diff processes (safer)${NC}"
else
cecho " ${GREEN}✓ Improved: $diff processes (more efficient)${NC}"
fi
fi
echo ""
fi
# Recommend PM mode if different
if [ -n "$current_pm_mode" ] && [ "$current_pm_mode" != "$improved_pm_mode" ]; then
opt_count=$((opt_count + 1))
opt_available["pm_mode"]="true"
opt_description["pm_mode"]="Change pm mode from $current_pm_mode to $improved_pm_mode"
cecho "${GREEN}$opt_count.${NC} Change ${BOLD}pm${NC} mode from ${RED}$current_pm_mode${NC} to ${GREEN}$improved_pm_mode${NC}"
cecho " Recommended: $improved_pm_mode with min_spare=$improved_min_spare, max_spare=$improved_max_spare"
echo ""
fi
fi
# Check OPcache
local opcache_status
opcache_status=$(analyze_opcache_effectiveness "$username")
local opcache_state hit_rate opcache_rec
opcache_state=$(echo "$opcache_status" | cut -d'|' -f1)
hit_rate=$(echo "$opcache_status" | cut -d'|' -f2)
opcache_rec=$(echo "$opcache_status" | cut -d'|' -f5)
if [ "$opcache_state" = "DISABLED" ]; then
opt_count=$((opt_count + 1))
opt_available["opcache"]="true"
opt_description["opcache"]="Enable OPcache for 40-70% performance boost"
cecho "${GREEN}$opt_count.${NC} ${BOLD}Enable OPcache${NC} for 40-70% performance boost"
echo ""
else
# Check if hit_rate is numeric before using awk
if [ -n "$hit_rate" ] && [[ "$hit_rate" =~ ^[0-9]+$ ]]; then
local hit_rate_low=$(awk "BEGIN {print ($hit_rate < 90 ? 1 : 0)}" 2>/dev/null || echo 0)
if [ "$hit_rate_low" -eq 1 ]; then
opt_count=$((opt_count + 1))
opt_available["opcache"]="true"
opt_description["opcache"]="Increase opcache.memory_consumption (current hit rate: ${hit_rate}%)"
cecho "${GREEN}$opt_count.${NC} ${BOLD}Increase opcache.memory_consumption${NC} (current hit rate: ${hit_rate}%)"
echo ""
fi
fi
fi
# Check if there are any optimizations to apply
if [ "$opt_count" -eq 0 ]; then
cecho "${GREEN}${BOLD}✓ All settings are optimal - no changes needed!${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
# Ask user which optimizations to apply
cecho "${WHITE}${BOLD}SELECT OPTIMIZATIONS TO APPLY${NC}"
echo ""
cecho " ${GREEN}a${NC}) Apply all $opt_count recommendations"
[ "${opt_available[max_children]}" = "true" ] && cecho " ${GREEN}1${NC}) Apply max_children optimization only"
[ "${opt_available[opcache]}" = "true" ] && cecho " ${GREEN}2${NC}) Apply OPcache optimization only"
cecho " ${RED}n${NC}) Cancel - don't apply any changes"
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
# Validate optimization selection with retry loop
while true; do
read -p "Select option: " apply_choice
if ! [[ "$apply_choice" =~ ^[a1A2nN]$ ]]; then
echo ""
cecho "${RED}Invalid selection. Please enter a, 1, 2, or n${NC}"
echo ""
continue
fi
break
done
# Determine which optimizations to apply
local apply_max_children=false
local apply_opcache=false
# Convert to lowercase for consistent case matching
apply_choice=${apply_choice,,}
case "$apply_choice" in
a)
apply_max_children=${opt_available[max_children]:-false}
apply_opcache=${opt_available[opcache]:-false}
;;
1)
apply_max_children=${opt_available[max_children]:-false}
;;
2)
apply_opcache=${opt_available[opcache]:-false}
;;
n)
cecho "${YELLOW}Optimization cancelled - no changes made${NC}"
echo ""
read -p "Press Enter to continue..."
return
;;
esac
# Check if nothing was selected
if [ "$apply_max_children" = "false" ] && [ "$apply_opcache" = "false" ]; then
cecho "${YELLOW}No optimizations selected${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
# Create backup first
cecho "${CYAN}Creating backup before making changes...${NC}"
local backup_dir
backup_dir=$(backup_user_php_configs "$username" "$domain" 2>&1)
if [ $? -ne 0 ]; then
cecho "${RED}${BOLD}✗ Backup failed - aborting changes${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
cecho "${GREEN}✓ Backup created: $(basename "$backup_dir")${NC}"
echo ""
# Apply changes
cecho "${CYAN}Applying selected optimizations...${NC}"
echo ""
local changes_made=0
local changes_failed=0
# Apply max_children if selected (uses improved algorithm)
if [ "$apply_max_children" = "true" ]; then
if [ -n "$pool_config" ] && [ -f "$pool_config" ]; then
if [ -n "$improved_max_children" ] && [ -n "$current_max_children" ]; then
if modify_fpm_pool_setting "$pool_config" "pm.max_children" "$improved_max_children" >/dev/null 2>&1; then
cecho " ${GREEN}${NC} Set pm.max_children = $improved_max_children"
changes_made=$((changes_made + 1))
else
cecho " ${RED}${NC} Failed to set pm.max_children"
changes_failed=$((changes_failed + 1))
fi
fi
fi
fi
# Apply OPcache if selected
if [ "$apply_opcache" = "true" ]; then
# Note: OPcache settings are in php.ini, not FPM pool config
# This would require php.ini modification (not implemented yet)
cecho " ${YELLOW}${NC} OPcache optimization requires php.ini modification (not yet implemented)"
# TODO: Implement php.ini modification for OPcache
fi
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
if [ "$changes_made" -gt 0 ]; then
cecho "${GREEN}${BOLD}✓ Applied $changes_made optimization(s)${NC}"
if [ "$changes_failed" -gt 0 ]; then
cecho "${YELLOW}⚠ Failed to apply $changes_failed optimization(s)${NC}"
fi
echo ""
cecho "${YELLOW}Changes have been applied. Restart PHP-FPM for changes to take effect.${NC}"
echo ""
if confirm "Restart PHP-FPM now?"; then
# Detect PHP version
local php_version
php_version=$(detect_php_version_for_domain "$username" "$domain")
cecho "${CYAN}Restarting PHP-FPM ($php_version)...${NC}"
if reload_php_fpm "$php_version" >/dev/null 2>&1; then
cecho "${GREEN}✓ PHP-FPM reloaded successfully${NC}"
# Verify it's running
sleep 1
if verify_php_fpm_running "$php_version" >/dev/null 2>&1; then
cecho "${GREEN}✓ PHP-FPM is running${NC}"
else
cecho "${RED}✗ WARNING: PHP-FPM may not be running!${NC}"
cecho "${YELLOW}Run: systemctl status ea-php${php_version#ea-php}-php-fpm${NC}"
fi
else
cecho "${RED}✗ Failed to restart PHP-FPM${NC}"
cecho "${YELLOW}You may need to restart manually:${NC}"
cecho "${YELLOW}systemctl restart ea-php${php_version#ea-php}-php-fpm${NC}"
fi
else
cecho "${YELLOW}Skipping restart - remember to restart PHP-FPM manually!${NC}"
fi
echo ""
cecho "${CYAN}Backup location: ${WHITE}$(basename "$backup_dir")${NC}"
cecho "${CYAN}To rollback: Use Option 'r' (Restore from Backup)${NC}"
else
cecho "${YELLOW}No changes were applied${NC}"
fi
echo ""
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTIMIZE ALL DOMAINS (SERVER-WIDE)
# ============================================================================
optimize_all_domains() {
show_banner
cecho "${WHITE}${BOLD}SERVER-WIDE OPTIMIZATION${NC}"
echo ""
cecho "${YELLOW}This will analyze and optimize PHP settings for ALL domains on the server.${NC}"
echo ""
cecho "${CYAN}What will be optimized:${NC}"
cecho " • pm.max_children (based on memory analysis)"
cecho " • OPcache settings (enable if disabled)"
echo ""
cecho "${RED}${BOLD}WARNING:${NC} ${RED}This will modify PHP-FPM pool configurations server-wide!${NC}"
echo ""
if ! confirm "Continue with server-wide optimization?"; then
cecho "${YELLOW}Operation cancelled${NC}"
read -p "Press Enter to continue..."
return
fi
show_banner
cecho "${WHITE}${BOLD}Analyzing server capacity...${NC}"
echo ""
# First: Calculate server-wide memory capacity
cecho "${CYAN}Step 1: Checking server memory capacity...${NC}"
local capacity_result
capacity_result=$(calculate_server_memory_capacity 2>&1)
local total_required_mb total_ram_mb percentage status
total_required_mb=$(echo "$capacity_result" | head -1 | cut -d'|' -f1)
total_ram_mb=$(echo "$capacity_result" | head -1 | cut -d'|' -f2)
percentage=$(echo "$capacity_result" | head -1 | cut -d'|' -f3)
status=$(echo "$capacity_result" | head -1 | cut -d'|' -f4)
echo ""
cecho " Total RAM: ${WHITE}${total_ram_mb}MB${NC}"
cecho " Current max capacity: ${WHITE}${total_required_mb}MB${NC} (${percentage}%)"
case "$status" in
CRITICAL)
cecho " Status: ${RED}${BOLD}CRITICAL${NC} - Server may OOM if all pools hit max!"
;;
WARNING)
cecho " Status: ${YELLOW}${BOLD}WARNING${NC} - High memory pressure possible"
;;
CAUTION)
cecho " Status: ${YELLOW}CAUTION${NC} - Approaching memory limits"
;;
HEALTHY)
cecho " Status: ${GREEN}HEALTHY${NC} - Sufficient headroom"
;;
esac
echo ""
cecho "${CYAN}Step 2: Calculating balanced optimization...${NC}"
echo ""
# Get balanced recommendations based on total RAM
# Use per-domain optimization for cPanel, per-user for other panels
local balanced_result
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
cecho " ${GREEN}${NC} Detected cPanel - using per-domain optimization"
echo ""
# Don't redirect stderr so progress messages are visible
balanced_result=$(calculate_balanced_memory_allocation_per_domain)
else
cecho " ${GREEN}${NC} Detected $SYS_CONTROL_PANEL - using per-user optimization"
echo ""
balanced_result=$(calculate_balanced_memory_allocation)
fi
# Parse recommendations based on control panel type
declare -A recommended_values
declare -A domain_to_username
declare -A recommendation_reasons
local changes_needed=0
local no_change_needed=0
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
# cPanel format: DOMAIN|USERNAME|PHP_VER|CURRENT_MAX|AVG_MB|TRAFFIC_RPM|RECOMMENDED_MAX|ALLOCATED_MB|REASON
while IFS='|' read -r domain username php_ver current_max avg_mb traffic_rpm recommended_max allocated_mb reason; do
[ "$domain" = "DOMAIN" ] && continue # Skip header
[ -z "$domain" ] && continue
recommended_values["$domain"]="$recommended_max"
domain_to_username["$domain"]="$username"
recommendation_reasons["$domain"]="$reason"
# Track if change is needed
if [ "$current_max" != "$recommended_max" ]; then
changes_needed=$((changes_needed + 1))
if [[ "$reason" == *"REDUCE"* ]]; then
cecho " ${YELLOW}${NC} ${CYAN}$domain${NC} [$username]: $current_max${YELLOW}$recommended_max${NC} (${reason})"
elif [[ "$reason" == *"INCREASE"* ]]; then
cecho " ${GREEN}${NC} ${CYAN}$domain${NC} [$username]: $current_max${GREEN}$recommended_max${NC} (${reason})"
else
cecho " ${CYAN}$domain${NC} [$username]: $current_max$recommended_max (${reason})"
fi
else
no_change_needed=$((no_change_needed + 1))
fi
done <<< "$balanced_result"
else
# Other panels format: USER|CURRENT_MAX|AVG_MB|TRAFFIC_RPM|RECOMMENDED_MAX|ALLOCATED_MB|REASON
while IFS='|' read -r username current_max avg_mb traffic_rpm recommended_max allocated_mb reason; do
[ "$username" = "USER" ] && continue # Skip header
[ -z "$username" ] && continue
recommended_values["$username"]="$recommended_max"
recommendation_reasons["$username"]="$reason"
# Track if change is needed
if [ "$current_max" != "$recommended_max" ]; then
changes_needed=$((changes_needed + 1))
if [[ "$reason" == *"REDUCE"* ]]; then
cecho " ${YELLOW}${NC} ${CYAN}$username${NC}: $current_max${YELLOW}$recommended_max${NC} (${reason})"
elif [[ "$reason" == *"INCREASE"* ]]; then
cecho " ${GREEN}${NC} ${CYAN}$username${NC}: $current_max${GREEN}$recommended_max${NC} (${reason})"
else
cecho " ${CYAN}$username${NC}: $current_max$recommended_max (${reason})"
fi
else
no_change_needed=$((no_change_needed + 1))
fi
done <<< "$balanced_result"
fi
echo ""
cecho "${WHITE}${BOLD}SUMMARY${NC}"
cecho " Domains/users requiring changes: ${YELLOW}${BOLD}$changes_needed${NC}"
cecho " Already optimal: ${GREEN}$no_change_needed${NC}"
if [ "$changes_needed" -eq 0 ]; then
echo ""
cecho "${GREEN}${BOLD}✓ All domains are already optimally configured!${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
echo ""
cecho "${WHITE}${BOLD}APPLY OPTIONS${NC}"
cecho " ${GREEN}a${NC}) Apply ALL $changes_needed optimizations"
cecho " ${GREEN}s${NC}) Select individual domains/users to optimize"
cecho " ${RED}n${NC}) Cancel - don't apply any changes"
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
# Validate apply selection with retry loop
while true; do
read -p "Select option: " apply_confirm
if ! [[ "$apply_confirm" =~ ^[aAsSnN]$ ]]; then
echo ""
cecho "${RED}Invalid selection. Please enter a, s, or n${NC}"
echo ""
continue
fi
break
done
# Handle selection mode
declare -A domains_to_apply
apply_confirm=${apply_confirm,,}
case "$apply_confirm" in
a)
# Apply all - mark all domains/users for optimization
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
for domain in "${!recommended_values[@]}"; do
domains_to_apply["$domain"]="true"
done
else
for username in "${!recommended_values[@]}"; do
domains_to_apply["$username"]="true"
done
fi
;;
s)
# Individual selection
echo ""
cecho "${WHITE}${BOLD}SELECT DOMAINS/USERS TO OPTIMIZE${NC}"
cecho "${CYAN}Enter numbers separated by spaces (e.g., 1 3 5) or 'all' for all, 'none' to cancel${NC}"
echo ""
# Build selection list
local -a selection_list
local index=1
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
for domain in "${!recommended_values[@]}"; do
local username=${domain_to_username[$domain]}
local current_max=$(echo "$balanced_result" | grep "^$domain|" | cut -d'|' -f4)
local recommended_max=${recommended_values[$domain]}
if [ "$current_max" != "$recommended_max" ]; then
selection_list+=("$domain")
cecho " ${GREEN}$index${NC}) $domain [$username]: $current_max$recommended_max"
index=$((index + 1))
fi
done
else
for username in "${!recommended_values[@]}"; do
local current_max=$(echo "$balanced_result" | grep "^$username|" | cut -d'|' -f2)
local recommended_max=${recommended_values[$username]}
if [ "$current_max" != "$recommended_max" ]; then
selection_list+=("$username")
cecho " ${GREEN}$index${NC}) $username: $current_max$recommended_max"
index=$((index + 1))
fi
done
fi
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
read -p "Enter selection: " user_selection
# Normalize input to lowercase
user_selection=$(echo "$user_selection" | tr '[:upper:]' '[:lower:]')
if [[ "$user_selection" == "all" ]]; then
# Select all
for item in "${selection_list[@]}"; do
domains_to_apply["$item"]="true"
done
elif [[ "$user_selection" =~ ^(none|n)$ ]]; then
cecho "${YELLOW}Optimization cancelled${NC}"
echo ""
read -p "Press Enter to continue..."
return
else
# Parse selected numbers
for num in $user_selection; do
if [[ "$num" =~ ^[0-9]+$ ]] && [ "$num" -ge 1 ] && [ "$num" -le "${#selection_list[@]}" ]; then
local selected_item="${selection_list[$((num - 1))]}"
domains_to_apply["$selected_item"]="true"
fi
done
fi
# Check if anything was selected
if [ "${#domains_to_apply[@]}" -eq 0 ]; then
cecho "${YELLOW}No domains/users selected${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
echo ""
cecho "${GREEN}Selected ${#domains_to_apply[@]} domain(s)/user(s) for optimization${NC}"
;;
n)
cecho "${YELLOW}Optimization cancelled${NC}"
read -p "Press Enter to continue..."
return
;;
esac
show_banner
cecho "${WHITE}${BOLD}Applying optimizations...${NC}"
cecho "${CYAN}Selected: ${#domains_to_apply[@]} domain(s)/user(s)${NC}"
echo ""
# Get all users
local users
users=$(list_all_users)
if [ -z "$users" ]; then
cecho "${RED}ERROR: No users found on system${NC}"
read -p "Press Enter to continue..."
return
fi
# Statistics tracking
local total_domains=0
local optimized_count=0
local skipped_count=0
declare -A optimization_summary
# Process each user
while IFS= read -r username; do
[ -z "$username" ] && continue
local user_domains
user_domains=$(get_user_domains "$username")
while IFS= read -r domain; do
[ -z "$domain" ] && continue
total_domains=$((total_domains + 1))
# Check if this domain/user was selected for optimization
local should_optimize=false
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
[ "${domains_to_apply[$domain]}" = "true" ] && should_optimize=true
else
[ "${domains_to_apply[$username]}" = "true" ] && should_optimize=true
fi
if [ "$should_optimize" = "false" ]; then
cecho "${CYAN}[$total_domains] Skipping: ${WHITE}$domain${NC} ${CYAN}[$username]${NC} (not selected)"
skipped_count=$((skipped_count + 1))
continue
fi
cecho "${CYAN}[$total_domains] Processing: ${WHITE}$domain${NC} ${CYAN}[$username]${NC}"
# Detect issues first
local issues
issues=$(detect_php_config_issues "$username" "$domain")
# Count critical/high issues
local critical_high_count
critical_high_count=$(echo "$issues" | grep -cE "^[^|]*\|(CRITICAL|HIGH)\|" || true)
if [ "$critical_high_count" -eq 0 ]; then
cecho " ${GREEN}${NC} No optimization needed - domain is healthy"
skipped_count=$((skipped_count + 1))
continue
fi
# Use balanced recommendation from server-wide analysis
local recommended_max_children
if [ "$SYS_CONTROL_PANEL" = "cpanel" ]; then
# cPanel uses per-domain recommendations
recommended_max_children="${recommended_values[$domain]}"
else
# Other panels use per-user recommendations
recommended_max_children="${recommended_values[$username]}"
fi
# If no recommendation found, skip
if [ -z "$recommended_max_children" ]; then
continue
fi
# Get current pool config
local pool_config
pool_config=$(find_fpm_pool_config "$username" "$domain")
if [ -z "$pool_config" ] || [ ! -f "$pool_config" ]; then
cecho " ${YELLOW}${NC} No FPM pool config found - skipping"
skipped_count=$((skipped_count + 1))
continue
fi
# Get current max_children
local current_max_children
current_max_children=$(grep "^pm.max_children" "$pool_config" | awk -F'=' '{print $2}' | tr -d ' ')
# Apply optimizations
local changes_made=0
local changes_list=""
# Optimization 1: Adjust max_children if needed
if [ -n "$recommended_max_children" ] && [ -n "$current_max_children" ] && [ "$recommended_max_children" -ne "$current_max_children" ]; then
if modify_fpm_pool_setting "$pool_config" "pm.max_children" "$recommended_max_children" >/dev/null 2>&1; then
changes_made=$((changes_made + 1))
changes_list+="max_children: $current_max_children$recommended_max_children, "
optimization_summary["max_children"]=$((${optimization_summary["max_children"]:-0} + 1))
fi
fi
# Optimization 2: Enable OPcache if disabled
local opcache_enabled
opcache_enabled=$(check_opcache_enabled "$username")
if [ "$opcache_enabled" = "0" ]; then
# OPcache is disabled - mark for enabling (note: requires php.ini edit, not implemented yet)
changes_list+="OPcache: needs_enable, "
optimization_summary["opcache_needs_enable"]=$((${optimization_summary["opcache_needs_enable"]:-0} + 1))
fi
# Remove trailing comma
changes_list=${changes_list%, }
if [ "$changes_made" -gt 0 ]; then
cecho " ${GREEN}${NC} Optimized ($changes_made changes): $changes_list"
optimized_count=$((optimized_count + 1))
else
if [ -n "$changes_list" ]; then
cecho " ${YELLOW}${NC} Detected: $changes_list (manual intervention needed)"
skipped_count=$((skipped_count + 1))
else
cecho " ${YELLOW}${NC} No changes applied"
skipped_count=$((skipped_count + 1))
fi
fi
done <<< "$user_domains"
done <<< "$users"
# Restart PHP-FPM services if changes were made
if [ "$optimized_count" -gt 0 ]; then
echo ""
cecho "${CYAN}Restarting PHP-FPM services...${NC}"
# Get all PHP versions
local php_versions
php_versions=$(detect_installed_php_versions)
while IFS= read -r php_version; do
[ -z "$php_version" ] && continue
if reload_php_fpm "$php_version" >/dev/null 2>&1; then
cecho " ${GREEN}${NC} Reloaded $php_version"
fi
done <<< "$php_versions"
fi
# Display summary
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}OPTIMIZATION SUMMARY${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
cecho " Total domains processed: ${WHITE}$total_domains${NC}"
cecho " ${GREEN}Optimized: $optimized_count${NC}"
cecho " ${YELLOW}Skipped (healthy): $skipped_count${NC}"
echo ""
if [ ${#optimization_summary[@]} -gt 0 ]; then
cecho "${WHITE}Changes applied:${NC}"
for key in "${!optimization_summary[@]}"; do
cecho "$key: ${optimization_summary[$key]} domains"
done
fi
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 6: VIEW OPCACHE STATISTICS
# ============================================================================
view_opcache_stats() {
show_banner
local selection
selection=$(select_domain "opcache")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}OPCACHE STATISTICS: ${GREEN}$domain${NC}"
echo ""
# Get OPcache stats
local stats
stats=$(get_opcache_stats "$username")
if [ -z "$stats" ]; then
cecho "${RED}Unable to retrieve OPcache statistics${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
# Parse and display
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}OPCACHE STATUS${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
local enabled
enabled=$(check_opcache_enabled "$username")
if [ "$enabled" = "1" ]; then
cecho " Status: ${GREEN}ENABLED${NC}"
else
cecho " Status: ${RED}DISABLED${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
# Display stats
while IFS='=' read -r key value; do
[ -z "$key" ] && continue
case "$key" in
memory_usage_mb)
cecho " Memory Used: ${WHITE}${value}MB${NC}"
;;
hits)
cecho " Cache Hits: ${GREEN}$value${NC}"
;;
misses)
cecho " Cache Misses: ${YELLOW}$value${NC}"
;;
num_cached_scripts)
cecho " Cached Scripts: ${WHITE}$value${NC}"
;;
max_cached_keys)
cecho " Max Cached Keys: ${WHITE}$value${NC}"
;;
wasted_memory_mb)
cecho " Wasted Memory: ${RED}${value}MB${NC}"
;;
esac
done <<< "$stats"
# Calculate and display hit rate
local hit_rate
hit_rate=$(calculate_opcache_hit_rate "$username")
echo ""
cecho " Hit Rate: ${GREEN}${BOLD}${hit_rate}%${NC}"
# Recommendation
echo ""
# Check if hit_rate is numeric before using awk
if [ -n "$hit_rate" ] && [[ "$hit_rate" =~ ^[0-9]+$ ]]; then
local hit_rate_low=$(awk "BEGIN {print ($hit_rate < 90 ? 1 : 0)}" 2>/dev/null || echo 0)
if [ "$hit_rate_low" -eq 1 ]; then
cecho " ${YELLOW}⚠ Hit rate below 90% - Consider increasing opcache.memory_consumption${NC}"
else
cecho " ${GREEN}✓ Hit rate is excellent${NC}"
fi
else
cecho " ${YELLOW}⚠ Unable to determine OPcache hit rate${NC}"
fi
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 7: VIEW PHP-FPM PROCESS STATS
# ============================================================================
view_fpm_stats() {
show_banner
local selection
selection=$(select_domain "fpm")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}PHP-FPM PROCESS STATISTICS: ${GREEN}$domain${NC}"
echo ""
# Get process stats
local memory_stats
memory_stats=$(calculate_memory_per_process "$username")
local avg_kb process_count total_mb
avg_kb=$(echo "$memory_stats" | cut -d'|' -f1)
process_count=$(echo "$memory_stats" | cut -d'|' -f2)
total_mb=$(echo "$memory_stats" | cut -d'|' -f3)
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}CURRENT RESOURCE USAGE${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
if [ "$process_count" -eq 0 ]; then
cecho "${YELLOW}No active PHP-FPM processes found${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
cecho " Active Processes: ${WHITE}$process_count${NC}"
cecho " Avg Memory/Process: ${WHITE}$((avg_kb / 1024))MB${NC} (${avg_kb}KB)"
cecho " Total Memory: ${WHITE}${total_mb}MB${NC}"
echo ""
# Get pool config
local pool_config
pool_config=$(find_fpm_pool_config "$username")
if [ -n "$pool_config" ] && [ -f "$pool_config" ]; then
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
cecho "${WHITE}${BOLD}POOL CONFIGURATION${NC}"
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
local pool_settings
pool_settings=$(parse_fpm_pool_config "$pool_config")
# Display key settings
while IFS='=' read -r key value; do
[ -z "$key" ] && continue
cecho " $key: ${WHITE}$value${NC}"
done <<< "$pool_settings"
fi
# Calculate optimal max_children using improved algorithm
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
cecho "${WHITE}${BOLD}RECOMMENDATION${NC}"
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
# Get total system memory for improved calculation
local total_sys_ram
total_sys_ram=$(free -m | awk '/^Mem:/ {print $2}')
# NEW: Improved algorithm
local improved_opt
improved_opt=$(calculate_optimal_php_settings "$username" "$total_sys_ram")
local improved_max improved_pm improved_min improved_max_spare improved_opt_reason
improved_max=$(get_field "$improved_opt" 1)
improved_pm=$(get_field "$improved_opt" 2)
improved_min=$(get_field "$improved_opt" 3)
improved_max_spare=$(get_field "$improved_opt" 4)
improved_opt_reason=$(get_field "$improved_opt" 5)
# OLD: Legacy algorithm (for comparison)
local legacy_result
legacy_result=$(calculate_optimal_max_children "$username" 1024)
local legacy_recommended legacy_reason
legacy_recommended=$(echo "$legacy_result" | cut -d'|' -f1)
legacy_reason=$(echo "$legacy_result" | cut -d'|' -f2)
# Display comparison
cecho " ${GREEN}${BOLD}Improved Recommendation:${NC}"
cecho " pm.max_children: ${GREEN}$improved_max${NC}"
cecho " pm mode: ${GREEN}$improved_pm${NC}"
cecho " min_spare_servers: ${GREEN}$improved_min${NC}"
cecho " max_spare_servers: ${GREEN}$improved_max_spare${NC}"
cecho " Reason: $improved_opt_reason"
echo ""
if [ "$improved_max" -ne "$legacy_recommended" ]; then
cecho " ${YELLOW}Legacy Recommendation (for reference):${NC}"
cecho " pm.max_children: ${YELLOW}$legacy_recommended${NC} ($legacy_reason)"
echo ""
fi
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 8: CHECK FOR ISSUES
# ============================================================================
check_config_issues() {
show_banner
local selection
selection=$(select_domain "check")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}CONFIGURATION ISSUES: ${GREEN}$domain${NC}"
echo ""
# Detect issues
local issues
issues=$(detect_php_config_issues "$username" "$domain")
# Display by severity
local has_critical=false
local has_high=false
local has_medium=false
local has_low=false
# Check for each severity level
echo "$issues" | grep -q "CRITICAL" && has_critical=true
echo "$issues" | grep -q "HIGH" && has_high=true
echo "$issues" | grep -q "MEDIUM" && has_medium=true
echo "$issues" | grep -q "LOW" && has_low=true
# Display CRITICAL
if [ "$has_critical" = true ]; then
cecho "${RED}${BOLD}CRITICAL ISSUES:${NC}"
echo ""
while IFS='|' read -r issue_type severity message recommendation; do
[ "$severity" != "CRITICAL" ] && continue
cecho " ${RED}${NC} $message"
cecho " ${CYAN}${NC} $recommendation"
echo ""
done <<< "$issues"
fi
# Display HIGH
if [ "$has_high" = true ]; then
cecho "${YELLOW}${BOLD}HIGH PRIORITY ISSUES:${NC}"
echo ""
while IFS='|' read -r issue_type severity message recommendation; do
[ "$severity" != "HIGH" ] && continue
cecho " ${YELLOW}${NC} $message"
cecho " ${CYAN}${NC} $recommendation"
echo ""
done <<< "$issues"
fi
# Display MEDIUM
if [ "$has_medium" = true ]; then
cecho "${BLUE}${BOLD}MEDIUM PRIORITY ISSUES:${NC}"
echo ""
while IFS='|' read -r issue_type severity message recommendation; do
[ "$severity" != "MEDIUM" ] && continue
cecho " ${BLUE}${NC} $message"
cecho " ${CYAN}${NC} $recommendation"
echo ""
done <<< "$issues"
fi
# Display LOW
if [ "$has_low" = true ]; then
cecho "${GREEN}${BOLD}LOW PRIORITY ISSUES:${NC}"
echo ""
while IFS='|' read -r issue_type severity message recommendation; do
[ "$severity" != "LOW" ] && continue
cecho " ${GREEN}${NC} $message"
cecho " ${CYAN}${NC} $recommendation"
echo ""
done <<< "$issues"
fi
# No issues
if [ "$has_critical" = false ] && [ "$has_high" = false ] && [ "$has_medium" = false ] && [ "$has_low" = false ]; then
cecho "${GREEN}${BOLD}✓ No configuration issues detected!${NC}"
cecho "${GREEN}Configuration appears to be optimal.${NC}"
echo ""
fi
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION 9: CHECK SERVER MEMORY CAPACITY
# ============================================================================
check_server_memory_capacity() {
show_banner
cecho "${WHITE}${BOLD}SERVER MEMORY CAPACITY CHECK${NC}"
echo ""
cecho "${YELLOW}This checks if all PHP-FPM pools hitting max_children would cause OOM...${NC}"
echo ""
# Run capacity analysis
cecho "${CYAN}Analyzing PHP-FPM memory requirements...${NC}"
echo ""
local result
result=$(calculate_server_memory_capacity 2>/dev/null)
# Parse result - first line is summary, remaining lines are details
local main_result
main_result=$(echo "$result" | head -1)
local total_required total_ram percentage status details
total_required=$(echo "$main_result" | cut -d'|' -f1)
total_ram=$(echo "$main_result" | cut -d'|' -f2)
percentage=$(echo "$main_result" | cut -d'|' -f3)
status=$(echo "$main_result" | cut -d'|' -f4)
# Get details (all lines after first)
details=$(echo "$result" | tail -n +2)
# Display summary
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
cecho "${WHITE}${BOLD}MEMORY CAPACITY ANALYSIS${NC}"
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
cecho " Total Server RAM: ${WHITE}${total_ram}MB${NC}"
cecho " Required if ALL pools at max_children: ${WHITE}${total_required}MB${NC}"
cecho " Percentage of RAM: ${WHITE}${percentage}%${NC}"
echo ""
# Display status with color
case "$status" in
CRITICAL)
cecho " Status: ${RED}${BOLD}CRITICAL - HIGH OOM RISK!${NC}"
cecho ""
cecho " ${RED}WARNING: If all PHP-FPM pools hit their max_children limit,${NC}"
cecho " ${RED}the server will likely run out of memory and kill processes!${NC}"
;;
WARNING)
cecho " Status: ${YELLOW}${BOLD}WARNING - MODERATE OOM RISK${NC}"
cecho ""
cecho " ${YELLOW}CAUTION: Memory usage is high. Some pools may need reduction.${NC}"
;;
CAUTION)
cecho " Status: ${YELLOW}${BOLD}CAUTION - WATCH MEMORY USAGE${NC}"
cecho ""
cecho " ${YELLOW}Memory usage is elevated but manageable.${NC}"
;;
HEALTHY)
cecho " Status: ${GREEN}${BOLD}HEALTHY - LOW OOM RISK${NC}"
cecho ""
cecho " ${GREEN}Memory allocation appears safe.${NC}"
;;
esac
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
echo ""
# Ask if user wants detailed breakdown
if confirm "Show detailed per-user breakdown?"; then
echo ""
cecho "${WHITE}${BOLD}PER-USER BREAKDOWN${NC}"
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
printf "%-30s %-20s %12s %12s %12s\n" "DOMAIN" "USER" "MAX_CHILDREN" "AVG/PROCESS" "MAX_MEMORY"
printf "%-30s %-20s %12s %12s %12s\n" "------------------------------" "--------------------" "------------" "------------" "------------"
while IFS='|' read -r domain username max_children avg_mb pool_max_mb; do
[ -z "$domain" ] && continue
printf "%-30s %-20s %12s %12s %12s\n" "$domain" "$username" "$max_children" "$avg_mb" "$pool_max_mb"
done <<< "$details"
echo ""
fi
# Ask if user wants balanced recommendations
echo ""
if confirm "Calculate balanced memory allocation recommendations?"; then
echo ""
cecho "${WHITE}${BOLD}BALANCED MEMORY ALLOCATION RECOMMENDATIONS${NC}"
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
echo ""
cecho "${YELLOW}Calculating optimal max_children based on traffic...${NC}"
echo ""
local recommendations
recommendations=$(calculate_balanced_memory_allocation 2>/dev/null)
# Display header
local header
header=$(echo "$recommendations" | head -1)
echo "$header" | awk -F'|' '{printf "%-15s %8s %10s %12s %12s %15s %20s\n", $1, $2, $3, $4, $5, $6, $7}'
echo "────────────────────────────────────────────────────────────────────────────────────────────────────────"
# Display recommendations
echo "$recommendations" | tail -n +2 | while IFS='|' read -r user current_max avg_mb traffic_rpm recommended_max allocated_mb reason; do
[ -z "$user" ] && continue
# Color code reason
if [[ "$reason" == *"REDUCE"* ]]; then
color="${RED}"
elif [[ "$reason" == *"INCREASE"* ]]; then
color="${GREEN}"
else
color="${WHITE}"
fi
printf "%-15s %8s %10s %12s ${color}%12s${NC} %15s %20s\n" "$user" "$current_max" "$avg_mb" "$traffic_rpm" "$recommended_max" "$allocated_mb" "$reason"
done
echo ""
cecho "${YELLOW}NOTE: These are recommendations based on proportional traffic.${NC}"
cecho "${YELLOW}Actual needs may vary. Always test changes carefully.${NC}"
echo ""
fi
echo ""
cecho "${CYAN}═══════════════════════════════════════════════════════════════════${NC}"
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION B: BACKUP CONFIGURATIONS
# ============================================================================
backup_configurations() {
show_banner
cecho "${WHITE}${BOLD}BACKUP PHP CONFIGURATIONS${NC}"
echo ""
# Select domain
local selection
selection=$(select_domain "backup")
if [ $? -ne 0 ] || [ -z "$selection" ]; then
return
fi
local domain username
domain=$(echo "$selection" | cut -d'|' -f1)
username=$(echo "$selection" | cut -d'|' -f2)
show_banner
cecho "${WHITE}${BOLD}BACKUP: ${GREEN}$domain${NC}"
echo ""
# Initialize backup system
initialize_backup_system
# Create backup
cecho "${YELLOW}Creating backup of PHP configurations...${NC}"
echo ""
local backup_dir
backup_dir=$(backup_user_php_configs "$username" "$domain" 2>&1)
if [ $? -eq 0 ]; then
cecho "${GREEN}${BOLD}✓ Backup created successfully!${NC}"
echo ""
cecho " Backup location: ${WHITE}$backup_dir${NC}"
echo ""
# Show what was backed up
if [ -f "$backup_dir/metadata.txt" ]; then
cecho "${CYAN}Files backed up:${NC}"
grep "^ /" "$backup_dir/metadata.txt" | while IFS= read -r line; do
local file=$(echo "$line" | awk '{print $1}')
cecho " ${GREEN}${NC} $file"
done
fi
else
cecho "${RED}${BOLD}✗ Backup failed${NC}"
echo ""
echo "$backup_dir"
fi
echo ""
read -p "Press Enter to continue..."
}
# ============================================================================
# OPTION R: RESTORE FROM BACKUP
# ============================================================================
restore_configurations() {
show_banner
cecho "${WHITE}${BOLD}RESTORE FROM BACKUP${NC}"
echo ""
# List available backups
cecho "${CYAN}Available backups:${NC}"
echo ""
local backups
backups=$(list_backups)
if [ $? -ne 0 ]; then
cecho "${YELLOW}No backups found${NC}"
echo ""
read -p "Press Enter to continue..."
return
fi
# Display backups
local backup_array=()
local index=1
echo "$backups" | tail -n +2 | while IFS='|' read -r backup_name created username domain file_count; do
printf "${GREEN}%3d${NC}) %-20s %s ${CYAN}[%s]${NC} ${YELLOW}(%s)${NC} %s files\n" \
"$index" "$backup_name" "$created" "$username" "$domain" "$file_count"
backup_array+=("$backup_name")
index=$((index + 1))
done
# Store backup names in array for selection
mapfile -t backup_array < <(echo "$backups" | tail -n +2 | cut -d'|' -f1)
echo ""
cecho "${CYAN}─────────────────────────────────────────────────────────────────────${NC}"
# Validate backup selection with retry loop
while true; do
read -p "Select backup number to restore (or 'q' to cancel): " selection
if [[ "$selection" == "q" ]]; then
return
fi
if ! [[ "$selection" =~ ^[0-9]+$ ]] || [ "$selection" -lt 1 ] || [ "$selection" -gt ${#backup_array[@]} ]; then
echo ""
cecho "${RED}Invalid selection. Please enter a number 1-${#backup_array[@]}${NC}"
echo ""
continue
fi
break
done
local selected_backup="${backup_array[$((selection - 1))]}"
# Confirm restoration
echo ""
cecho "${YELLOW}${BOLD}WARNING: This will overwrite current configurations!${NC}"
echo ""
if ! confirm "Are you sure you want to restore from $selected_backup?"; then
cecho "${YELLOW}Restore cancelled${NC}"
sleep 2
return
fi
# Perform restore
show_banner
cecho "${WHITE}${BOLD}RESTORING FROM BACKUP${NC}"
echo ""
if restore_from_backup "$selected_backup"; then
echo ""
cecho "${GREEN}${BOLD}✓ Restore completed successfully!${NC}"
echo ""
cecho "${YELLOW}Don't forget to restart PHP-FPM for changes to take effect!${NC}"
else
echo ""
cecho "${RED}${BOLD}✗ Restore failed${NC}"
fi
echo ""
read -p "Press Enter to continue..."
}
# ============================================================================
# MAIN LOOP
# ============================================================================
main() {
# Detect system
initialize_system_detection
# Check if running as root
if [ "$EUID" -ne 0 ]; then
cecho "${RED}ERROR: This script must be run as root${NC}"
exit 1
fi
# Check for PHP-FPM
if ! is_using_php_fpm; then
cecho "${YELLOW}WARNING: PHP-FPM not detected. Some features may not work.${NC}"
read -p "Press Enter to continue anyway..."
fi
# Main loop
while true; do
show_banner
show_main_menu
# Validate choice input with retry loop
while true; do
read -p "Select option (0-9, b, r): " choice
if ! [[ "$choice" =~ ^([0-9]|[bBrR])$ ]]; then
echo ""
cecho "${RED}Invalid choice. Please enter 0-9, b, or r${NC}"
echo ""
continue
fi
break
done
# Convert uppercase to lowercase for case statement
choice=${choice,,}
case "$choice" in
1)
analyze_single_domain
;;
2)
analyze_all_domains
;;
3)
quick_health_check
;;
4)
optimize_domain
;;
5)
optimize_all_domains
;;
6)
view_opcache_stats
;;
7)
view_fpm_stats
;;
8)
check_config_issues
;;
9)
check_server_memory_capacity
;;
b)
backup_configurations
;;
r)
restore_configurations
;;
0)
cecho "${GREEN}Exiting PHP Optimizer...${NC}"
exit 0
;;
esac
done
}
# Run main
main "$@"