import { rowHTML, setRows, safe, showToast, badge } from './helpers.js'; import { api } from './api.js'; // --- State --- const liveSockets = new Map(); let slowTimer = null; let fastTimer = null; const activeSids = new Set(); // --- Small helpers (no optional chaining) --- function qSel(root, sel){ return root ? root.querySelector(sel) : null; } function text(el){ return (el && el.textContent) ? el.textContent : ''; } function val(el){ return el ? el.value : undefined; } function low(x){ return String(x||'').toLowerCase(); } function injectOnceCSS() { if (document.getElementById('vmadmin-live-css')) return; const style = document.createElement('style'); style.id = 'vmadmin-live-css'; style.textContent = ` .pulse-dot{display:inline-block;width:8px;height:8px;border-radius:50%;background:#22c55e;margin-left:.5rem;opacity:.25;animation:pulse-vm 1s ease-in-out 4} @keyframes pulse-vm {0%,100%{opacity:.25;transform:scale(1)}50%{opacity:1;transform:scale(1.25)}} `; document.head.appendChild(style); } function flashDot(cell) { if (!cell) return; const dot = document.createElement('span'); dot.className = 'pulse-dot'; cell.appendChild(dot); setTimeout(() => { if (dot && dot.parentNode) dot.parentNode.removeChild(dot); }, 1500); } function setBadgeCell(cell, textOrState) { if (!cell) return false; let html = ''; const s = low(textOrState); if (/running|online|started/.test(s)) html = badge('running','ok'); else if (/stopp|shutdown|offline/.test(s)) html = badge('stopped','dark'); else if (/working|progress|busy/.test(s)) html = badge('working','info'); else html = badge(textOrState || '—','dark'); if (cell.innerHTML !== html) { cell.innerHTML = html; return true; } return false; } function rebuildTargetSelect(selectEl, currentNode, nodes) { if (!selectEl) return []; const current = String(currentNode || '').trim(); const others = (nodes || []) .map(n => String(n && (n.name || n.node || n)).trim()) .filter(Boolean) .filter(n => n !== current); selectEl.innerHTML = others.map(n => ``).join(''); if (selectEl.options.length > 0) selectEl.selectedIndex = 0; return others; } function updateMigrateButton(tr, isRunning) { const btn = qSel(tr, '.act-migrate'); const targetSel = qSel(tr, '.target-node'); const hasTarget = !!(targetSel && targetSel.options && targetSel.options.length > 0); const enable = !!(isRunning && hasTarget); if (!btn) return; if (enable) { btn.removeAttribute('disabled'); btn.classList.remove('disabled'); } else { btn.setAttribute('disabled',''); btn.classList.add('disabled'); } } function updateActionButtons(tr, isRunning) { const bStart = qSel(tr, '.act-start'); const bStop = qSel(tr, '.act-stop'); const bShutdown = qSel(tr, '.act-shutdown'); if (bStart) { if (isRunning) { bStart.setAttribute('disabled',''); bStart.classList.add('disabled'); } else { bStart.removeAttribute('disabled'); bStart.classList.remove('disabled'); } } if (bStop) { if (isRunning) { bStop.removeAttribute('disabled'); bStop.classList.remove('disabled'); } else { bStop.setAttribute('disabled',''); bStop.classList.add('disabled'); } } if (bShutdown) { if (isRunning) { bShutdown.removeAttribute('disabled'); bShutdown.classList.remove('disabled'); } else { bShutdown.setAttribute('disabled',''); bShutdown.classList.add('disabled'); } } } export function stopAllAdminWatches() { liveSockets.forEach(function(ws){ try { ws.close(); } catch(e){} }); liveSockets.clear(); if (slowTimer) clearInterval(slowTimer); if (fastTimer) clearInterval(fastTimer); slowTimer = null; fastTimer = null; activeSids.clear(); } function ensureWatchOn() { const tbody = document.querySelector('#vm-admin tbody'); if (!tbody) return; const rows = Array.from(tbody.querySelectorAll('tr[data-sid]')); rows.forEach(function(tr){ const sid = tr.getAttribute('data-sid'); if (!sid || liveSockets.has(sid)) return; try { const wsProto = (location.protocol === 'https:') ? 'wss' : 'ws'; const ws = new WebSocket(wsProto + '://' + location.host + '/ws/observe?sid=' + encodeURIComponent(sid)); liveSockets.set(sid, ws); ws.onopen = function(){}; ws.onclose = function(){ liveSockets.delete(sid); }; ws.onerror = function(){}; ws.onmessage = function(ev){ try { const msg = JSON.parse(ev.data || '{}'); if (!msg || !msg.type) return; const tr = tbody.querySelector('tr[data-sid="' + sid + '"]'); if (!tr) return; const statusCell = tr.children[4]; const nameCell = tr.children[2]; const nodeCell = tr.children[3]; const targetSel = qSel(tr, '.target-node'); if (msg.type === 'status') { const stRaw = low(msg.status); const changed = setBadgeCell(statusCell, stRaw); const isRunning = /running|online|started/.test(stRaw); updateMigrateButton(tr, isRunning); updateActionButtons(tr, isRunning); if (changed) flashDot(nameCell); if (stRaw && /running|stopped|shutdown/.test(stRaw)) { setTimeout(function(){ activeSids.delete(sid); }, 3000); } } if (msg.type === 'node' && msg.node) { const newNode = String(msg.node).trim(); if (nodeCell && newNode && text(nodeCell).trim() !== newNode) { nodeCell.textContent = newNode; rebuildTargetSelect(targetSel, newNode, window.__nodesCache || []); flashDot(nameCell); } } } catch(e){} }; } catch(e){} }); } function dedupe(arr){ return Array.from(new Set(arr)); } function extractNodeNames(ns){ if (!ns) return []; if (Array.isArray(ns.nodes)) return dedupe(ns.nodes.map(n => String(n.name || n.node || n).trim()).filter(Boolean)); if (Array.isArray(ns)) return dedupe(ns.map(n => String(n.name || n.node || n).trim()).filter(Boolean)); return []; } export async function startAdminWatches() { injectOnceCSS(); const tbody = document.querySelector('#vm-admin tbody'); if (!tbody) return; const ns = await api.nodesSummary(); const availableNodes = extractNodeNames(ns); setRows(tbody, []); try { const latest = await api.listAllVmct(); const all = Array.isArray(latest.all) ? latest.all : []; const nodesNow = Array.isArray(latest.nodes) ? latest.nodes : []; window.__nodesCache = nodesNow.slice(); const rows = all.map(vm => ({ sid: safe(vm.sid), name: safe(vm.name || vm.vmid || vm.sid), node: safe(vm.node || '—'), status: safe(vm.status || '—') })); const htmlRows = rows.map(r => rowHTML([ ``, safe(r.sid), safe(r.name), safe(r.node), badge(safe(r.status), /running|online|started/i.test(r.status) ? 'ok' : 'dark'), `
`, ``, `` ])); setRows(tbody, htmlRows); // wire per-row Array.from(tbody.querySelectorAll('tr[data-sid]')).forEach(function(tr){ const nodeCell = tr.children[3]; const targetSel = qSel(tr, '.target-node'); const currentNode = text(nodeCell).trim(); rebuildTargetSelect(targetSel, currentNode, availableNodes); updateMigrateButton(tr, /running|online|started/i.test(tr.children[4].innerText)); const sid = tr.getAttribute('data-sid'); const nameCell = tr.children[2]; async function doAction(kind, needsTarget) { try { const targetNode = needsTarget ? val(targetSel) : undefined; activeSids.add(sid); setBadgeCell(tr.children[4], 'working'); updateMigrateButton(tr, false); const res = await api.vmAction(sid, kind, targetNode); if (res && res.ok) { showToast(`Task ${kind} started for ${safe(text(nameCell))}`); } else { showToast(`Task ${kind} failed for ${safe(text(nameCell))}`, 'danger'); } } catch (e) { showToast(`Error: ${e && e.message ? e.message : e}`, 'danger'); } } const bStart = qSel(tr, '.act-start'); if (bStart) bStart.addEventListener('click', function(){ doAction('start'); }); const bStop = qSel(tr, '.act-stop'); if (bStop) bStop.addEventListener('click', function(){ doAction('stop'); }); const bShut = qSel(tr, '.act-shutdown'); if (bShut) bShut.addEventListener('click', function(){ doAction('shutdown'); }); const bUnl = qSel(tr, '.act-unlock'); if (bUnl) bUnl.addEventListener('click', function(){ doAction('unlock'); }); const bMig = qSel(tr, '.act-migrate'); if (bMig) bMig.addEventListener('click', function(){ doAction('migrate', true); }); ensureWatchOn(); }); window.__nodesCache = availableNodes.slice(); // slow: full refresh every 30s slowTimer = setInterval(async function(){ try { const latest = await api.listAllVmct(); const all = Array.isArray(latest.all) ? latest.all : []; const bySid = new Map(all.map(x => [String(x.sid), x])); const nodesNow = Array.isArray(latest.nodes) ? latest.nodes : (window.__nodesCache || []); window.__nodesCache = nodesNow; Array.from(tbody.querySelectorAll('tr[data-sid]')).forEach(function(tr){ const sid = tr.getAttribute('data-sid'); const rowData = bySid.get(sid); if (!rowData) return; const nodeCell = tr.children[3]; const statusCell= tr.children[4]; const nameCell = tr.children[2]; const targetSel = qSel(tr, '.target-node'); const newNode = String(rowData.node || '').trim(); if (nodeCell && newNode && text(nodeCell).trim() !== newNode) { nodeCell.textContent = newNode; rebuildTargetSelect(targetSel, newNode, nodesNow); flashDot(nameCell); } // status from slow reconcile — only when not 'working' to avoid overruling WS const currentTxt = low((statusCell && statusCell.innerText) || ''); if (!/working/.test(currentTxt)) { const stRaw = low(rowData.status || ''); if (stRaw) { const changed = setBadgeCell(statusCell, stRaw); const isRunning = /running|online|started/.test(stRaw); updateMigrateButton(tr, isRunning); updateActionButtons(tr, isRunning); if (changed) flashDot(nameCell); } } }); } catch(e){} }, 30000); // fast: active sids every 10s fastTimer = setInterval(async function(){ try { const sids = Array.from(activeSids); if (!sids.length) return; for (let i=0;i