<!DOCTYPE html>
<html lang="ko">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>MCP Problem Management System</title>
    <link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500;600&display=swap" rel="stylesheet">
    <style>
        * {
            margin: 0;
            padding: 0;
            box-sizing: border-box;
        }

        body {
            font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
            background: linear-gradient(135deg, #f8fafc 0%, #e2e8f0 100%);
            min-height: 100vh;
            padding: 20px;
            color: #1e293b;
            line-height: 1.6;
        }

        .container {
            max-width: 1400px;
            margin: 0 auto;
            background: #ffffff;
            border-radius: 16px;
            box-shadow: 
                0 4px 6px -1px rgba(0, 0, 0, 0.1),
                0 2px 4px -1px rgba(0, 0, 0, 0.06),
                0 0 0 1px rgba(0, 0, 0, 0.05);
            overflow: hidden;
            position: relative;
        }

        .header {
            background: linear-gradient(135deg, #1e40af 0%, #3b82f6 50%, #60a5fa 100%);
            color: #ffffff;
            padding: 40px 30px;
            text-align: center;
            position: relative;
            overflow: hidden;
        }

        .header::before {
            content: '';
            position: absolute;
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            background: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100"><defs><pattern id="grid" width="10" height="10" patternUnits="userSpaceOnUse"><path d="M 10 0 L 0 0 0 10" fill="none" stroke="rgba(255,255,255,0.1)" stroke-width="0.5"/></pattern></defs><rect width="100" height="100" fill="url(%23grid)"/></svg>');
            opacity: 0.3;
        }

        .header h1 {
            font-family: 'JetBrains Mono', monospace;
            font-size: 2.5rem;
            font-weight: 700;
            margin-bottom: 12px;
            position: relative;
            z-index: 1;
            letter-spacing: -0.025em;
        }

        .header p {
            font-size: 1.125rem;
            opacity: 0.9;
            position: relative;
            z-index: 1;
            font-weight: 400;
        }

        .content {
            padding: 40px;
            background: #ffffff;
        }

        .problems-grid {
            display: grid;
            grid-template-columns: repeat(auto-fill, minmax(380px, 1fr));
            gap: 24px;
            margin-top: 32px;
        }

        .problem-card {
            background: #ffffff;
            border: 1px solid #e2e8f0;
            border-radius: 12px;
            padding: 24px;
            transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
            position: relative;
            overflow: hidden;
            min-height: 280px;
            display: flex;
            flex-direction: column;
            justify-content: space-between;
            box-shadow: 
                0 1px 3px 0 rgba(0, 0, 0, 0.1),
                0 1px 2px 0 rgba(0, 0, 0, 0.06);
        }

        .problem-card::before {
            content: '';
            position: absolute;
            top: 0;
            left: 0;
            right: 0;
            height: 3px;
            background: linear-gradient(90deg, #3b82f6, #8b5cf6, #06b6d4);
            opacity: 0;
            transition: opacity 0.3s ease;
        }

        .problem-card:hover {
            transform: translateY(-4px);
            box-shadow: 
                0 10px 25px -3px rgba(0, 0, 0, 0.1),
                0 4px 6px -2px rgba(0, 0, 0, 0.05);
            border-color: #cbd5e1;
        }

        .problem-card:hover::before {
            opacity: 1;
        }

        .problem-card.solved {
            border-color: #10b981;
            background: linear-gradient(135deg, #ffffff 0%, #f0fdf4 100%);
            box-shadow: 
                0 4px 6px -1px rgba(16, 185, 129, 0.1),
                0 2px 4px -1px rgba(16, 185, 129, 0.06);
        }

        .problem-card.solved::before {
            background: linear-gradient(90deg, #10b981, #34d399, #6ee7b7);
            opacity: 1;
        }

        .problem-id {
            position: absolute;
            top: 16px;
            right: 16px;
            background: #f1f5f9;
            color: #475569;
            padding: 6px 12px;
            border-radius: 20px;
            font-size: 0.75rem;
            font-weight: 600;
            font-family: 'JetBrains Mono', monospace;
            border: 1px solid #e2e8f0;
        }

        .problem-name {
            font-size: 1.25rem;
            font-weight: 600;
            color: #1e293b;
            margin-bottom: 16px;
            padding-right: 80px;
            word-break: break-word;
            line-height: 1.4;
        }

        .problem-status {
            margin-bottom: 20px;
        }

        .status-badge {
            display: inline-flex;
            align-items: center;
            padding: 6px 12px;
            border-radius: 20px;
            font-size: 0.875rem;
            font-weight: 500;
            font-family: 'JetBrains Mono', monospace;
            box-shadow: 0 1px 2px 0 rgba(0, 0, 0, 0.05);
        }

        .status-solved {
            background: #dcfce7;
            color: #166534;
            border: 1px solid #bbf7d0;
        }

        .status-unsolved {
            background: #fef3c7;
            color: #92400e;
            border: 1px solid #fde68a;
        }

        .problem-button {
            width: 100%;
            padding: 12px 20px;
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 8px;
            font-size: 0.875rem;
            font-weight: 600;
            cursor: pointer;
            transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
            box-shadow: 0 1px 3px 0 rgba(59, 130, 246, 0.3);
        }

        .problem-button:hover {
            transform: translateY(-1px);
            box-shadow: 0 4px 12px 0 rgba(59, 130, 246, 0.4);
            background: linear-gradient(135deg, #1d4ed8 0%, #1e40af 100%);
        }

        .options-modal {
            display: none;
            position: fixed;
            top: 0;
            left: 0;
            width: 100%;
            height: 100%;
            background: rgba(0, 0, 0, 0.5);
            z-index: 1000;
            backdrop-filter: blur(8px);
        }

        .modal-content {
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: #ffffff;
            padding: 32px;
            border-radius: 16px;
            box-shadow: 
                0 20px 25px -5px rgba(0, 0, 0, 0.1),
                0 10px 10px -5px rgba(0, 0, 0, 0.04);
            max-width: 500px;
            width: 90%;
            color: #1e293b;
            border: 1px solid #e2e8f0;
        }

        .modal-header {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 24px;
        }

        .modal-title {
            font-size: 1.5rem;
            font-weight: 600;
            color: #1e293b;
            font-family: 'JetBrains Mono', monospace;
        }

        .close-button {
            background: none;
            border: none;
            font-size: 1.5rem;
            cursor: pointer;
            color: #64748b;
            padding: 8px;
            border-radius: 6px;
            transition: all 0.2s ease;
            line-height: 1;
        }

        .close-button:hover {
            color: #1e293b;
            background: #f1f5f9;
        }

        .options-list {
            display: flex;
            flex-direction: column;
            gap: 12px;
        }

        .option-item {
            padding: 16px 20px;
            background: #f8fafc;
            border: 1px solid #e2e8f0;
            border-radius: 8px;
            cursor: pointer;
            transition: all 0.2s ease;
            color: #1e293b;
        }

        .option-item:hover {
            background: #3b82f6;
            color: #ffffff;
            border-color: #3b82f6;
            transform: translateY(-1px);
            box-shadow: 0 4px 12px 0 rgba(59, 130, 246, 0.3);
        }

        .option-title {
            font-weight: 600;
            margin-bottom: 4px;
            font-family: 'JetBrains Mono', monospace;
        }

        .option-description {
            font-size: 0.875rem;
            opacity: 0.8;
        }

        .loading {
            text-align: center;
            padding: 60px 20px;
            color: #64748b;
            font-size: 1.125rem;
        }

        .error {
            background: #fef2f2;
            color: #dc2626;
            padding: 16px;
            border-radius: 8px;
            margin: 20px 0;
            border: 1px solid #fecaca;
            font-weight: 500;
        }

        .refresh-button, .load-button {
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            padding: 12px 24px;
            border-radius: 8px;
            cursor: pointer;
            font-size: 0.875rem;
            font-weight: 600;
            margin-bottom: 20px;
            min-width: 120px;
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
            transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
            box-shadow: 0 1px 3px 0 rgba(59, 130, 246, 0.3);
        }

        .button-group {
            display: flex;
            gap: 12px;
            margin-bottom: 24px;
            flex-wrap: wrap;
        }

        .load-button:hover, .refresh-button:hover {
            transform: translateY(-1px);
            box-shadow: 0 4px 12px 0 rgba(59, 130, 246, 0.4);
            background: linear-gradient(135deg, #1d4ed8 0%, #1e40af 100%);
        }

        .run-action-button {
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 6px;
            padding: 8px 16px;
            font-size: 0.875rem;
            font-weight: 600;
            cursor: pointer;
            box-shadow: 0 1px 3px 0 rgba(59, 130, 246, 0.3);
            transition: all 0.2s cubic-bezier(0.4, 0, 0.2, 1);
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
        }
        
        .run-action-button:hover {
            background: linear-gradient(135deg, #1d4ed8 0%, #1e40af 100%);
            box-shadow: 0 4px 12px 0 rgba(59, 130, 246, 0.4);
            transform: translateY(-1px);
        }

        /* Auto-run spinner */
        .auto-run-spinner {
            display: none;
            position: fixed;
            top: 20px;
            right: 20px;
            width: 40px;
            height: 40px;
            border: 3px solid #e2e8f0;
            border-top: 3px solid #3b82f6;
            border-radius: 50%;
            animation: spin 1s linear infinite;
            z-index: 1000;
            background: #ffffff;
            box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);
        }

        @keyframes spin {
            0% { transform: rotate(0deg); }
            100% { transform: rotate(360deg); }
        }

        /* Responsive design */
        @media (max-width: 768px) {
            .problems-grid {
                grid-template-columns: 1fr;
            }
            
            .problem-card {
                min-width: auto;
            }
            
            .header h1 {
                font-size: 2rem;
            }
            
            .content {
                padding: 24px;
            }
        }

        /* Additional modern touches */
        .problem-card::after {
            content: '';
            position: absolute;
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            background: linear-gradient(135deg, transparent 0%, rgba(59, 130, 246, 0.02) 100%);
            pointer-events: none;
            opacity: 0;
            transition: opacity 0.3s ease;
        }

        .problem-card:hover::after {
            opacity: 1;
        }

        /* Status indicators */
        .status-indicator {
            display: inline-block;
            width: 8px;
            height: 8px;
            border-radius: 50%;
            margin-right: 8px;
        }

        .status-solved .status-indicator {
            background: #10b981;
        }

        .status-unsolved .status-indicator {
            background: #f59e0b;
        }

        /* Top status bar styling */
        .top-status-bar {
            position: fixed !important;
            top: 0;
            left: 0;
            right: 0;
            z-index: 1000;
            background: #ffffff;
            border-bottom: 1px solid #e2e8f0;
            padding: 8px 24px;
            display: flex;
            justify-content: space-between;
            align-items: center;
            box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1);
            height: 48px;
            box-sizing: border-box;
        }

        .status-info {
            display: flex;
            align-items: center;
            gap: 12px;
        }

        .status-text {
            color: #475569;
            font-weight: 500;
            font-family: 'JetBrains Mono', monospace;
            font-size: 0.875rem;
        }

        .status-divider {
            color: #cbd5e1;
            font-weight: 300;
        }

        .auto-run-button {
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 6px;
            padding: 8px 16px;
            font-size: 0.875rem;
            font-weight: 600;
            cursor: pointer;
            transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
            box-shadow: 0 1px 3px 0 rgba(59, 130, 246, 0.3);
        }

        .auto-run-button:hover {
            background: linear-gradient(135deg, #1d4ed8 0%, #1e40af 100%);
            transform: translateY(-1px);
            box-shadow: 0 4px 12px 0 rgba(59, 130, 246, 0.4);
        }

        .auto-run-button.active {
            background: linear-gradient(135deg, #dc2626 0%, #b91c1c 100%);
        }

        .auto-run-button.active:hover {
            background: linear-gradient(135deg, #b91c1c 0%, #991b1b 100%);
            box-shadow: 0 4px 12px 0 rgba(220, 38, 38, 0.4);
        }

        /* Add top margin to body to account for fixed status bar */
        body {
            padding-top: 48px;
            margin: 0;
        }

        /* Auto run animation for top status bar */
        .top-status-bar.auto-running {
            position: fixed;
            overflow: hidden;
            height: 48px;
        }

        .top-status-bar.auto-running::before {
            content: '';
            position: absolute;
            top: 0;
            left: -100%;
            width: 100%;
            height: 100%;
            background: linear-gradient(90deg, transparent, rgba(59, 130, 246, 0.1), transparent);
            animation: autoRunScan 2s ease-in-out infinite;
            pointer-events: none;
            z-index: 1;
        }

        @keyframes autoRunScan {
            0% { left: -100%; }
            50% { left: 100%; }
            100% { left: 100%; }
        }

        /* Auto run indicator */
        .auto-run-indicator {
            display: none;
            align-items: center;
            gap: 8px;
            margin-left: 12px;
            padding: 4px 8px;
            background: rgba(59, 130, 246, 0.1);
            border-radius: 12px;
            border: 1px solid rgba(59, 130, 246, 0.2);
            height: 24px;
            box-sizing: border-box;
        }

        .auto-run-indicator.show {
            display: flex;
        }

        .auto-run-spinner-small {
            width: 12px;
            height: 12px;
            border: 2px solid rgba(59, 130, 246, 0.3);
            border-top: 2px solid #3b82f6;
            border-radius: 50%;
            animation: spin 1s linear infinite;
        }

        .auto-run-text {
            font-size: 0.75rem;
            color: #3b82f6;
            font-weight: 600;
            font-family: 'JetBrains Mono', monospace;
        }

        /* Pulse animation for auto run button */
        .auto-run-button.active {
            animation: pulse 2s ease-in-out infinite;
        }

        @keyframes pulse {
            0%, 100% { 
                box-shadow: 0 1px 3px 0 rgba(220, 38, 38, 0.3);
            }
            50% { 
                box-shadow: 0 1px 3px 0 rgba(220, 38, 38, 0.6), 0 0 0 2px rgba(220, 38, 38, 0.2);
            }
        }
    </style>
</head>
<body>
    <div class="top-status-bar" id="top-status-bar">
        <div class="status-info">
            <span id="running-process-count" class="status-text">Running: -</span>
            <span class="status-divider">|</span>
            <span class="status-text">CTF Server: <span id="ctf-server-url">{{ CTFD_BASE_URL }}</span></span>
            <div class="auto-run-indicator" id="auto-run-indicator">
                <div class="auto-run-spinner-small"></div>
                <span class="auto-run-text" id="auto-run-text">AUTO RUNNING</span>
            </div>
        </div>
        <button id="auto-run-btn" class="auto-run-button">Auto Run</button>
    </div>

    <!-- Auto run spinner -->
    <div id="auto-run-spinner" class="auto-run-spinner"></div>

    <div class="container">
        <div class="header">
            <h1>MCP Problem Management System</h1>
            <p>Select a problem and perform the desired operation</p>
        </div>
        
        <div class="content">
            <div class="button-group">
                <button class="load-button" onclick="loadProblemsFromServer()">Load Problems</button>
                <button class="refresh-button" onclick="location.href='/edit-attempts'">Edit Attempts</button>
                <button class="refresh-button" onclick="downloadAllProblems()">Download All Files</button>
                <button class="refresh-button" onclick="location.href='/status'">Execution Status</button>
            </div>
            
            <div id="loading" class="loading">
                Loading problem list...
            </div>
            
            <div id="error" class="error" style="display: none;">
                Failed to load problem list.
            </div>
            
            <div id="problems-grid" class="problems-grid" style="display: none;">
                <!-- Problem cards will be dynamically added here -->
            </div>
        </div>
    </div>

    <!-- Options modal -->
    <div id="options-modal" class="options-modal">
        <div class="modal-content">
            <div class="modal-header">
                <h3 class="modal-title" id="modal-title">Problem Options</h3>
                <button class="close-button" onclick="closeModal()">&times;</button>
            </div>
            <div id="options-list" class="options-list">
                <!-- Options will be dynamically added here -->
            </div>
        </div>
    </div>

    <script>
        let currentProblemId = null;
        let currentProblemName = null;
        let detailCostInterval = null;

        let autoRunActive = false;
        let autoRunInterval = null;
        let autoRunRefreshInterval = null;
        let autoRunExecuting = false; // Flag to prevent concurrent execution
        let runningProcessesMap = {};
        let executingProcesses = new Set(); // Track currently running processes

        // Load problem list on page load
        document.addEventListener('DOMContentLoaded', function() {
            loadProblems();
            
            // Check running processes on page load
            checkAllRunningProcesses();
            
            // Register as global function for manual checking in developer tools
            window.checkAllRunningProcesses = checkAllRunningProcesses;

            // Update running process count on page load
            updateRunningProcessCount();
            setInterval(updateRunningProcessCount, 5000); // Update every 5 seconds

            document.getElementById('auto-run-btn').addEventListener('click', function() {
                autoRunActive = !autoRunActive;
                this.textContent = autoRunActive ? 'Stop Auto Run' : 'Auto Run';
                this.classList.toggle('active', autoRunActive);
                
                const topStatusBar = document.getElementById('top-status-bar');
                const autoRunIndicator = document.getElementById('auto-run-indicator');
                
                if (autoRunActive) {
                    topStatusBar.classList.add('auto-running');
                    autoRunIndicator.classList.add('show');
                    startAutoRun();
                } else {
                    topStatusBar.classList.remove('auto-running');
                    autoRunIndicator.classList.remove('show');
                    if (autoRunInterval) clearInterval(autoRunInterval);
                    if (autoRunRefreshInterval) clearInterval(autoRunRefreshInterval);
                    hideAutoRunSpinner();
                }
            });
        });

        // Load problem list (from DB)
        async function loadProblems() {
            const loading = document.getElementById('loading');
            const error = document.getElementById('error');
            const problemsGrid = document.getElementById('problems-grid');

            loading.style.display = 'block';
            error.style.display = 'none';
            problemsGrid.style.display = 'none';

            try {
                const response = await fetch('/api/problems');
                const data = await response.json();

                if (data.success) {
                    displayProblems(data.problems);
                } else {
                    throw new Error('Failed to load problem list.');
                }
            } catch (err) {
                error.style.display = 'block';
                error.textContent = err.message;
            } finally {
                loading.style.display = 'none';
            }
        }

        // Load problems from server
        async function loadProblemsFromServer() {
            const loading = document.getElementById('loading');
            const error = document.getElementById('error');
            const problemsGrid = document.getElementById('problems-grid');

            loading.style.display = 'block';
            error.style.display = 'none';
            problemsGrid.style.display = 'none';
            loading.textContent = 'Loading problems from server...';

            try {
                const response = await fetch('/api/problems/load', {
                    method: 'POST',
                    headers: {
                        'Content-Type': 'application/json',
                    }
                });
                const data = await response.json();

                if (data.success) {
                    alert(data.message);
                    // displayProblems(data.problems); // Commented out existing code
                    loadProblems(); // Auto refresh after loading problems
                } else {
                    throw new Error(data.message || 'Failed to load problems.');
                }
            } catch (err) {
                error.style.display = 'block';
                error.textContent = err.message;
            } finally {
                loading.style.display = 'none';
                loading.textContent = 'Loading problem list...';
            }
        }

        // Pre-fetch running processes list and store as map
        async function fetchRunningProcessesMap() {
            try {
                const response = await fetch('/api/running_processes');
                const data = await response.json();
                runningProcessesMap = {};
                if (data.processes) {
                    data.processes.forEach(proc => {
                        runningProcessesMap[`${proc.problem_id}_${proc.key}`] = true;
                    });
                }
            } catch (e) {
                runningProcessesMap = {};
            }
        }

        // Display problem list (completed problems sorted to bottom)
        async function displayProblems(problems) {
            // Store problem list in global variable
            window.currentProblems = problems;
            
            await fetchRunningProcessesMap();
            const problemsGrid = document.getElementById('problems-grid');
            problemsGrid.innerHTML = '';

            // Classify as unsolved/in-progress/completed
            const incomplete = [];
            const complete = [];
            problems.forEach(problem => {
                // Check if all models are solved or 3/3 completed
                const allSolved =
                    (problem.dcipher_gpt_solved && problem.dcipher_claude_solved && problem.dcipher_gemini_solved) ||
                    ([problem.dcipher_gpt_count, problem.dcipher_claude_count, problem.dcipher_gemini_count]
                        .every(cnt => cnt >= 3));
                
                // Check if all items are solved or completed
                const allCompleted = [
                    problem.dcipher_gpt_solved || (problem.dcipher_gpt_count ?? 0) >= 3,
                    problem.dcipher_claude_solved || (problem.dcipher_claude_count ?? 0) >= 3,
                    problem.dcipher_gemini_solved || (problem.dcipher_gemini_count ?? 0) >= 3
                ].every(completed => completed);
                
                if (allCompleted) {
                    complete.push(problem);
                } else {
                    incomplete.push(problem);
                }
            });
            const sorted = [...incomplete, ...complete];
            sorted.forEach(problem => {
                const problemCard = createProblemCard(problem);
                problemsGrid.appendChild(problemCard);
            });
            problemsGrid.style.display = 'grid';
        }

        // Create problem card
        function createProblemCard(problem) {
            const card = document.createElement('div');
            card.className = `problem-card ${problem.solved_by_name ? 'solved' : ''}`;
            
            // Display attempt count, solution status, and run button for each item
            function getAttemptHtml(label, count, solved, actionKey, problem) {
                const isSolved = solved === 1;
                const isCompleted = count >= 3;  // Check 3/3 completion status
                const runningKey = `${problem.id}_${actionKey}`;
                const isRunning = runningProcessesMap[runningKey];
                
                // Handle running case with priority
                if (isRunning) {
                    const color = 'red'; // Display in red when running
                    const status = 'Unsolved';
                    const buttonHtml = '<span style="color: #007bff; font-weight: bold;">Running</span>';
                    return `
                        <div style="display: flex; align-items: center; justify-content: space-between; margin-bottom: 4px;">
                            <span><b>${label}</b>: <span style='color:${color}'>${count}/3</span> <span style='color:${color};font-weight:bold;'>${status}</span></span>
                            ${buttonHtml}
                        </div>
                    `;
                }
                
                // Apply existing logic for non-running cases
                const color = isSolved ? 'green' : (isCompleted ? 'orange' : 'red');
                const status = isSolved ? 'Solved' : (isCompleted ? 'Completed' : 'Unsolved');
                let buttonHtml = '';
                if (isSolved) {
                    buttonHtml = '<span style="color: green; font-weight: bold;">Solved</span>';
                } else if (isCompleted) {
                    buttonHtml = '<span style="color: orange; font-weight: bold;">Completed</span>';
                } else {
                    buttonHtml = `<button class="run-action-button"
                        data-problem-id="${problem.id}"
                        data-problem-name="${encodeURIComponent(problem.name)}"
                        data-category="${problem.category || ''}"
                        onclick="handleSolveActionFromEvent(event, '${actionKey}')">Run</button>`;
                }
                return `
                    <div style="display: flex; align-items: center; justify-content: space-between; margin-bottom: 4px;">
                        <span><b>${label}</b>: <span style='color:${color}'>${count}/3</span> <span style='color:${color};font-weight:bold;'>${status}</span></span>
                        ${buttonHtml}
                    </div>
                `;
            }
            const attemptsHtml = `
                <div style="margin-bottom: 10px;">
                    ${getAttemptHtml('DCIPHER-GPT', problem.dcipher_gpt_count ?? 0, problem.dcipher_gpt_solved ?? 0, 'dcipher_gpt', problem)}
                    ${getAttemptHtml('DCIPHER-CLAUDE', problem.dcipher_claude_count ?? 0, problem.dcipher_claude_solved ?? 0, 'dcipher_claude', problem)}
                    ${getAttemptHtml('DCIPHER-GEMINI', problem.dcipher_gemini_count ?? 0, problem.dcipher_gemini_solved ?? 0, 'dcipher_gemini', problem)}
                </div>
            `;
            
            card.innerHTML = `
                <div class="problem-id">#${problem.id}</div>
                <div class="problem-name">${problem.name}</div>
                ${attemptsHtml}
                <button class="problem-button" 
                    data-problem-id="${problem.id}"
                    data-problem-name="${encodeURIComponent(problem.name)}"
                    data-solves="${problem.solves ?? 0}"
                    data-value="${problem.value ?? 0}"
                    onclick="showOptionsFromEvent(event)">
                    Details
                </button>
            `;

            return card;
        }

        // Safely read values from button and call showOptions
        function showOptionsFromEvent(e) {
            const btn = e.currentTarget;
            const problemId = btn.getAttribute('data-problem-id');
            const problemName = decodeURIComponent(btn.getAttribute('data-problem-name'));
            const solves = parseInt(btn.getAttribute('data-solves')) || 0;
            const value = parseInt(btn.getAttribute('data-value')) || 0;
            showOptions(problemId, problemName, solves, value);
        }

        // Safely read values when run button is clicked and execute
        function handleSolveActionFromEvent(e, actionKey) {
            const btn = e.currentTarget;
            const problemId = btn.getAttribute('data-problem-id');
            const problemName = decodeURIComponent(btn.getAttribute('data-problem-name'));
            const category = btn.getAttribute('data-category') || 'base';
            handleSolveAction(problemId, actionKey, problemName, category);
        }

        // Actual API call function
        async function handleSolveAction(problemId, actionKey, problemName, category) {
            // Prevent duplicate execution
            const processKey = `${problemId}_${actionKey}`;
            if (executingProcesses.has(processKey)) {
                alert('Already running. Please try again later.');
                return;
            }
            
            // Add to running processes
            executingProcesses.add(processKey);
            
            try {
                // Disable run button
                const button = event.target;
                if (button) {
                    button.disabled = true;
                    button.textContent = 'Running...';
                }
                
                // Start execution first
                const response = await fetch(`/api/problem/${problemId}/run`, {
                    method: 'POST',
                    headers: { 'Content-Type': 'application/json' },
                    body: JSON.stringify({
                        key: actionKey,
                        problem_name: problemName,
                        category: category
                    })
                });
                const data = await response.json();
                if (data.success) {
                    alert('Execution started.');
                    // Start execution status monitoring
                    monitorRunStatus(problemId, actionKey);
                    
                    // Refresh page to reflect updated attempt count
                    setTimeout(() => {
                        loadProblems();
                    }, 1000);
                } else {
                    alert(data.message || 'Execution failed.');
                    // Restore button on failure
                    if (button) {
                        button.disabled = false;
                        button.textContent = 'Run';
                    }
                }
            } catch (err) {
                alert('Error occurred during execution: ' + err.message);
                // Restore button on error
                const button = event.target;
                if (button) {
                    button.disabled = false;
                                            button.textContent = 'Run';
                }
            } finally {
                executingProcesses.delete(processKey);
            }
        }
        


        // Monitor execution status
        async function monitorRunStatus(problemId, actionKey) {
            const interval = setInterval(async () => {
                try {
                    const response = await fetch(`/api/problem/${problemId}/run_status?key=${actionKey}`);
                    const data = await response.json();
                    
                    if (!data.running) {
                        clearInterval(interval);
                        console.log(`Process completed: ${problemId}/${actionKey}`);
                        
                        // Refresh page after process completion to reflect latest status
                        setTimeout(() => {
                            loadProblems();
                        }, 2000); // Refresh page after 2 seconds
                        return;
                    }
                    
                    // Output cost information
                    if (data.total_cost !== null) {
                        console.log(`[${problemId}/${actionKey}] Total cost: $${data.total_cost}`);
                    }
                    
                    // Output logs (last 5 lines)
                    if (data.stdout && data.stdout.length > 0) {
                        console.log(`[${problemId}/${actionKey}] STDOUT:`, data.stdout.slice(-5));
                    }
                    if (data.stderr && data.stderr.length > 0) {
                        console.log(`[${problemId}/${actionKey}] STDERR:`, data.stderr.slice(-5));
                    }
                    
                } catch (err) {
                    console.error('Status check error:', err);
                    clearInterval(interval);
                }
            }, 60000); // Check every 60 seconds
        }

        // Check all running processes
        async function checkAllRunningProcesses() {
            try {
                const response = await fetch('/api/running_processes');
                const data = await response.json();
                
                if (data.processes && data.processes.length > 0) {
                    console.log('Running processes:');
                    data.processes.forEach(proc => {
                        console.log(`- ${proc.problem_id}/${proc.key}: ${proc.cmd}`);
                        console.log(`  Start time: ${proc.start_time}`);
                        console.log(`  Return code: ${proc.return_code}`);
                        console.log(`  stdout lines: ${proc.stdout_lines}, stderr lines: ${proc.stderr_lines}`);
                    });
                } else {
                    console.log('No running processes.');
                }
            } catch (err) {
                console.error('Process check error:', err);
            }
        }

        // Show options modal → Changed to problem detail modal
        async function showOptions(problemId, problemName, solves, value) {
            currentProblemId = problemId;
            currentProblemName = problemName;
            try {
                await loadAndDisplayDetail(problemId, problemName, solves, value);
                document.getElementById('options-modal').style.display = 'block';

                // Start real-time cost update polling
                if (detailCostInterval) clearInterval(detailCostInterval);
                detailCostInterval = setInterval(() => {
                    loadAndDisplayDetail(problemId, problemName, solves, value, true);
                }, 3000);
            } catch (err) {
                alert('Failed to load problem information.');
            }
        }

        // Separate detail update function
        async function loadAndDisplayDetail(problemId, problemName, solves, value, silent) {
            try {
                const response = await fetch(`/challenge/${problemId}`);
                const data = await response.json();
                if (data) {
                    displayProblemDetail(data, problemName, solves, value);
                } else if (!silent) {
                    alert('Failed to load problem information.');
                }
            } catch (err) {
                if (!silent) alert('Failed to load problem information.');
            }
        }

        function displayProblemDetail(info, problemName, solves, value) {
            const modalTitle = document.getElementById('modal-title');
            const optionsList = document.getElementById('options-list');

            modalTitle.textContent = `${problemName} - Details`;

            // Display 0 or 'None' if value is missing
            solves = solves ?? 0;
            value = value ?? 0;
            const category = info.category ?? 'None';
            const description = info.description ?? 'None';
            const files = (info.files && info.files.length > 0)
                ? info.files.map(f => {
                    const filename = f.split('/').pop().split('?')[0];
                    return `<li><a href="${f}" target="_blank">${filename}</a></li>`;
                  }).join('')
                : '<li>None</li>';
            const connectionInfo = info.connection_info ?? 'None';

            // Display correct flag
            let correctFlagHtml = '';
            if (info.correct_flag) {
                correctFlagHtml = `
                    <div style="margin-top: 20px; padding: 20px; background: linear-gradient(135deg, #d4edda 0%, #c3e6cb 100%); border: 2px solid #28a745; border-radius: 12px; box-shadow: 0 4px 12px rgba(40, 167, 69, 0.2);">
                        <h4 style="color: #155724; margin-bottom: 15px; font-size: 1.2rem; display: flex; align-items: center; gap: 8px;">
                            <span style="font-size: 1.5rem;">✅</span> Correct Flag
                        </h4>
                        <div style="color: #155724; font-size: 1.1rem;">
                            <code style="background: #ffffff; padding: 12px 18px; border-radius: 8px; font-family: 'Courier New', monospace; font-size: 1.2rem; color: #000; border: 2px solid #28a745; display: block; word-break: break-all; white-space: pre-wrap; box-shadow: 0 2px 4px rgba(0,0,0,0.1); text-align: center;">${info.correct_flag.flag}</code>
                        </div>
                    </div>
                `;
            }

            // Create model cost table
            let modelCostsTable = '';
            if (info.model_costs) {
                const modelNames = {
                    'dcipher_gpt': 'DCIPHER-GPT',
                    'dcipher_claude': 'DCIPHER-CLAUDE',
                    'dcipher_gemini': 'DCIPHER-GEMINI'
                };
                
                modelCostsTable = '<div style="margin-top: 20px;"><h4>Model Attempt Costs</h4><table style="width: 100%; border-collapse: collapse; margin-top: 10px;">';
                modelCostsTable += '<tr style="background: #f8f9fa;"><th style="border: 1px solid #ddd; padding: 8px;">Model</th><th style="border: 1px solid #ddd; padding: 8px; min-width: 120px;">1st</th><th style="border: 1px solid #ddd; padding: 8px; min-width: 120px;">2nd</th><th style="border: 1px solid #ddd; padding: 8px; min-width: 120px;">3rd</th></tr>';
                
                for (const [modelKey, modelName] of Object.entries(modelNames)) {
                    const attempts = info.model_costs[modelKey] || [];
                    modelCostsTable += '<tr>';
                    modelCostsTable += `<td style="border: 1px solid #ddd; padding: 8px; font-weight: bold;">${modelName}</td>`;
                    
                    for (let i = 1; i <= 3; i++) {
                        const attempt = attempts.find(a => a.attempt === i);
                        if (attempt && attempt.cost != null) {
                            modelCostsTable += `<td style="border: 1px solid #ddd; padding: 8px; color: #333; font-weight: bold;">$${attempt.cost.toFixed(3)}</td>`;
                        } else {
                            modelCostsTable += '<td style="border: 1px solid #ddd; padding: 8px; color: #ccc;">-</td>';
                        }
                    }
                    modelCostsTable += '</tr>';
                }
                modelCostsTable += '</table></div>';
            }

            optionsList.innerHTML = `
                <div><b>Solvers:</b> ${solves}</div>
                <div><b>Points:</b> ${value}</div>
                <div><b>Category:</b> ${category}</div>
                <div><b>Description:</b><br>${description}</div>
                <div><b>Connection Info:</b> ${connectionInfo}</div>
                ${correctFlagHtml}
                ${modelCostsTable}
            `;
        }

        // Handle option click
        async function handleOptionClick(option) {
            try {
                const response = await fetch(`/api/problem/${currentProblemId}/action`, {
                    method: 'POST',
                    headers: {
                        'Content-Type': 'application/json',
                    },
                    body: JSON.stringify({
                        action: option.action
                    })
                });

                const data = await response.json();

                if (data.success) {
                    alert(data.message);
                    closeModal();
                } else {
                    alert('Failed to perform operation.');
                }
            } catch (err) {
                alert('Failed to perform operation.');
            }
        }

        // Stop polling when modal is closed
        function closeModal() {
            document.getElementById('options-modal').style.display = 'none';
            currentProblemId = null;
            currentProblemName = null;
            if (detailCostInterval) {
                clearInterval(detailCostInterval);
                detailCostInterval = null;
            }
        }

        // Close modal when clicking outside
        document.getElementById('options-modal').addEventListener('click', function(e) {
            if (e.target === this) {
                closeModal();
            }
        });

        // Download all files button action
        async function downloadAllProblems() {
            if (!confirm('Save all problem files to downloads folder?')) return;
            try {
                const response = await fetch('/api/download_all', { method: 'POST' });
                const data = await response.json();
                if (data.success) {
                    alert(data.message);
                } else {
                    alert(data.message || 'Failed to save.');
                }
            } catch (err) {
                alert('Error occurred while saving.');
            }
        }

        // Update running process count on page load
        async function updateRunningProcessCount() {
            try {
                const response = await fetch('/api/running_processes');
                const data = await response.json();
                const count = data.processes ? data.processes.length : 0;
                document.getElementById('running-process-count').textContent = `Running: ${count}`;
                
                // Update auto run text if active
                if (autoRunActive) {
                    const autoRunText = document.getElementById('auto-run-text');
                    autoRunText.textContent = `AUTO RUNNING (${count})`;
                }
            } catch (e) {
                document.getElementById('running-process-count').textContent = 'Running: -';
                if (autoRunActive) {
                    const autoRunText = document.getElementById('auto-run-text');
                    autoRunText.textContent = 'AUTO RUNNING';
                }
            }
        }

        async function startAutoRun() {
            if (autoRunInterval) clearInterval(autoRunInterval);
            if (autoRunRefreshInterval) clearInterval(autoRunRefreshInterval);
            
            // Start auto run
            autoRunActive = true;
            document.getElementById('auto-run-btn').textContent = 'Stop Auto Run';
            document.getElementById('auto-run-btn').classList.add('active');
            
            // Add animation to top status bar
            const topStatusBar = document.getElementById('top-status-bar');
            const autoRunIndicator = document.getElementById('auto-run-indicator');
            topStatusBar.classList.add('auto-running');
            autoRunIndicator.classList.add('show');
            
            // Execute auto run step every 10 seconds
            autoRunInterval = setInterval(autoRunStep, 10000);
            // Refresh screen every 10 seconds
            autoRunRefreshInterval = setInterval(refreshProblemsDuringAutoRun, 10000);
            
            showAutoRunSpinner();
            await autoRunStep();
        }

        async function autoRunStep() {
            // Prevent concurrent execution
            if (autoRunExecuting) {
                console.log('Auto run is already in progress. Waiting...');
                return;
            }
            
            autoRunExecuting = true;
            
            try {
                // 1. Check current running processes count
                let process_max = 10;
                let runningCount = 0;
                let runningMap = {};
                try {
                    const resp = await fetch('/api/running_processes');
                    const data = await resp.json();
                    runningCount = data.processes ? data.processes.length : 0;
                    if (data.processes) {
                        data.processes.forEach(proc => {
                            runningMap[`${proc.problem_id}_${proc.key}`] = true;
                        });
                    }
                } catch (e) {}
                if (runningCount >= process_max) {
                    console.log(`Maximum process count reached: ${runningCount}/${process_max}`);
                    return;
                }

                // 2. Load problem list
                let problems = [];
                try {
                    const resp = await fetch('/api/problems');
                    const data = await resp.json();
                    if (data.success) problems = data.problems;
                } catch (e) {}

                // 3. Find executable models (unsolved/incomplete/not running)
                const modelKeys = [
                    'dcipher_gpt', 'dcipher_claude', 'dcipher_gemini'
                ];
                let toRun = [];
                for (const p of problems) {
                    for (const key of modelKeys) {
                        if (p[`${key}_solved`] !== 1 && (p[`${key}_count`] ?? 0) < 3 && !runningMap[`${p.id}_${key}`]) {
                            toRun.push({ id: p.id, name: p.name, key, category: p.category || '' });
                        }
                    }
                }
                
                // 4. Execute one by one sequentially
                const canRun = Math.max(0, process_max - runningCount);
                if (toRun.length > 0 && canRun > 0) {
                    const { id, name, key, category } = toRun[0]; // Execute only first item
                    
                    // Prevent duplicate execution
                    const processKey = `${id}_${key}`;
                    if (executingProcesses.has(processKey)) {
                        console.log(`Already running process: ${processKey}`);
                        return;
                    }
                    
                    // Add to running processes
                    executingProcesses.add(processKey);
                    
                    try {
                        console.log(`Auto run: ${id}/${key} - ${name}`);
                        
                        // Check status once more before execution
                        const statusCheck = await fetch(`/api/problem/${id}/run_status?key=${key}`);
                        const statusData = await statusCheck.json();
                        if (statusData.running) {
                            console.log(`Found already running process: ${id}/${key}`);
                            executingProcesses.delete(processKey);
                            return;
                        }
                        
                        const response = await fetch(`/api/problem/${id}/run`, {
                            method: 'POST',
                            headers: { 'Content-Type': 'application/json' },
                            body: JSON.stringify({ key, problem_name: name, category })
                        });
                        
                        const responseData = await response.json();
                        if (responseData.success) {
                            console.log(`Auto run completed: ${id}/${key}`);
                        } else {
                            console.log(`Auto run failed: ${id}/${key} - ${responseData.message}`);
                        }
                        
                        // Wait 3 seconds
                        await new Promise(resolve => setTimeout(resolve, 3000));
                    } catch (e) {
                        console.error(`Auto run failed: ${id}/${key}`, e);
                    } finally {
                        // Remove process key after execution
                        executingProcesses.delete(processKey);
                    }
                }
                
                // 5. Stop auto run when all items are executed/solved/completed
                if (toRun.length === 0 && runningCount === 0) {
                    autoRunActive = false;
                    document.getElementById('auto-run-btn').textContent = 'Auto Run';
                    document.getElementById('auto-run-btn').classList.remove('active');
                    
                    // Remove animation from top status bar
                    const topStatusBar = document.getElementById('top-status-bar');
                    const autoRunIndicator = document.getElementById('auto-run-indicator');
                    topStatusBar.classList.remove('auto-running');
                    autoRunIndicator.classList.remove('show');
                    
                    if (autoRunInterval) clearInterval(autoRunInterval);
                    if (autoRunRefreshInterval) clearInterval(autoRunRefreshInterval);
                    hideAutoRunSpinner();
                    alert('All executable items have been auto-run!');
                }
            } finally {
                // Reset flag after execution
                autoRunExecuting = false;
            }
        }

        // Refresh screen during auto run
        async function refreshProblemsDuringAutoRun() {
            if (!autoRunActive) return;
            
            try {
                // Update running processes map
                await fetchRunningProcessesMap();
                
                // Reload problem list
                const response = await fetch('/api/problems');
                const data = await response.json();
                
                if (data.success) {
                    // Update existing cards (without refresh)
                    updateProblemCards(data.problems);
                }
            } catch (e) {
                console.log('Screen refresh failed during auto run:', e);
            }
        }

        // Update problem cards (without refresh)
        function updateProblemCards(problems) {
            const problemsGrid = document.getElementById('problems-grid');
            const existingCards = problemsGrid.querySelectorAll('.problem-card');
            
            // Update data of existing cards
            existingCards.forEach(card => {
                const problemId = card.querySelector('.problem-id').textContent.replace('#', '');
                const problem = problems.find(p => p.id == problemId);
                
                if (problem) {
                    // Update attempt count and solution status
                    updateProblemCardContent(card, problem);
                }
            });
        }

            // Update problem card content
            function updateProblemCardContent(card, problem) {
                // Display attempt count, solution status, and run button for each item
                function getAttemptHtml(label, count, solved, actionKey, problem) {
                    const isSolved = solved === 1;
                    const isCompleted = count >= 3;  // Check 3/3 completion status
                    const runningKey = `${problem.id}_${actionKey}`;
                    const isRunning = runningProcessesMap[runningKey];
                    
                    // Handle running case with priority
                    if (isRunning) {
                        const color = 'red'; // Display in red when running
                        const status = 'Unsolved';
                        const buttonHtml = '<span style="color: #007bff; font-weight: bold;">Running</span>';
                        return `
                            <div style="display: flex; align-items: center; justify-content: space-between; margin-bottom: 4px;">
                                <span><b>${label}</b>: <span style='color:${color}'>${count}/3</span> <span style='color:${color};font-weight:bold;'>${status}</span></span>
                                ${buttonHtml}
                            </div>
                        `;
                    }
                    
                    // Apply existing logic for non-running cases
                    const color = isSolved ? 'green' : (isCompleted ? 'orange' : 'red');
                    const status = isSolved ? 'Solved' : (isCompleted ? 'Completed' : 'Unsolved');
                    let buttonHtml = '';
                    if (isSolved) {
                        buttonHtml = '<span style="color: green; font-weight: bold;">Solved</span>';
                    } else if (isCompleted) {
                        buttonHtml = '<span style="color: orange; font-weight: bold;">Completed</span>';
                    } else {
                        buttonHtml = `<button class="run-action-button"
                            data-problem-id="${problem.id}"
                            data-problem-name="${encodeURIComponent(problem.name)}"
                            data-category="${problem.category || ''}"
                            onclick="handleSolveActionFromEvent(event, '${actionKey}')">Run</button>`;
                    }
                    return `
                        <div style="display: flex; align-items: center; justify-content: space-between; margin-bottom: 4px;">
                            <span><b>${label}</b>: <span style='color:${color}'>${count}/3</span> <span style='color:${color};font-weight:bold;'>${status}</span></span>
                            ${buttonHtml}
                        </div>
                    `;
                }
            
            const attemptsHtml = `
                <div style="margin-bottom: 10px;">
                    ${getAttemptHtml('DCIPHER-GPT', problem.dcipher_gpt_count ?? 0, problem.dcipher_gpt_solved ?? 0, 'dcipher_gpt', problem)}
                    ${getAttemptHtml('DCIPHER-CLAUDE', problem.dcipher_claude_count ?? 0, problem.dcipher_claude_solved ?? 0, 'dcipher_claude', problem)}
                    ${getAttemptHtml('DCIPHER-GEMINI', problem.dcipher_gemini_count ?? 0, problem.dcipher_gemini_solved ?? 0, 'dcipher_gemini', problem)}
                </div>
            `;
            
            // Update card content
            const attemptsContainer = card.querySelector('div[style*="margin-bottom: 10px"]');
            if (attemptsContainer) {
                attemptsContainer.outerHTML = attemptsHtml;
            }
        }

        // Show auto run spinner
        function showAutoRunSpinner() {
            const spinner = document.getElementById('auto-run-spinner');
            const status = document.getElementById('auto-run-status');
            spinner.classList.add('show');
            status.classList.add('show');
        }

        // Hide auto run spinner
        function hideAutoRunSpinner() {
            const spinner = document.getElementById('auto-run-spinner');
            const status = document.getElementById('auto-run-status');
            spinner.classList.remove('show');
            status.classList.remove('show');
        }
    </script>
</body>
</html> 