""" BreakPilot Studio - Mac Mini Control Module Funktionen: - Fernsteuerung des Mac Mini Servers - Status-Uebersicht (Ping, SSH, Docker, Ollama) - Docker Container Management - Ollama Modell-Downloads mit Fortschrittsanzeige - Power Management (Wake-on-LAN, Restart, Shutdown) """ class MacMiniControlModule: """Modul fuer die Mac Mini Server-Steuerung.""" @staticmethod def get_css() -> str: """CSS fuer das Mac Mini Control Modul.""" return """ /* ============================================= MAC MINI CONTROL MODULE ============================================= */ .panel-mac-mini { display: flex; flex-direction: column; height: 100%; background: var(--bp-bg); overflow-y: auto; } .mac-mini-header { padding: 32px 40px 24px; background: var(--bp-surface); border-bottom: 1px solid var(--bp-border); display: flex; justify-content: space-between; align-items: center; } .mac-mini-header h1 { font-size: 28px; font-weight: 700; color: var(--bp-text); margin: 0; display: flex; align-items: center; gap: 12px; } .mac-mini-status-badge { padding: 6px 16px; border-radius: 20px; font-size: 14px; font-weight: 600; } .mac-mini-status-badge.online { background: rgba(34, 197, 94, 0.2); color: #22c55e; border: 1px solid #22c55e; } .mac-mini-status-badge.offline { background: rgba(239, 68, 68, 0.2); color: #ef4444; border: 1px solid #ef4444; } .mac-mini-status-badge.checking { background: rgba(251, 191, 36, 0.2); color: #fbbf24; border: 1px solid #fbbf24; } .mac-mini-content { padding: 32px 40px; flex: 1; } .mac-mini-controls { display: flex; gap: 12px; margin-bottom: 24px; flex-wrap: wrap; } .mac-mini-controls .btn { padding: 10px 20px; border-radius: 8px; font-weight: 600; cursor: pointer; border: none; transition: all 0.2s; } .mac-mini-controls .btn:hover { transform: translateY(-2px); box-shadow: 0 4px 12px rgba(0,0,0,0.2); } .mac-mini-controls .btn-wake { background: #22c55e; color: white; } .mac-mini-controls .btn-restart { background: #f59e0b; color: white; } .mac-mini-controls .btn-shutdown { background: #ef4444; color: white; } .mac-mini-controls .btn-refresh { background: var(--bp-surface-elevated); color: var(--bp-text); border: 1px solid var(--bp-border); } .mac-mini-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(280px, 1fr)); gap: 20px; margin-bottom: 24px; } .mac-mini-card { background: var(--bp-surface); border: 1px solid var(--bp-border); border-radius: 12px; padding: 20px; } .mac-mini-card h3 { color: var(--bp-text); font-size: 16px; margin: 0 0 16px 0; display: flex; align-items: center; gap: 8px; } .mac-mini-card-row { display: flex; justify-content: space-between; padding: 10px 0; border-bottom: 1px solid var(--bp-border); } .mac-mini-card-row:last-child { border-bottom: none; } .mac-mini-card-label { color: var(--bp-text-muted); } .mac-mini-card-value { font-weight: 500; } .mac-mini-card-value.ok { color: #22c55e; } .mac-mini-card-value.error { color: #ef4444; } /* Docker Containers */ .mac-mini-container-list { display: flex; flex-direction: column; gap: 8px; } .mac-mini-container-item { display: flex; justify-content: space-between; align-items: center; padding: 12px 16px; background: var(--bp-surface-elevated); border-radius: 8px; border: 1px solid var(--bp-border); } .mac-mini-container-name { color: var(--bp-text); font-size: 14px; font-family: monospace; } .mac-mini-container-status { padding: 4px 12px; border-radius: 12px; font-size: 12px; font-weight: 600; } .mac-mini-container-status.healthy { background: rgba(34, 197, 94, 0.2); color: #22c55e; } .mac-mini-container-status.running { background: rgba(59, 130, 246, 0.2); color: #3b82f6; } .mac-mini-container-status.stopped { background: rgba(239, 68, 68, 0.2); color: #ef4444; } /* Ollama Models */ .mac-mini-model-list { display: flex; flex-direction: column; gap: 12px; margin-bottom: 20px; } .mac-mini-model-item { display: flex; justify-content: space-between; align-items: center; padding: 16px; background: var(--bp-surface-elevated); border-radius: 8px; border: 1px solid var(--bp-border); } .mac-mini-model-info { display: flex; flex-direction: column; gap: 4px; } .mac-mini-model-name { color: var(--bp-text); font-size: 16px; font-weight: 600; } .mac-mini-model-details { color: var(--bp-text-muted); font-size: 13px; } .mac-mini-model-size { color: var(--bp-primary); font-size: 14px; font-weight: 600; } /* Download Progress */ .mac-mini-download { margin-top: 20px; padding-top: 20px; border-top: 1px solid var(--bp-border); } .mac-mini-download h4 { color: var(--bp-text); margin: 0 0 12px 0; } .mac-mini-download-input { display: flex; gap: 12px; margin-bottom: 16px; } .mac-mini-download-input input { flex: 1; padding: 12px 16px; border-radius: 8px; border: 1px solid var(--bp-border); background: var(--bp-surface-elevated); color: var(--bp-text); font-size: 14px; } .mac-mini-download-input input:focus { outline: none; border-color: var(--bp-primary); } .mac-mini-progress { display: none; } .mac-mini-progress.active { display: block; } .mac-mini-progress-header { display: flex; justify-content: space-between; margin-bottom: 8px; } .mac-mini-progress-name { color: var(--bp-text); font-weight: 600; } .mac-mini-progress-stats { color: var(--bp-text-muted); font-size: 13px; } .mac-mini-progress-bar-container { height: 24px; background: var(--bp-surface-elevated); border-radius: 12px; overflow: hidden; position: relative; } .mac-mini-progress-bar { height: 100%; background: linear-gradient(90deg, var(--bp-primary), #991b1b); border-radius: 12px; transition: width 0.3s ease; width: 0%; } .mac-mini-progress-text { position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); color: white; font-size: 12px; font-weight: 600; text-shadow: 0 1px 2px rgba(0,0,0,0.5); } .mac-mini-progress-log { margin-top: 16px; padding: 16px; background: #0a0a0a; border-radius: 8px; font-family: monospace; font-size: 12px; color: #22c55e; max-height: 200px; overflow-y: auto; white-space: pre-wrap; } /* Docker Controls */ .mac-mini-docker-controls { margin-top: 16px; display: flex; gap: 8px; } .mac-mini-docker-controls .btn { flex: 1; padding: 10px 16px; border-radius: 8px; font-weight: 500; cursor: pointer; background: var(--bp-surface-elevated); color: var(--bp-text); border: 1px solid var(--bp-border); transition: all 0.2s; } .mac-mini-docker-controls .btn:hover { background: var(--bp-surface); border-color: var(--bp-primary); } """ @staticmethod def get_html() -> str: """HTML fuer das Mac Mini Control Modul.""" return """
""" @staticmethod def get_js() -> str: """JavaScript fuer das Mac Mini Control Modul.""" return """ // ============================================= // MAC MINI CONTROL MODULE // ============================================= let macMiniModuleState = { ip: '192.168.178.100', isOnline: false, downloadInProgress: false, pollInterval: null }; function loadMacMiniModule() { console.log('Loading Mac Mini Control Module...'); macMiniRefresh(); startMacMiniPolling(); } function unloadMacMiniModule() { stopMacMiniPolling(); } function startMacMiniPolling() { stopMacMiniPolling(); macMiniModuleState.pollInterval = setInterval(macMiniRefresh, 30000); } function stopMacMiniPolling() { if (macMiniModuleState.pollInterval) { clearInterval(macMiniModuleState.pollInterval); macMiniModuleState.pollInterval = null; } } async function macMiniRefresh() { const statusBadge = document.getElementById('mac-mini-status-badge'); if (!statusBadge) return; statusBadge.className = 'mac-mini-status-badge checking'; statusBadge.textContent = 'Prüfe...'; try { const response = await fetch('/api/mac-mini/status'); const data = await response.json(); macMiniModuleState.isOnline = data.online; macMiniModuleState.ip = data.ip || macMiniModuleState.ip; // Update status badge if (data.online) { statusBadge.className = 'mac-mini-status-badge online'; statusBadge.textContent = 'Online'; } else { statusBadge.className = 'mac-mini-status-badge offline'; statusBadge.textContent = 'Offline'; } // Update IP setMacMiniValue('mac-mini-ip', macMiniModuleState.ip); // Update connection setMacMiniStatus('mac-mini-ssh', data.ssh ? 'Verbunden' : 'Nicht verfügbar', data.ssh); setMacMiniStatus('mac-mini-ping', data.ping ? 'OK' : 'Timeout', data.ping); // Update services setMacMiniStatus('mac-mini-backend', data.backend ? 'Läuft' : 'Offline', data.backend); setMacMiniStatus('mac-mini-ollama', data.ollama ? 'Läuft' : 'Offline', data.ollama); setMacMiniStatus('mac-mini-docker', data.docker ? 'Läuft' : 'Offline', data.docker); // Update system setMacMiniValue('mac-mini-uptime', data.uptime || '--'); setMacMiniValue('mac-mini-cpu', data.cpu_load || '--'); setMacMiniValue('mac-mini-memory', data.memory || '--'); // Update containers renderMacMiniContainers(data.containers || []); // Update models renderMacMiniModels(data.models || []); // Enable/disable buttons const btnWake = document.getElementById('btn-mac-mini-wake'); const btnRestart = document.getElementById('btn-mac-mini-restart'); const btnShutdown = document.getElementById('btn-mac-mini-shutdown'); if (btnWake) btnWake.disabled = data.online; if (btnRestart) btnRestart.disabled = !data.online; if (btnShutdown) btnShutdown.disabled = !data.online; } catch (error) { console.error('Mac Mini status error:', error); statusBadge.className = 'mac-mini-status-badge offline'; statusBadge.textContent = 'Fehler'; } } function setMacMiniValue(id, value) { const el = document.getElementById(id); if (el) el.textContent = value; } function setMacMiniStatus(id, text, isOk) { const el = document.getElementById(id); if (el) { el.textContent = text; el.className = 'mac-mini-card-value ' + (isOk ? 'ok' : 'error'); } } function renderMacMiniContainers(containers) { const list = document.getElementById('mac-mini-containers'); if (!list) return; if (containers.length === 0) { list.innerHTML = '