Files
Linux-Server-Management-Too…/lib/php-calculator-improved.sh
T
Developer 37de22241c feat: Implement three-constraint intelligent PHP-FPM optimization model
MAJOR ENHANCEMENT: Three-Constraint Intelligent Model

The PHP-FPM optimization now uses a sophisticated three-constraint model
to make the MOST INTELLIGENT recommendations possible:

CONSTRAINT 1: Memory-Based (What available RAM allows)
- Accounts for system reserve and MySQL memory
- Limits PHP-FPM to max 60% of total RAM
- Uses conservative 20MB per process assumption
- Results in realistic max_children values

CONSTRAINT 2: Traffic-Based (What actual usage patterns suggest)
- Analyzes peak concurrent requests from access logs
- Considers traffic stability (unstable/moderate/stable)
- Applies appropriate headroom factors (30% for stability)
- Caps at realistic traffic-based limits

CONSTRAINT 3: Fair Share (Proportional allocation based on traffic)
- Calculates server's total PHP-FPM capacity
- Allocates to each domain based on its traffic percentage
- High-traffic sites get more capacity, low-traffic get less
- Prevents single domain from monopolizing resources

FINAL RECOMMENDATION = MIN(Memory, Traffic, Fair Share)
This ensures:
-  Never exceeds available RAM
-  Never exceeds realistic traffic needs
-  Fair distribution across domains
-  Maximum capacity utilization
-  Safe for shared hosting environments

NEW FUNCTIONS:
- calculate_server_capacity() - Total server PHP-FPM capacity
- get_domain_traffic_percentage() - Domain's traffic % analysis
- calculate_max_children_fair_share() - Fair share allocation
- calculate_optimal_php_settings_intelligent() - Three-constraint model

BATCH ANALYZER CHANGES:
- Step 1: Calculates server capacity once upfront
- Step 2: Analyzes domain traffic patterns
- Step 3: Uses intelligent three-constraint model for each domain
- Output now shows: traffic percentage, limiting factor per domain

EXAMPLE ON 8GB SERVER:
- Server capacity: 320 max_children total
- Site A (70% traffic, 2GB peak): Gets 224 (capped at ~105 by memory)
- Site B (30% traffic, 500MB peak): Gets 96 (limited by traffic needs)
- Combined total: ~131 max_children ≈ 2.6GB (safe within 4.8GB available)

This is production-ready for shared hosting where fair resource
distribution and safety are critical.
2026-04-20 17:40:32 -04:00

