|
| 1 | +#!/bin/bash |
| 2 | +# |
| 3 | +# Aquiis Startup Performance Diagnostic Tool |
| 4 | +# Profiles AppImage startup to identify bottlenecks |
| 5 | +# |
| 6 | + |
| 7 | +RED='\033[0;31m' |
| 8 | +GREEN='\033[0;32m' |
| 9 | +YELLOW='\033[1;33m' |
| 10 | +BLUE='\033[0;34m' |
| 11 | +NC='\033[0m' |
| 12 | + |
| 13 | +APPIMAGE="${1:-$HOME/Applications/AquiisPropertyManagement-1.0.0.AppImage}" |
| 14 | + |
| 15 | +if [ ! -f "$APPIMAGE" ]; then |
| 16 | + echo -e "${RED}Error: AppImage not found at: $APPIMAGE${NC}" |
| 17 | + echo "Usage: $0 [path-to-appimage]" |
| 18 | + exit 1 |
| 19 | +fi |
| 20 | + |
| 21 | +echo -e "${BLUE}Aquiis Startup Performance Diagnostic${NC}" |
| 22 | +echo "==========================================" |
| 23 | +echo "" |
| 24 | +echo "AppImage: $(basename "$APPIMAGE")" |
| 25 | +echo "Size: $(du -h "$APPIMAGE" | cut -f1)" |
| 26 | +echo "" |
| 27 | + |
| 28 | +# Test 1: AppImage Mount Time |
| 29 | +echo -e "${YELLOW}Test 1: AppImage FUSE Mount Time${NC}" |
| 30 | +START=$(date +%s.%N) |
| 31 | +"$APPIMAGE" --appimage-help > /dev/null 2>&1 |
| 32 | +END=$(date +%s.%N) |
| 33 | +MOUNT_TIME=$(echo "$END - $START" | bc) |
| 34 | +echo " Mount time: ${MOUNT_TIME}s" |
| 35 | +echo "" |
| 36 | + |
| 37 | +# Test 2: .NET Runtime Initialization |
| 38 | +echo -e "${YELLOW}Test 2: .NET Runtime Startup${NC}" |
| 39 | +echo " Starting AppImage with timestamps..." |
| 40 | +START=$(date +%s.%N) |
| 41 | + |
| 42 | +# Launch AppImage and capture its PID |
| 43 | +"$APPIMAGE" > /tmp/aquiis-startup.log 2>&1 & |
| 44 | +APPIMAGE_PID=$! |
| 45 | + |
| 46 | +# Monitor for process start |
| 47 | +DOTNET_PID="" |
| 48 | +ELAPSED=0 |
| 49 | +while [ -z "$DOTNET_PID" ] && [ $ELAPSED -lt 60 ]; do |
| 50 | + sleep 0.5 |
| 51 | + DOTNET_PID=$(pgrep -P $APPIMAGE_PID | head -1) |
| 52 | + ELAPSED=$((ELAPSED + 1)) |
| 53 | +done |
| 54 | + |
| 55 | +if [ -n "$DOTNET_PID" ]; then |
| 56 | + DOTNET_START=$(date +%s.%N) |
| 57 | + DOTNET_LAUNCH=$(echo "$DOTNET_START - $START" | bc) |
| 58 | + echo " .NET process spawned: ${DOTNET_LAUNCH}s" |
| 59 | + |
| 60 | + # Wait for Kestrel to start (check for port 8888) |
| 61 | + KESTREL_READY=0 |
| 62 | + KESTREL_ELAPSED=0 |
| 63 | + while [ $KESTREL_READY -eq 0 ] && [ $KESTREL_ELAPSED -lt 120 ]; do |
| 64 | + if netstat -tunl 2>/dev/null | grep -q ":8888 " || ss -tunl 2>/dev/null | grep -q ":8888 "; then |
| 65 | + KESTREL_READY=1 |
| 66 | + KESTREL_END=$(date +%s.%N) |
| 67 | + KESTREL_TIME=$(echo "$KESTREL_END - $DOTNET_START" | bc) |
| 68 | + echo " Kestrel listening on port 8888: ${KESTREL_TIME}s" |
| 69 | + fi |
| 70 | + sleep 0.5 |
| 71 | + KESTREL_ELAPSED=$((KESTREL_ELAPSED + 1)) |
| 72 | + done |
| 73 | + |
| 74 | + if [ $KESTREL_READY -eq 0 ]; then |
| 75 | + echo " ${RED}Warning: Kestrel did not start within 60s${NC}" |
| 76 | + fi |
| 77 | + |
| 78 | + # Wait for Electron window |
| 79 | + ELECTRON_PID="" |
| 80 | + ELECTRON_ELAPSED=0 |
| 81 | + while [ -z "$ELECTRON_PID" ] && [ $ELECTRON_ELAPSED -lt 120 ]; do |
| 82 | + ELECTRON_PID=$(pgrep -f "electron.*aquiis" | head -1) |
| 83 | + if [ -z "$ELECTRON_PID" ]; then |
| 84 | + sleep 0.5 |
| 85 | + ELECTRON_ELAPSED=$((ELECTRON_ELAPSED + 1)) |
| 86 | + fi |
| 87 | + done |
| 88 | + |
| 89 | + if [ -n "$ELECTRON_PID" ]; then |
| 90 | + ELECTRON_END=$(date +%s.%N) |
| 91 | + ELECTRON_TIME=$(echo "$ELECTRON_END - $START" | bc) |
| 92 | + echo " Electron window opened: ${ELECTRON_TIME}s" |
| 93 | + fi |
| 94 | + |
| 95 | + # Total startup time |
| 96 | + TOTAL_END=$(date +%s.%N) |
| 97 | + TOTAL_TIME=$(echo "$TOTAL_END - $START" | bc) |
| 98 | + |
| 99 | + echo "" |
| 100 | + echo -e "${GREEN}Startup Timeline:${NC}" |
| 101 | + echo " 1. AppImage mount: ${MOUNT_TIME}s" |
| 102 | + echo " 2. .NET launch: ${DOTNET_LAUNCH}s" |
| 103 | + echo " 3. Kestrel ready: ${KESTREL_TIME}s (at ${KESTREL_END}s total)" |
| 104 | + echo " 4. Window open: ${ELECTRON_TIME}s" |
| 105 | + echo "" |
| 106 | + echo -e "${BLUE}Total startup: ${TOTAL_TIME}s${NC}" |
| 107 | + |
| 108 | + # Kill the test instance |
| 109 | + echo "" |
| 110 | + echo -e "${YELLOW}Stopping test instance...${NC}" |
| 111 | + kill $APPIMAGE_PID 2>/dev/null |
| 112 | + sleep 2 |
| 113 | + pkill -9 -f "Aquiis" 2>/dev/null |
| 114 | +else |
| 115 | + echo " ${RED}Error: .NET process did not start${NC}" |
| 116 | + kill $APPIMAGE_PID 2>/dev/null |
| 117 | +fi |
| 118 | + |
| 119 | +echo "" |
| 120 | +echo -e "${YELLOW}Test 3: System Information${NC}" |
| 121 | +echo " CPU: $(grep "model name" /proc/cpuinfo | head -1 | cut -d: -f2 | xargs)" |
| 122 | +echo " RAM: $(free -h | awk '/^Mem:/ {print $2}')" |
| 123 | +echo " Disk (AppImage): $(df -h "$APPIMAGE" | tail -1 | awk '{print $1 " (" $4 " free)"}')" |
| 124 | +echo " Disk (Config): $(df -h ~/.config/Aquiis 2>/dev/null | tail -1 | awk '{print $1 " (" $4 " free)")' || echo "Not initialized"}" |
| 125 | +
|
| 126 | +echo "" |
| 127 | +echo -e "${YELLOW}Test 4: Database Check${NC}" |
| 128 | +if [ -f ~/.config/Aquiis/Data/app_v1.0.0.db ]; then |
| 129 | + DB_SIZE=$(du -h ~/.config/Aquiis/Data/app_v1.0.0.db | cut -f1) |
| 130 | + echo " Database size: $DB_SIZE" |
| 131 | + echo " Database location: ~/.config/Aquiis/Data/app_v1.0.0.db" |
| 132 | +else |
| 133 | + echo " Database: Not initialized (first run will be slower)" |
| 134 | +fi |
| 135 | +
|
| 136 | +echo "" |
| 137 | +echo -e "${YELLOW}Test 5: AppImage Properties${NC}" |
| 138 | +"$APPIMAGE" --appimage-version 2>/dev/null || echo " AppImage runtime: Type 2" |
| 139 | +echo " Compression: $(file "$APPIMAGE" | grep -o "gzip\|xz\|zstd" || echo "unknown")" |
| 140 | +
|
| 141 | +echo "" |
| 142 | +echo -e "${GREEN}Diagnostic complete!${NC}" |
| 143 | +echo "" |
| 144 | +echo "If startup is still slow (>20s), check:" |
| 145 | +echo " 1. Antivirus/security software scanning AppImage" |
| 146 | +echo " 2. Slow disk (HDD vs SSD)" |
| 147 | +echo " 3. Check /tmp/aquiis-startup.log for .NET errors" |
| 148 | +echo " 4. Try: export ELECTRON_NO_ASAR=1 (before running)" |
| 149 | +echo "" |
0 commit comments