<!DOCTYPE html>
<html lang="ko">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Execution Status Monitoring</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;
            padding-top: 48px;
        }

        .container {
            max-width: 1200px;
            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;
        }

        .nav-buttons {
            display: flex;
            gap: 12px;
            margin-bottom: 32px;
            flex-wrap: wrap;
        }

        .nav-button {
            padding: 12px 24px;
            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);
            text-decoration: none;
            display: inline-block;
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
            box-shadow: 0 1px 3px 0 rgba(59, 130, 246, 0.3);
        }

        .nav-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%);
        }

        .control-buttons {
            display: flex;
            gap: 12px;
            margin-bottom: 32px;
            flex-wrap: wrap;
        }

        .control-button {
            padding: 10px 20px;
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 6px;
            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);
        }

        .control-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);
        }

        .control-button.secondary {
            background: linear-gradient(135deg, #64748b 0%, #475569 100%);
            color: #ffffff;
        }

        .control-button.secondary:hover {
            background: linear-gradient(135deg, #475569 0%, #334155 100%);
        }

        .status-section {
            margin-bottom: 40px;
        }

        .section-title {
            font-size: 1.5rem;
            font-weight: 600;
            color: #1e293b;
            margin-bottom: 20px;
            padding-bottom: 12px;
            border-bottom: 2px solid #e2e8f0;
            font-family: 'JetBrains Mono', monospace;
        }

        .process-grid {
            display: grid;
            grid-template-columns: repeat(auto-fill, minmax(400px, 1fr));
            gap: 20px;
            margin-bottom: 30px;
        }

        .process-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;
            box-shadow: 
                0 1px 3px 0 rgba(0, 0, 0, 0.1),
                0 1px 2px 0 rgba(0, 0, 0, 0.06);
        }

        .process-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;
        }

        .process-card:hover {
            transform: translateY(-2px);
            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;
        }

        .process-card:hover::before {
            opacity: 1;
        }

        .process-card.running {
            border-color: #3b82f6;
            background: linear-gradient(135deg, #ffffff 0%, #eff6ff 100%);
        }

        .process-card.running::before {
            background: linear-gradient(90deg, #3b82f6, #1d4ed8, #1e40af);
            opacity: 1;
        }

        .process-card.completed {
            border-color: #10b981;
            background: linear-gradient(135deg, #ffffff 0%, #f0fdf4 100%);
        }

        .process-card.completed::before {
            background: linear-gradient(90deg, #10b981, #34d399, #6ee7b7);
            opacity: 1;
        }

        .process-card.failed {
            border-color: #ef4444;
            background: linear-gradient(135deg, #ffffff 0%, #fef2f2 100%);
        }

        .process-card.failed::before {
            background: linear-gradient(90deg, #ef4444, #f87171, #fca5a5);
            opacity: 1;
        }

        .process-header {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 16px;
        }

        .process-name {
            font-size: 1.125rem;
            font-weight: 600;
            color: #1e293b;
            font-family: 'JetBrains Mono', monospace;
        }

        .process-status {
            padding: 6px 12px;
            border-radius: 20px;
            font-size: 0.75rem;
            font-weight: 600;
            font-family: 'JetBrains Mono', monospace;
            text-transform: uppercase;
            letter-spacing: 0.05em;
        }

        .status-running {
            background: #dbeafe;
            color: #1e40af;
            border: 1px solid #93c5fd;
        }

        .status-completed {
            background: #dcfce7;
            color: #166534;
            border: 1px solid #bbf7d0;
        }

        .status-failed {
            background: #fee2e2;
            color: #dc2626;
            border: 1px solid #fca5a5;
        }

        .process-info {
            margin-bottom: 16px;
        }

        .info-item {
            display: flex;
            justify-content: space-between;
            margin-bottom: 8px;
            font-size: 0.875rem;
        }

        .info-label {
            color: #64748b;
            font-weight: 500;
        }

        .info-value {
            color: #1e293b;
            font-weight: 600;
            font-family: 'JetBrains Mono', monospace;
        }

        .process-actions {
            display: flex;
            gap: 8px;
            flex-wrap: wrap;
        }

        .action-button {
            padding: 8px 16px;
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 6px;
            font-size: 0.75rem;
            font-weight: 600;
            cursor: pointer;
            transition: all 0.2s 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);
        }

        .action-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);
        }

        .action-button.danger {
            background: linear-gradient(135deg, #ef4444 0%, #dc2626 100%);
        }

        .action-button.danger:hover {
            background: linear-gradient(135deg, #dc2626 0%, #b91c1c 100%);
            box-shadow: 0 4px 12px 0 rgba(239, 68, 68, 0.4);
        }

        .action-button.success {
            background: linear-gradient(135deg, #10b981 0%, #059669 100%);
        }

        .action-button.success:hover {
            background: linear-gradient(135deg, #059669 0%, #047857 100%);
            box-shadow: 0 4px 12px 0 rgba(16, 185, 129, 0.4);
        }

        .log-container {
            background: #f8fafc;
            border: 1px solid #e2e8f0;
            border-radius: 8px;
            padding: 16px;
            margin-top: 16px;
            max-height: 200px;
            overflow-y: auto;
            font-family: 'JetBrains Mono', monospace;
            font-size: 0.875rem;
            line-height: 1.5;
            color: #374151;
        }

        .log-entry {
            margin-bottom: 4px;
            padding: 4px 0;
        }

        .log-entry.error {
            color: #dc2626;
        }

        .log-entry.warning {
            color: #d97706;
        }

        .log-entry.success {
            color: #059669;
        }

        .log-entry.info {
            color: #2563eb;
        }

        .status-summary {
            background: #f8fafc;
            border: 1px solid #e2e8f0;
            border-radius: 12px;
            padding: 24px;
            margin-bottom: 32px;
        }

        .summary-grid {
            display: grid;
            grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
            gap: 20px;
        }

        .summary-item {
            text-align: center;
            padding: 16px;
            background: #ffffff;
            border-radius: 8px;
            border: 1px solid #e2e8f0;
        }

        .summary-number {
            font-size: 2rem;
            font-weight: 700;
            color: #1e293b;
            font-family: 'JetBrains Mono', monospace;
        }

        .summary-label {
            font-size: 0.875rem;
            color: #64748b;
            margin-top: 4px;
            font-weight: 500;
        }

        .summary-running .summary-number {
            color: #1e40af;
        }

        .summary-completed .summary-number {
            color: #059669;
        }

        .summary-failed .summary-number {
            color: #dc2626;
        }

        /* Responsive design */
        @media (max-width: 768px) {
            .process-grid {
                grid-template-columns: 1fr;
            }
            
            .summary-grid {
                grid-template-columns: repeat(2, 1fr);
            }
            
            .header h1 {
                font-size: 2rem;
            }
            
            .content {
                padding: 24px;
            }
            
            .nav-buttons, .control-buttons {
                flex-direction: column;
            }
            
            .top-status-bar {
                padding: 0 12px;
            }
            
            .status-info {
                gap: 8px;
            }
            
            .status-text {
                font-size: 0.75rem;
            }
            
            .auto-run-button {
                font-size: 0.75rem;
                padding: 6px 12px;
            }
        }

        /* Loading animation */
        .loading-spinner {
            display: inline-block;
            width: 16px;
            height: 16px;
            border: 2px solid #e2e8f0;
            border-top: 2px solid #3b82f6;
            border-radius: 50%;
            animation: spin 1s linear infinite;
            margin-right: 8px;
        }

        @keyframes spin {
            0% { transform: rotate(0deg); }
            100% { transform: rotate(360deg); }
        }

        /* Progress bar */
        .progress-bar {
            width: 100%;
            height: 8px;
            background: #e2e8f0;
            border-radius: 4px;
            overflow: hidden;
            margin-top: 12px;
        }

        .progress-fill {
            height: 100%;
            background: linear-gradient(90deg, #3b82f6, #1d4ed8);
            border-radius: 4px;
            transition: width 0.3s ease;
        }

        .progress-fill.completed {
            background: linear-gradient(90deg, #10b981, #059669);
        }

        .progress-fill.failed {
            background: linear-gradient(90deg, #ef4444, #dc2626);
        }

        /* Error message styling */
        .error {
            background: #fef2f2;
            border: 1px solid #fecaca;
            border-radius: 12px;
            padding: 20px;
            margin: 24px 0;
            display: flex;
            align-items: center;
            justify-content: center;
        }

        .error-content {
            display: flex;
            align-items: center;
            gap: 12px;
        }

        .error-icon {
            font-size: 1.5rem;
        }

        .error-text {
            color: #dc2626;
            font-weight: 600;
            font-size: 1rem;
            font-family: 'JetBrains Mono', monospace;
        }

        /* Section controls styling */
        .section-controls {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 20px;
            padding-bottom: 12px;
            border-bottom: 2px solid #e2e8f0;
        }

        .section-toggle {
            display: flex;
            align-items: center;
        }

        .toggle-button {
            background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
            color: #ffffff;
            border: none;
            border-radius: 8px;
            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);
            display: flex;
            align-items: center;
            gap: 6px;
        }

        .toggle-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);
        }

        .toggle-button.active {
            background: linear-gradient(135deg, #10b981 0%, #059669 100%);
        }

        .toggle-button.active:hover {
            background: linear-gradient(135deg, #059669 0%, #047857 100%);
            box-shadow: 0 4px 12px 0 rgba(16, 185, 129, 0.4);
        }

        .toggle-icon {
            font-size: 0.875rem;
        }

        .toggle-text {
            font-size: 0.75rem;
        }

        /* Process count styling */
        .process-count {
            background: #f1f5f9;
            color: #475569;
            padding: 4px 8px;
            border-radius: 12px;
            font-size: 0.75rem;
            font-weight: 600;
            font-family: 'JetBrains Mono', monospace;
            border: 1px solid #e2e8f0;
            margin-left: 8px;
        }

        /* No processes message */
        .no-processes {
            text-align: center;
            padding: 40px 20px;
            color: #64748b;
            font-size: 1.125rem;
            font-weight: 500;
            background: #f8fafc;
            border: 1px solid #e2e8f0;
            border-radius: 12px;
            margin: 20px 0;
        }

        /* Top status bar */
        .top-status-bar {
            position: fixed;
            top: 0;
            left: 0;
            right: 0;
            height: 48px;
            background: #ffffff;
            border-bottom: 1px solid #e2e8f0;
            display: flex;
            align-items: center;
            justify-content: space-between;
            padding: 0 20px;
            z-index: 1000;
            box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1);
        }

        .status-info {
            display: flex;
            align-items: center;
            gap: 16px;
        }

        .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%);
            color: #ffffff;
        }

        .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);
        }

        /* 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 class="status-text">Status Monitor</span>
            <span class="status-divider">|</span>
            <span class="status-text">Real-time Updates</span>
        </div>
        <button id="auto-run-btn" class="auto-run-button">AUTO REFRESH OFF</button>
    </div>

    <div class="container">
        <div class="header">
            <h1>Execution Status Monitoring</h1>
            <p>Real-time monitoring of running and completed processes</p>
        </div>
        
        <div class="content">
            <div class="nav-buttons">
                <a href="/" class="nav-button">Main Page</a>
                <a href="/edit-attempts" class="nav-button">Edit Attempts</a>
            </div>
            
            <div id="loading" class="loading" style="display: none;">
                Loading status...
            </div>
            
            <div id="error" class="error" style="display: none;">
                <div class="error-content">
                    <div class="error-icon">⚠️</div>
                    <div class="error-text">Failed to load status</div>
                </div>
            </div>
            
            <div id="status-content">
                <div class="status-section">
                    <div class="section-controls">
                        <h2 class="section-title">
                            Running Processes
                            <span class="process-count" id="running-count">0</span>
                        </h2>
                        <div class="section-toggle">
                            <button class="toggle-button active" onclick="toggleSection('running')">
                                <span class="toggle-icon">👁️</span>
                                <span class="toggle-text">Show</span>
                            </button>
                        </div>
                    </div>
                    <div id="running-processes" class="process-grid">
                        <!-- Running processes will be dynamically added here -->
                    </div>
                </div>
                
                <div class="status-section">
                    <div class="section-controls">
                        <h2 class="section-title">
                            Completed Processes
                            <span class="process-count" id="completed-count">0</span>
                        </h2>
                        <div class="section-toggle">
                            <button class="toggle-button active" onclick="toggleSection('completed')">
                                <span class="toggle-icon">👁️</span>
                                <span class="toggle-text">Show</span>
                            </button>
                        </div>
                    </div>
                    <div id="completed-processes" class="process-grid">
                        <!-- Completed processes will be dynamically added here -->
                    </div>
                </div>
            </div>
        </div>
    </div>



    <script>
        let autoRefreshInterval = null;
        let autoRefreshEnabled = false;

        // Auto refresh button event listener
        document.addEventListener('DOMContentLoaded', function() {
            document.getElementById('auto-run-btn').addEventListener('click', function() {
                autoRefreshEnabled = !autoRefreshEnabled;
                this.textContent = autoRefreshEnabled ? 'AUTO REFRESH ON (30s)' : 'AUTO REFRESH OFF';
                this.classList.toggle('active', autoRefreshEnabled);
                
                if (autoRefreshEnabled) {
                    autoRefreshInterval = setInterval(refreshStatus, 30000); // 30 seconds
                } else {
                    if (autoRefreshInterval) {
                        clearInterval(autoRefreshInterval);
                        autoRefreshInterval = null;
                    }
                }
            });
        });

        function refreshStatus() {
            // Hide error message initially
            document.getElementById('error').style.display = 'none';
            
            // Fetch running processes
            fetch('/api/running_processes')
                .then(response => response.json())
                .then(data => {
                    const processes = data.processes || [];
                    updateRunningProcesses(processes);
                    
                    // If auto refresh is ON, fetch detailed cost info for each process
                    if (autoRefreshEnabled && processes.length > 0) {
                        fetchDetailedCostInfo(processes);
                    } else if (!autoRefreshEnabled) {
                        // Only start cost monitoring if auto refresh is OFF
                        setTimeout(startCostMonitoring, 1000);
                    }
                })
                .catch(error => {
                    console.error('Error fetching running processes:', error);
                    updateRunningProcesses([]);
                });
            
            // Fetch completed processes
            fetch('/api/completed_processes')
                .then(response => response.json())
                .then(data => {
                    updateCompletedProcesses(data.processes || []);
                })
                .catch(error => {
                    console.error('Error fetching completed processes:', error);
                    updateCompletedProcesses([]);
                });
        }

        function fetchDetailedCostInfo(processes) {
            // Fetch cost info for each running process
            processes.forEach(process => {
                fetch(`/api/problem/${process.problem_id}/run_status?key=${process.key}`)
                    .then(response => response.json())
                    .then(data => {
                        if (data.running && data.total_cost !== undefined) {
                            updateProcessCost(process.seed, data.total_cost);
                        }
                    })
                    .catch(error => {
                        console.error('Error fetching cost info for process:', error);
                    });
            });
        }

        function updateRunningProcesses(processes) {
            const container = document.getElementById('running-processes');
            const countElement = document.getElementById('running-count');
            
            countElement.textContent = processes.length;
            
            if (processes.length === 0) {
                container.innerHTML = '<div class="no-processes">No running processes</div>';
                return;
            }
            
            container.innerHTML = processes.map(process => `
                <div class="process-card running" data-process-id="${process.problem_id}" data-model-key="${process.key}" data-seed="${process.seed}">
                    <div class="process-header">
                        <div class="process-name">${process.problem_id} - ${process.problem_name || 'Unknown Problem'}</div>
                        <div class="process-status status-running">Running</div>
                    </div>
                    <div class="process-info">
                        <div class="info-item">
                            <span class="info-label">Model:</span>
                            <span class="info-value">${process.model_name || process.key}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Seed:</span>
                            <span class="info-value">${process.seed}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Start Time:</span>
                            <span class="info-value">${new Date(process.start_time).toLocaleString()}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Duration:</span>
                            <span class="info-value">${formatDuration(Date.now() - new Date(process.start_time).getTime())}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Cost:</span>
                            <span class="info-value">${process.total_cost ? `$${process.total_cost.toFixed(4)}` : 'N/A'}</span>
                        </div>
                    </div>
                    <div class="process-actions">
                        <button class="action-button danger" onclick="terminateProcess('${process.seed}')">Terminate</button>
                    </div>
                    <div class="log-container" id="log-${process.seed}">
                        Loading logs...
                    </div>
                </div>
            `).join('');
        }

        function updateCompletedProcesses(processes) {
            const container = document.getElementById('completed-processes');
            const countElement = document.getElementById('completed-count');
            
            countElement.textContent = processes.length;
            
            if (processes.length === 0) {
                container.innerHTML = '<div class="no-processes">No completed processes</div>';
                return;
            }
            
            container.innerHTML = processes.map(process => `
                <div class="process-card completed collapsed" data-process-id="${process.problem_id}" data-model-key="${process.key}" data-seed="${process.seed}">
                    <div class="process-header">
                        <div class="process-name">${process.problem_id} - ${process.problem_name || 'Unknown Problem'}</div>
                        <div class="process-status status-completed">Completed</div>
                    </div>
                    <div class="process-info">
                        <div class="info-item">
                            <span class="info-label">Model:</span>
                            <span class="info-value">${process.model_name || process.key}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Seed:</span>
                            <span class="info-value">${process.seed}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Start Time:</span>
                            <span class="info-value">${new Date(process.start_time).toLocaleString()}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">End Time:</span>
                            <span class="info-value">${process.end_time ? new Date(process.end_time).toLocaleString() : 'N/A'}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Status:</span>
                            <span class="info-value">${process.status}</span>
                        </div>
                        <div class="info-item">
                            <span class="info-label">Cost:</span>
                            <span class="info-value">${process.total_cost ? `$${process.total_cost.toFixed(4)}` : 'N/A'}</span>
                        </div>
                    </div>
                    <div class="log-container" id="log-${process.seed}" style="display: none;">
                        ${process.stdout || 'No output'}
                    </div>
                </div>
            `).join('');
        }

        function formatDuration(ms) {
            const seconds = Math.floor(ms / 1000);
            const minutes = Math.floor(seconds / 60);
            const hours = Math.floor(minutes / 60);
            
            if (hours > 0) {
                return `${hours}h ${minutes % 60}m ${seconds % 60}s`;
            } else if (minutes > 0) {
                return `${minutes}m ${seconds % 60}s`;
            } else {
                return `${seconds}s`;
            }
        }

        function streamLogs(seed) {
            const logElement = document.getElementById(`log-${seed}`);
            logElement.textContent = 'Streaming logs...';
            
            // Get the model key from the process card using seed
            const processCard = document.querySelector(`[data-seed="${seed}"]`);
            const modelKey = processCard ? processCard.getAttribute('data-model-key') : null;
            const problemId = processCard ? processCard.getAttribute('data-process-id') : null;
            
            if (!modelKey || !problemId) {
                logElement.textContent = 'Error: Model key or problem ID not found';
                return;
            }
            
            fetch(`/api/stream_logs/${problemId}?key=${modelKey}`)
                .then(response => {
                    if (!response.ok) {
                        throw new Error(`HTTP ${response.status}`);
                    }
                    const reader = response.body.getReader();
                    const decoder = new TextDecoder();
                    
                    function readStream() {
                        reader.read().then(({done, value}) => {
                            if (done) return;
                            
                            const chunk = decoder.decode(value);
                            logElement.textContent += chunk;
                            logElement.scrollTop = logElement.scrollHeight;
                            
                            readStream();
                        });
                    }
                    
                    readStream();
                })
                .catch(error => {
                    logElement.textContent = 'Error streaming logs: ' + error.message;
                });
        }

        function terminateProcess(seed) {
            // Get the model key from the process card using seed
            const processCard = document.querySelector(`[data-seed="${seed}"]`);
            const modelKey = processCard ? processCard.getAttribute('data-model-key') : null;
            const problemId = processCard ? processCard.getAttribute('data-process-id') : null;
            
            if (!modelKey || !problemId) {
                alert('Error: Model key or problem ID not found');
                return;
            }
            
            fetch(`/api/process/${problemId}/terminate?key=${modelKey}`, {
                method: 'POST'
            })
            .then(response => response.json())
            .then(data => {
                if (data.success) {
                    alert('Process terminated successfully');
                    refreshStatus();
                } else {
                    alert('Failed to terminate process: ' + (data.message || 'Unknown error'));
                }
            })
            .catch(error => {
                console.error('Error terminating process:', error);
                alert('Error terminating process');
            });
        }

        function toggleLogs(seed) {
            const logElement = document.getElementById(`log-${seed}`);
            const isVisible = logElement.style.display !== 'none';
            logElement.style.display = isVisible ? 'none' : 'block';
        }

        function downloadLogs(seed) {
            // Get the problem ID from the process card using seed
            const processCard = document.querySelector(`[data-seed="${seed}"]`);
            const problemId = processCard ? processCard.getAttribute('data-process-id') : null;
            
            if (!problemId) {
                alert('Error: Problem ID not found');
                return;
            }
            
            window.open(`/api/logs/${problemId}`, '_blank');
        }

        function toggleSection(section) {
            const button = event.target.closest('.toggle-button');
            const grid = section === 'running' ? 
                document.getElementById('running-processes') : 
                document.getElementById('completed-processes');
            const toggleText = button.querySelector('.toggle-text');
            
            if (button.classList.contains('active')) {
                button.classList.remove('active');
                toggleText.textContent = 'Show';
                grid.style.display = 'none';
            } else {
                button.classList.add('active');
                toggleText.textContent = 'Hide';
                grid.style.display = 'grid';
            }
        }



        function toggleCompleted() {
            const completedCards = document.querySelectorAll('.process-card.completed');
            completedCards.forEach(card => {
                card.classList.toggle('hidden');
            });
        }

        // Real-time cost update function
        function updateProcessCost(seed, cost) {
            const processCard = document.querySelector(`[data-seed="${seed}"]`);
            if (processCard) {
                const infoItems = processCard.querySelectorAll('.info-item');
                for (const item of infoItems) {
                    const label = item.querySelector('.info-label');
                    if (label && label.textContent === 'Cost:') {
                        const valueElement = item.querySelector('.info-value');
                        if (valueElement) {
                            valueElement.textContent = cost ? `$${parseFloat(cost).toFixed(4)}` : 'N/A';
                            break;
                        }
                    }
                }
            }
        }

        // Monitor cost updates for running processes
        function startCostMonitoring() {
            const runningCards = document.querySelectorAll('.process-card.running');
            runningCards.forEach(card => {
                const seed = card.getAttribute('data-seed');
                const problemId = card.getAttribute('data-process-id');
                const modelKey = card.getAttribute('data-model-key');
                
                if (seed && problemId && modelKey) {
                    monitorProcessCost(seed, problemId, modelKey);
                }
            });
        }

        function monitorProcessCost(seed, problemId, modelKey) {
            // Don't start individual monitoring if auto refresh is ON
            if (autoRefreshEnabled) {
                return;
            }
            
            // Check cost updates every 5 seconds only when auto refresh is OFF
            const costInterval = setInterval(() => {
                // Stop monitoring if auto refresh is turned ON
                if (autoRefreshEnabled) {
                    clearInterval(costInterval);
                    return;
                }
                
                fetch(`/api/problem/${problemId}/run_status?key=${modelKey}`)
                    .then(response => response.json())
                    .then(data => {
                        if (data.running && data.total_cost !== undefined) {
                            updateProcessCost(seed, data.total_cost);
                        } else {
                            // Process might have finished, stop monitoring
                            clearInterval(costInterval);
                        }
                    })
                    .catch(error => {
                        console.error('Error monitoring cost for process:', error);
                        clearInterval(costInterval);
                    });
            }, 5000);
        }

        // Initial load
        refreshStatus();
        
        // Start cost monitoring after initial load
        setTimeout(startCostMonitoring, 2000);
    </script>
</body>
</html> 