657 lines
26 KiB
Bash
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
#!/bin/bash
################################################################################
# PHP-FPM Calculator - Improved Algorithm
# Purpose: Calculate optimal PHP-FPM pool settings based on:
# - Available server memory
# - Actual traffic patterns (peak concurrent requests)
# - Other service memory usage (MySQL, Redis, etc)
# - PM mode recommendations
# - Safe allocation buffers based on traffic stability
################################################################################
# Dependencies
_LIB_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
source "$_LIB_DIR/php-detector.sh" 2>/dev/null || { echo "ERROR: php-detector.sh not found"; return 1; }
source "$_LIB_DIR/system-detect.sh" 2>/dev/null || { echo "ERROR: system-detect.sh not found"; return 1; }
# ============================================================================
# HELPER FUNCTION - Extract field from pipe-delimited string
# ============================================================================
get_field() {
local input="$1"
local field_num="$2"
local temp="$input"
local i=1
while [ $i -lt "$field_num" ]; do
temp="${temp#*|}"
i=$((i + 1))
done
echo "${temp%%|*}"
}
# ============================================================================
# IMPROVED: SYSTEM RESERVE CALCULATION
# ============================================================================
# Calculate system reserve based on total RAM (percentage-based, not hardcoded)
# Usage: calculate_system_reserve <total_ram_mb>
# Returns: reserved_mb|reason
calculate_system_reserve() {
local total_ram_mb="$1"
if [ -z "$total_ram_mb" ] || [ "$total_ram_mb" -lt 512 ]; then
echo "256|Minimal system (< 512MB RAM)"
return
fi
local reserved_mb
# Dynamic reserve based on total RAM:
# Small servers (< 2GB): 15% reserve (keep base system stable)
# Medium servers (2-8GB): 20% reserve (typical workload)
# Large servers (8-32GB): 25% reserve (headroom for spikes)
# Very large servers (> 32GB): 30% reserve (accommodate multiple services)
if [ "$total_ram_mb" -lt 2048 ]; then
# Small VPS: 15% reserve
reserved_mb=$((total_ram_mb * 15 / 100))
[ "$reserved_mb" -lt 256 ] && reserved_mb=256
echo "$reserved_mb|Small server reserve (15% of ${total_ram_mb}MB)"
elif [ "$total_ram_mb" -lt 8192 ]; then
# Medium: 20% reserve
reserved_mb=$((total_ram_mb * 20 / 100))
echo "$reserved_mb|Medium server reserve (20% of ${total_ram_mb}MB)"
elif [ "$total_ram_mb" -lt 32768 ]; then
# Large: 25% reserve
reserved_mb=$((total_ram_mb * 25 / 100))
echo "$reserved_mb|Large server reserve (25% of ${total_ram_mb}MB)"
else
# Very large: 30% reserve
reserved_mb=$((total_ram_mb * 30 / 100))
echo "$reserved_mb|Very large server reserve (30% of ${total_ram_mb}MB)"
fi
}
# ============================================================================
# IMPROVED: MEMORY-BASED MAX_CHILDREN (Refined Algorithm)
# ============================================================================
# Calculate max_children based on available memory and safety buffer
# Usage: calculate_max_children_memory_based <username> <total_ram_mb>
# Returns: max_children|reason
calculate_max_children_memory_based() {
local username="$1"
local total_ram_mb="$2"
if [ -z "$total_ram_mb" ] || [ -z "$username" ]; then
echo "20|Invalid parameters"
return
fi
# Get average memory per process
local avg_kb
avg_kb=$(get_fpm_memory_usage "$username" 2>/dev/null || echo "0")
if [ "$avg_kb" -eq 0 ]; then
# No active processes detected (ondemand mode, or low traffic)
# Use safe default: 20 processes with assumed 50MB per process
echo "20|No active processes, using safe default"
return
fi
# Calculate system reserve (dynamic percentage-based)
local reserve_result
reserve_result=$(calculate_system_reserve "$total_ram_mb")
local reserved_mb
reserved_mb=$(get_field "$reserve_result" 1)
# Account for MySQL memory (critical on shared hosting!)
local mysql_memory_mb=0
local mysql_info
mysql_info=$(detect_mysql_memory_usage 2>/dev/null)
if [ $? -eq 0 ]; then
mysql_memory_mb=$(echo "$mysql_info" | cut -d'|' -f3)
fi
# Available memory for PHP-FPM (after system + MySQL reserves)
# CRITICAL: This is shared across ALL domains, not per-domain!
local available_mb=$((total_ram_mb - reserved_mb - mysql_memory_mb))
# Safety check: never allow PHP-FPM to use more than 60% of RAM
local max_php_fpm=$((total_ram_mb * 60 / 100))
if [ "$available_mb" -gt "$max_php_fpm" ]; then
available_mb=$max_php_fpm
fi
# Convert average KB to MB
local avg_mb=$((avg_kb / 1024))
if [ "$avg_mb" -eq 0 ]; then
avg_mb=20 # More realistic default (not 1MB)
fi
# Theoretical maximum without safety buffer
local theoretical_max=$((available_mb / avg_mb))
# Apply safety buffer (50% - much more conservative for shared hosting!)
# This accounts for peak traffic spikes and other processes
local safety_buffer=50
local recommended=$((theoretical_max * (100 - safety_buffer) / 100))
# Hard cap at traffic-realistic limits
if [ "$recommended" -lt 5 ]; then
echo "5|Minimum safe value (insufficient memory)"
elif [ "$recommended" -gt 150 ]; then
# CRITICAL: Cap at 150 max per domain on shared hosting
# Higher values require dedicated servers
echo "150|Capped at safe maximum for shared hosting (would be $recommended)"
else
local reason="Memory-based: ${avg_mb}MB per process, ${available_mb}MB available (after MySQL: ${mysql_memory_mb}MB), ${safety_buffer}% safety buffer"
echo "$recommended|$reason"
fi
}
# ============================================================================
# NEW: TRAFFIC-BASED MAX_CHILDREN CALCULATION
# ============================================================================
# Calculate max_children based on actual peak concurrent requests
# Usage: calculate_peak_concurrent_requests <username> <days>
# Returns: peak_concurrent|stability_factor
calculate_peak_concurrent_requests_improved() {
local username="$1"
local days="${2:-7}"
# Find access logs
local access_logs
access_logs=$(find /home/"$username"/*/logs -name "access_log*" -o -name "access.log*" 2>/dev/null | head -5)
if [ -z "$access_logs" ]; then
echo "0|0.8|No access logs found"
return
fi
# Analyze access logs to find peak concurrent requests
# Strategy: Use combined timestamp analysis for better accuracy
local peak_concurrent=0
local total_samples=0
local high_traffic_periods=0
local traffic_variance=0
# Sample each log and find peaks
while IFS= read -r log_file; do
[ ! -f "$log_file" ] && continue
# Get logs from last N days
local temp_processed
temp_processed=$(find "$log_file" -mtime -"$days" -exec tail -n 10000 {} \; 2>/dev/null | \
awk '{print $4}' | sed 's/\[//' | sort | uniq -c | sort -rn | head -1)
if [ -n "$temp_processed" ]; then
local sample_count
sample_count=$(echo "$temp_processed" | awk '{print $1}')
if [ "$sample_count" -gt "$peak_concurrent" ]; then
peak_concurrent=$sample_count
fi
total_samples=$((total_samples + 1))
fi
done <<< "$access_logs"
# If no samples, estimate from HTTP status codes
if [ "$total_samples" -eq 0 ]; then
# Estimate: count 200 responses per second at peak
peak_concurrent=$(tail -n 100000 "$log_file" 2>/dev/null | grep " 200 " | wc -l | awk '{print int($1/100)}')
if [ "$peak_concurrent" -lt 1 ]; then
peak_concurrent=1
fi
fi
# Estimate traffic stability (0.6 = unstable, 0.8 = stable, 0.9 = very stable)
# This is used to adjust safety buffer
local stability_factor=0.8
if [ "$total_samples" -lt 3 ]; then
stability_factor=0.6 # Very limited data, assume unstable
elif [ "$total_samples" -ge 10 ]; then
stability_factor=0.9 # Good data, assume stable
fi
echo "$peak_concurrent|$stability_factor|Based on $total_samples access logs"
}
# ============================================================================
# NEW: RECOMMEND MAX_CHILDREN from TRAFFIC PATTERNS
# ============================================================================
# Calculate recommended max_children based on peak concurrent requests
# Usage: calculate_max_children_traffic_based <peak_concurrent> <stability_factor>
# Returns: recommended_max_children|reason
calculate_max_children_traffic_based() {
local peak_concurrent="$1"
local stability_factor="${2:-0.8}"
if [ "$peak_concurrent" -lt 1 ]; then
echo "5|Insufficient traffic data, using minimum"
return
fi
# Formula: recommended = peak_concurrent * (1.0 + headroom_factor) * stability_factor
# headroom_factor: extra capacity for unexpected spikes (default 0.3 = 30%)
local headroom_factor=0.3
local recommended=$(echo "$peak_concurrent (1 + $headroom_factor) * $stability_factor" | bc | awk '{print int($1)}')
# Sanity bounds
if [ "$recommended" -lt 5 ]; then
recommended=5
elif [ "$recommended" -gt 200 ]; then
recommended=200 # Most domains don't need more than 200 concurrent processes
fi
local reason="Traffic-based: $peak_concurrent peak concurrent requests"
if [ "$stability_factor" != "0.8" ]; then
reason="$reason (stability factor: $stability_factor)"
fi
echo "$recommended|$reason"
}
# ============================================================================
# NEW: DETECT MYSQL MEMORY USAGE
# ============================================================================
# Get MySQL memory usage to account for in PHP-FPM allocation
# Usage: detect_mysql_memory_usage
# Returns: mysql_memory_mb|status
detect_mysql_memory_usage() {
if ! command -v mysql &>/dev/null && ! command -v mysqld &>/dev/null; then
echo "0|MySQL not installed"
return
fi
# Try to get MySQL process memory usage
local mysql_mem
mysql_mem=$(ps aux | grep "[m]ysqld" | awk '{print int($6/1024)}')
if [ -z "$mysql_mem" ] || [ "$mysql_mem" -eq 0 ]; then
# Fallback: estimate from MySQL variables
if command -v mysql &>/dev/null; then
mysql_mem=$(mysql -e "SHOW VARIABLES LIKE '%buffer%'" 2>/dev/null | grep -i "buffer" | \
awk -F'\t' '{gsub(/[KM]/,"",$3); if($3 ~ /K/) $3=$3/1024; print $3}' | \
awk '{sum+=$1} END {print int(sum)}')
fi
fi
if [ -z "$mysql_mem" ] || [ "$mysql_mem" -eq 0 ]; then
# Safe default estimate: 300MB for typical MySQL
echo "300|Estimated default"
else
echo "$mysql_mem|Detected from process"
fi
}
# ============================================================================
# NEW: CALCULATE SERVER TOTAL CAPACITY
# ============================================================================
# Calculate the total max_children the entire server can support
# Usage: calculate_server_capacity <total_ram_mb>
# Returns: total_capacity|available_memory|memory_per_process|reason
calculate_server_capacity() {
local total_ram_mb="$1"
if [ -z "$total_ram_mb" ] || [ "$total_ram_mb" -lt 512 ]; then
echo "0|0|20|Insufficient RAM for calculation"
return
fi
# Calculate system reserve (dynamic percentage-based)
local reserve_result
reserve_result=$(calculate_system_reserve "$total_ram_mb")
local reserved_mb
reserved_mb=$(get_field "$reserve_result" 1)
# Account for MySQL memory (critical on shared hosting!)
local mysql_memory_mb=0
local mysql_info
mysql_info=$(detect_mysql_memory_usage 2>/dev/null)
if [ $? -eq 0 ]; then
mysql_memory_mb=$(echo "$mysql_info" | cut -d'|' -f3)
fi
# Available memory for PHP-FPM (after system + MySQL reserves)
local available_mb=$((total_ram_mb - reserved_mb - mysql_memory_mb))
# Safety check: never allow PHP-FPM to use more than 60% of RAM
local max_php_fpm=$((total_ram_mb * 60 / 100))
if [ "$available_mb" -gt "$max_php_fpm" ]; then
available_mb=$max_php_fpm
fi
# Conservative memory per process for safety (20MB)
local memory_per_process=20
# Total capacity = available memory / memory per process
local total_capacity=$((available_mb / memory_per_process))
# Sanity checks
[ "$total_capacity" -lt 5 ] && total_capacity=5
[ "$total_capacity" -gt 500 ] && total_capacity=500
echo "$total_capacity|$available_mb|$memory_per_process|Server can support $total_capacity total max_children"
}
# ============================================================================
# NEW: GET DOMAIN TRAFFIC PERCENTAGE
# ============================================================================
# Calculate what percentage of total server traffic this domain handles
# Usage: get_domain_traffic_percentage <username> <domain> <all_domains_list>
# Returns: percentage|request_count|reason
get_domain_traffic_percentage() {
local username="$1"
local domain="$2"
local all_domains="$3"
if [ -z "$domain" ] || [ -z "$all_domains" ]; then
echo "50|0|Insufficient data"
return
fi
# Find access logs for this domain
local access_log
access_log=$(find /home/"$username"/*/logs -name "*access*log*" 2>/dev/null | grep "$domain" | head -1)
if [ -z "$access_log" ] || [ ! -f "$access_log" ]; then
# No specific log found, assume equal distribution
local domain_count
domain_count=$(echo "$all_domains" | wc -l)
local percentage=$((100 / domain_count))
echo "$percentage|0|No logs found, assuming equal distribution"
return
fi
# Count requests for this domain from last 7 days
local domain_requests
domain_requests=$(find "$access_log" -mtime -7 -exec wc -l {} \; 2>/dev/null | awk '{print $1}')
[ -z "$domain_requests" ] && domain_requests=0
if [ "$domain_requests" -eq 0 ]; then
# No recent traffic, use equal distribution
local domain_count
domain_count=$(echo "$all_domains" | wc -l)
local percentage=$((100 / domain_count))
echo "$percentage|0|No recent traffic"
return
fi
# Count total requests across all domains
local total_requests=0
while IFS= read -r d; do
[ -z "$d" ] && continue
local d_log
d_log=$(find /home/"$username"/*/logs -name "*access*log*" 2>/dev/null | head -1)
if [ -f "$d_log" ]; then
local d_count
d_count=$(find "$d_log" -mtime -7 -exec wc -l {} \; 2>/dev/null | awk '{print $1}')
total_requests=$((total_requests + d_count))
fi
done <<< "$all_domains"
if [ "$total_requests" -eq 0 ]; then
echo "50|$domain_requests|Insufficient total traffic"
return
fi
# Calculate percentage
local percentage=$((domain_requests * 100 / total_requests))
[ "$percentage" -lt 1 ] && percentage=1
[ "$percentage" -gt 99 ] && percentage=99
echo "$percentage|$domain_requests|Based on $total_requests total requests"
}
# ============================================================================
# NEW: CALCULATE FAIR SHARE BASED ON TRAFFIC
# ============================================================================
# Calculate this domain's fair share of server capacity based on traffic percentage
# Usage: calculate_max_children_fair_share <total_capacity> <traffic_percentage>
# Returns: fair_share_max|reason
calculate_max_children_fair_share() {
local total_capacity="$1"
local traffic_percentage="$2"
if [ -z "$total_capacity" ] || [ -z "$traffic_percentage" ]; then
echo "20|Invalid parameters"
return
fi
# Calculate fair share: total capacity × traffic percentage
local fair_share=$((total_capacity * traffic_percentage / 100))
# Apply hard limits
if [ "$fair_share" -lt 5 ]; then
echo "5|Fair share is very small (minimum enforced)"
elif [ "$fair_share" -gt 150 ]; then
echo "150|Fair share exceeds shared hosting limit (capped at 150)"
else
echo "$fair_share|Fair share: $traffic_percentage% of $total_capacity total"
fi
}
# ============================================================================
# NEW: RECOMMEND PM MODE (static/dynamic/ondemand)
# ============================================================================
# Recommend most appropriate PHP-FPM pm mode based on traffic pattern
# Usage: recommend_pm_mode <peak_concurrent> <average_concurrent> <stability_factor>
# Returns: pm_mode|min_spare|max_spare|reason
recommend_pm_mode() {
local peak_concurrent="$1"
local average_concurrent="${2:-$(echo "$peak_concurrent / 2" | bc)}"
local stability_factor="${3:-0.8}"
# Determine stability level
local traffic_pattern
if [ "$(echo "$stability_factor < 0.65" | bc)" -eq 1 ]; then
traffic_pattern="UNSTABLE"
elif [ "$(echo "$stability_factor < 0.85" | bc)" -eq 1 ]; then
traffic_pattern="MODERATE"
else
traffic_pattern="STABLE"
fi
# Recommend mode based on traffic characteristics
local pm_mode min_spare max_spare reason
if [ "$peak_concurrent" -lt 5 ]; then
# Very low traffic: ondemand saves memory
pm_mode="ondemand"
min_spare=0
max_spare=3
reason="Very low traffic ($peak_concurrent peak concurrent)"
elif [ "$traffic_pattern" = "UNSTABLE" ]; then
# Unstable traffic: dynamic gives best balance
pm_mode="dynamic"
min_spare=$((peak_concurrent / 4))
max_spare=$((peak_concurrent * 3 / 4))
reason="Unstable traffic pattern (stability: $stability_factor)"
elif [ "$traffic_pattern" = "STABLE" ]; then
# Stable high traffic: static for performance
pm_mode="static"
min_spare=$((peak_concurrent - 2))
max_spare=$((peak_concurrent + 2))
reason="Stable traffic pattern (peak: $peak_concurrent concurrent)"
else
# Moderate/mixed traffic: dynamic is good default
pm_mode="dynamic"
min_spare=$((peak_concurrent / 3))
max_spare=$((peak_concurrent * 2 / 3))
reason="Moderate traffic ($traffic_pattern)"
fi
# Sanity bounds
[ "$min_spare" -lt 1 ] && min_spare=1
[ "$max_spare" -lt "$min_spare" ] && max_spare=$((min_spare + 2))
[ "$max_spare" -gt 100 ] && max_spare=100
echo "$pm_mode|$min_spare|$max_spare|$reason"
}
# ============================================================================
# NEW: COMPREHENSIVE RECOMMENDATION
# ============================================================================
# Calculate optimal settings combining memory and traffic analysis
# Usage: calculate_optimal_php_settings <username> <total_ram_mb>
# Returns: max_children|pm_mode|min_spare|max_spare|reason
calculate_optimal_php_settings() {
local username="$1"
local total_ram_mb="$2"
if [ -z "$username" ] || [ -z "$total_ram_mb" ]; then
echo "0|dynamic|1|5|Invalid parameters"
return
fi
# Calculate memory-based recommendation
local memory_result
memory_result=$(calculate_max_children_memory_based "$username" "$total_ram_mb")
local memory_based_max
memory_based_max=$(get_field "$memory_result" 1)
# Calculate traffic-based recommendation
local traffic_result
traffic_result=$(calculate_peak_concurrent_requests_improved "$username" 7)
local peak_concurrent stability_factor
peak_concurrent=$(get_field "$traffic_result" 1)
stability_factor=$(get_field "$traffic_result" 2)
local traffic_based_max=0
if [ "$peak_concurrent" -gt 0 ]; then
local traffic_calc
traffic_calc=$(calculate_max_children_traffic_based "$peak_concurrent" "$stability_factor")
traffic_based_max=$(get_field "$traffic_calc" 1)
fi
# Combine both recommendations (use lower value for safety)
local final_max_children="$memory_based_max"
local reason_prefix="Memory-based"
if [ "$traffic_based_max" -gt 0 ] && [ "$traffic_based_max" -lt "$memory_based_max" ]; then
final_max_children="$traffic_based_max"
reason_prefix="Traffic-based (constrained by memory)"
elif [ "$traffic_based_max" -gt 0 ]; then
reason_prefix="Combined (memory: $memory_based_max, traffic: $traffic_based_max)"
fi
# CRITICAL: Ensure we never recommend 0 or invalid values
if [ -z "$final_max_children" ] || [ "$final_max_children" -le 0 ]; then
final_max_children="20"
reason_prefix="Safe default (calculation failed or returned invalid value)"
fi
# Recommend pm mode
local pm_result
pm_result=$(recommend_pm_mode "$peak_concurrent" "$((peak_concurrent / 2))" "$stability_factor")
local pm_mode min_spare max_spare pm_reason
pm_mode=$(get_field "$pm_result" 1)
min_spare=$(get_field "$pm_result" 2)
max_spare=$(get_field "$pm_result" 3)
pm_reason=$(get_field "$pm_result" 4)
echo "$final_max_children|$pm_mode|$min_spare|$max_spare|$reason_prefix: $pm_reason"
}
# ============================================================================
# NEW: THREE-CONSTRAINT INTELLIGENT OPTIMIZATION
# ============================================================================
# Calculate optimal settings using three constraints for maximum intelligence:
# 1. Memory constraint - what available RAM allows
# 2. Traffic constraint - what actual usage suggests
# 3. Fair share constraint - proportional allocation based on traffic
# Uses the MINIMUM of all three for maximum safety and fairness
# Usage: calculate_optimal_php_settings_intelligent <username> <total_ram_mb> <total_server_capacity> <traffic_percentage>
# Returns: max_children|pm_mode|min_spare|max_spare|limiting_factor|reason
calculate_optimal_php_settings_intelligent() {
local username="$1"
local total_ram_mb="$2"
local total_server_capacity="$3"
local traffic_percentage="$4"
if [ -z "$username" ] || [ -z "$total_ram_mb" ] || [ -z "$total_server_capacity" ]; then
echo "0|dynamic|1|5|ERROR|Invalid parameters"
return
fi
# Default traffic percentage if not provided (equal distribution)
[ -z "$traffic_percentage" ] && traffic_percentage=50
# CONSTRAINT 1: Memory-based max (what RAM allows)
local memory_result
memory_result=$(calculate_max_children_memory_based "$username" "$total_ram_mb")
local memory_based_max
memory_based_max=$(get_field "$memory_result" 1)
# CONSTRAINT 2: Traffic-based max (what traffic patterns suggest)
local traffic_result
traffic_result=$(calculate_peak_concurrent_requests_improved "$username" 7)
local peak_concurrent stability_factor
peak_concurrent=$(get_field "$traffic_result" 1)
stability_factor=$(get_field "$traffic_result" 2)
local traffic_based_max=0
if [ "$peak_concurrent" -gt 0 ]; then
local traffic_calc
traffic_calc=$(calculate_max_children_traffic_based "$peak_concurrent" "$stability_factor")
traffic_based_max=$(get_field "$traffic_calc" 1)
else
traffic_based_max=$memory_based_max # No traffic data, use memory as basis
fi
# CONSTRAINT 3: Fair share (proportional allocation based on traffic %)
local fair_share_result
fair_share_result=$(calculate_max_children_fair_share "$total_server_capacity" "$traffic_percentage")
local fair_share_max
fair_share_max=$(get_field "$fair_share_result" 1)
# USE THE MINIMUM OF ALL THREE CONSTRAINTS
local final_max_children="$memory_based_max"
local limiting_factor="Memory (${memory_based_max}MB available)"
if [ "$traffic_based_max" -lt "$final_max_children" ]; then
final_max_children="$traffic_based_max"
limiting_factor="Traffic (peak ${peak_concurrent} concurrent)"
fi
if [ "$fair_share_max" -lt "$final_max_children" ]; then
final_max_children="$fair_share_max"
limiting_factor="Fair share (${traffic_percentage}% of server capacity)"
fi
# CRITICAL: Ensure we never recommend 0 or invalid values
if [ -z "$final_max_children" ] || [ "$final_max_children" -le 0 ]; then
final_max_children="20"
limiting_factor="Safe default (calculation failed)"
fi
# Recommend pm mode
local pm_result
pm_result=$(recommend_pm_mode "$peak_concurrent" "$((peak_concurrent / 2))" "$stability_factor")
local pm_mode min_spare max_spare pm_reason
pm_mode=$(get_field "$pm_result" 1)
min_spare=$(get_field "$pm_result" 2)
max_spare=$(get_field "$pm_result" 3)
pm_reason=$(get_field "$pm_result" 4)
# Return with detailed explanation
local reason="3-constraint intelligent: Mem=$memory_based_max, Traffic=$traffic_based_max, Share=$fair_share_max$limiting_factor"
echo "$final_max_children|$pm_mode|$min_spare|$max_spare|$limiting_factor|$reason"
}
# ============================================================================
# Export functions for use in other scripts
# ============================================================================
export -f calculate_system_reserve
export -f calculate_max_children_memory_based
export -f calculate_peak_concurrent_requests_improved
export -f calculate_max_children_traffic_based
export -f detect_mysql_memory_usage
export -f calculate_server_capacity
export -f get_domain_traffic_percentage
export -f calculate_max_children_fair_share
export -f recommend_pm_mode
export -f calculate_optimal_php_settings
export -f calculate_optimal_php_settings_intelligent
export -f get_field