50ecb6463a
getHealth обогащён: вердикт здоровья (ok/warning/critical) по порогам (память %, диск, ошибки/24ч, лаг event-loop, размер БД) + причины; реальный % памяти, лаг event-loop (perf_hooks), load average, свободное место на диске (statfs), PID/NODE_ENV, версия+git-commit, число активных SSE-соединений, размер WAL, разбивка БД по крупнейшим таблицам. sse.js: экспорт stats() (онлайн-пользователи/гости/соединения). admin.js loadHealth: светофор-баннер вердикта с причинами, тумблер авто-обновления (live, поллинг 5с с самоостановкой при уходе с вкладки), 8 карточек (uptime/БД/файлы/ошибки/SSE/память/event-loop/диск), панели платформы и активности, горизонтальные бары крупнейших таблиц БД. Проверено: getHealth собирает полный payload, вердикт срабатывает (диск<2ГБ → warning), NaN-лаг защищён. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
96 lines
3.1 KiB
JavaScript
96 lines
3.1 KiB
JavaScript
/* ── SSE registry — shared between controllers ─────────────────────────── */
|
|
const clients = new Map(); // userId -> Set<res>
|
|
const guestClients = new Map(); // sessionId -> Set<res>
|
|
const db = require('./db/db');
|
|
|
|
function addClient(userId, res) {
|
|
if (!clients.has(userId)) clients.set(userId, new Set());
|
|
clients.get(userId).add(res);
|
|
}
|
|
|
|
function removeClient(userId, res) {
|
|
const set = clients.get(userId);
|
|
if (!set) return;
|
|
set.delete(res);
|
|
if (set.size === 0) clients.delete(userId);
|
|
}
|
|
|
|
function emit(userId, data) {
|
|
const conns = clients.get(userId);
|
|
if (!conns?.size) return;
|
|
const payload = `data: ${JSON.stringify(data)}\n\n`;
|
|
for (const res of conns) {
|
|
try { res.write(payload); } catch {}
|
|
}
|
|
}
|
|
|
|
// Heartbeat: detect and remove dead connections every 30s
|
|
setInterval(() => {
|
|
for (const [userId, conns] of clients) {
|
|
for (const res of conns) {
|
|
try {
|
|
if (res.writableEnded || res.destroyed) { conns.delete(res); continue; }
|
|
res.write(': heartbeat\n\n');
|
|
} catch { conns.delete(res); }
|
|
}
|
|
if (conns.size === 0) clients.delete(userId);
|
|
}
|
|
for (const [sessionId, conns] of guestClients) {
|
|
for (const res of conns) {
|
|
try {
|
|
if (res.writableEnded || res.destroyed) { conns.delete(res); continue; }
|
|
res.write(': heartbeat\n\n');
|
|
} catch { conns.delete(res); }
|
|
}
|
|
if (conns.size === 0) guestClients.delete(sessionId);
|
|
}
|
|
}, 30_000).unref();
|
|
|
|
/* Broadcast to all members of a class */
|
|
function emitToClass(classId, data) {
|
|
const members = db.prepare('SELECT user_id FROM class_members WHERE class_id=?').all(classId);
|
|
for (const { user_id } of members) emit(user_id, data);
|
|
}
|
|
|
|
/* ── Guest SSE (session-scoped, no userId) ── */
|
|
function addGuestClient(sessionId, res) {
|
|
if (!guestClients.has(sessionId)) guestClients.set(sessionId, new Set());
|
|
guestClients.get(sessionId).add(res);
|
|
}
|
|
|
|
function removeGuestClient(sessionId, res) {
|
|
const set = guestClients.get(sessionId);
|
|
if (!set) return;
|
|
set.delete(res);
|
|
if (set.size === 0) guestClients.delete(sessionId);
|
|
}
|
|
|
|
function emitToGuests(sessionId, data) {
|
|
const set = guestClients.get(sessionId);
|
|
if (!set?.size) return;
|
|
const payload = `data: ${JSON.stringify(data)}\n\n`;
|
|
for (const res of set) {
|
|
try { res.write(payload); } catch {}
|
|
}
|
|
}
|
|
|
|
/* Returns array of user IDs currently connected via SSE */
|
|
function getOnlineUserIds() {
|
|
return [...clients.keys()];
|
|
}
|
|
|
|
/* Сводка SSE-соединений для мониторинга: онлайн-пользователи, гости и
|
|
суммарное число открытых стримов. */
|
|
function stats() {
|
|
let conns = 0;
|
|
for (const set of clients.values()) conns += set.size;
|
|
let guestConns = 0;
|
|
for (const set of guestClients.values()) guestConns += set.size;
|
|
return { users: clients.size, guests: guestClients.size, connections: conns + guestConns };
|
|
}
|
|
|
|
module.exports = {
|
|
addClient, removeClient, emit, emitToClass, getOnlineUserIds, stats,
|
|
addGuestClient, removeGuestClient, emitToGuests,
|
|
};
|