HuggingClaw / health-server.js
somratpro's picture
feat: enable iframe embedding by injecting a preload script to strip security headers and update CSP policies
40855ff
Raw
History Blame
10.6 kB
// Lightweight health server and dashboard on port 7861
const http = require("http");
const fs = require("fs");
const PORT = process.env.HEALTH_PORT || 7861;
const startTime = Date.now();
const LLM_MODEL = process.env.LLM_MODEL || "Not Set";
const GATEWAY_TOKEN = process.env.GATEWAY_TOKEN || "huggingclaw";
const TELEGRAM_ENABLED = !!process.env.TELEGRAM_BOT_TOKEN;
const server = http.createServer((req, res) => {
const uptime = Math.floor((Date.now() - startTime) / 1000);
const uptimeHuman = `${Math.floor(uptime / 3600)}h ${Math.floor((uptime % 3600) / 60)}m`;
if (req.url === "/health") {
res.writeHead(200, { "Content-Type": "application/json" });
res.end(
JSON.stringify({
status: "ok",
uptime: uptime,
uptimeHuman: uptimeHuman,
timestamp: new Date().toISOString(),
}),
);
return;
}
if (req.url === "/status") {
let syncStatus = { status: "unknown", message: "No sync data yet" };
try {
if (fs.existsSync("/tmp/sync-status.json")) {
syncStatus = JSON.parse(
fs.readFileSync("/tmp/sync-status.json", "utf8"),
);
}
} catch (e) {}
res.writeHead(200, { "Content-Type": "application/json" });
res.end(
JSON.stringify({
model: LLM_MODEL,
whatsapp: true,
telegram: TELEGRAM_ENABLED,
sync: syncStatus,
uptime: uptimeHuman,
token: GATEWAY_TOKEN,
}),
);
return;
}
if (req.url === "/") {
res.writeHead(200, { "Content-Type": "text/html" });
res.end(`
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>HuggingClaw Dashboard</title>
<link href="https://fonts.googleapis.com/css2?family=Outfit:wght@300;400;600&display=swap" rel="stylesheet">
<style>
:root {
--bg: #0f172a;
--card-bg: rgba(30, 41, 59, 0.7);
--accent: linear-gradient(135deg, #3b82f6, #8b5cf6);
--text: #f8fafc;
--text-dim: #94a3b8;
--success: #10b981;
--error: #ef4444;
--warning: #f59e0b;
}
* { box-sizing: border-box; margin: 0; padding: 0; }
body {
font-family: 'Outfit', sans-serif;
background-color: var(--bg);
color: var(--text);
display: flex;
justify-content: center;
align-items: center;
min-height: 100vh;
overflow: hidden;
background-image:
radial-gradient(at 0% 0%, rgba(59, 130, 246, 0.15) 0px, transparent 50%),
radial-gradient(at 100% 0%, rgba(139, 92, 246, 0.15) 0px, transparent 50%);
}
.dashboard {
width: 90%;
max-width: 600px;
background: var(--card-bg);
backdrop-filter: blur(12px);
border: 1px solid rgba(255, 255, 255, 0.1);
border-radius: 24px;
padding: 40px;
box-shadow: 0 25px 50px -12px rgba(0, 0, 0, 0.5);
animation: fadeIn 0.8s ease-out;
}
@keyframes fadeIn {
from { opacity: 0; transform: translateY(20px); }
to { opacity: 1; transform: translateY(0); }
}
header {
text-align: center;
margin-bottom: 40px;
}
h1 {
font-size: 2.5rem;
margin-bottom: 8px;
background: var(--accent);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
font-weight: 600;
}
.subtitle {
color: var(--text-dim);
font-size: 0.9rem;
letter-spacing: 1px;
text-transform: uppercase;
}
.stats-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 20px;
margin-bottom: 30px;
}
.stat-card {
background: rgba(255, 255, 255, 0.03);
border: 1px solid rgba(255, 255, 255, 0.05);
padding: 20px;
border-radius: 16px;
transition: transform 0.3s ease, border-color 0.3s ease;
}
.stat-card:hover {
transform: translateY(-5px);
border-color: rgba(59, 130, 246, 0.3);
}
.stat-label {
color: var(--text-dim);
font-size: 0.75rem;
text-transform: uppercase;
margin-bottom: 8px;
display: block;
}
.stat-value {
font-size: 1.1rem;
font-weight: 600;
word-break: break-all;
}
.stat-btn {
grid-column: span 2;
background: var(--accent);
color: #fff;
padding: 16px;
border-radius: 16px;
text-align: center;
text-decoration: none;
font-weight: 600;
margin-top: 10px;
transition: transform 0.3s ease, box-shadow 0.3s ease;
box-shadow: 0 10px 20px -5px rgba(59, 130, 246, 0.4);
}
.stat-btn:hover {
transform: scale(1.02);
box-shadow: 0 15px 30px -5px rgba(59, 130, 246, 0.6);
}
.status-badge {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 4px 12px;
border-radius: 20px;
font-size: 0.8rem;
font-weight: 600;
}
.status-online { background: rgba(16, 185, 129, 0.1); color: var(--success); }
.status-offline { background: rgba(239, 68, 68, 0.1); color: var(--error); }
.status-syncing { background: rgba(59, 130, 246, 0.1); color: #3b82f6; }
.pulse {
width: 8px;
height: 8px;
border-radius: 50%;
background: currentColor;
box-shadow: 0 0 0 0 rgba(16, 185, 129, 0.7);
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(16, 185, 129, 0.7); }
70% { transform: scale(1); box-shadow: 0 0 0 10px rgba(16, 185, 129, 0); }
100% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(16, 185, 129, 0); }
}
.footer {
text-align: center;
color: var(--text-dim);
font-size: 0.8rem;
margin-top: 20px;
}
.sync-info {
background: rgba(255, 255, 255, 0.02);
padding: 15px;
border-radius: 12px;
font-size: 0.85rem;
color: var(--text-dim);
margin-top: 10px;
}
#sync-msg { color: var(--text); display: block; margin-top: 4px; }
</style>
</head>
<body>
<div class="dashboard">
<header>
<h1>🦞 HuggingClaw</h1>
<p class="subtitle">Space Control Panel</p>
</header>
<div class="stats-grid">
<div class="stat-card">
<span class="stat-label">Model</span>
<span class="stat-value" id="model-id">Loading...</span>
</div>
<div class="stat-card">
<span class="stat-label">Uptime</span>
<span class="stat-value" id="uptime">Loading...</span>
</div>
<div class="stat-card">
<span class="stat-label">WhatsApp</span>
<span id="wa-status">Loading...</span>
</div>
<div class="stat-card">
<span class="stat-label">Telegram</span>
<span id="tg-status">Loading...</span>
</div>
<a href="/" class="stat-btn">πŸš€ Open Control UI</a>
</div>
<div class="stat-card" style="width: 100%;">
<span class="stat-label">Workspace Sync Status</span>
<div id="sync-badge-container"></div>
<div class="sync-info">
Last Sync Activity: <span id="sync-time">Never</span>
<span id="sync-msg">Initializing synchronization...</span>
</div>
</div>
<div class="footer">
&bull; Live updates every 10s &bull;
</div>
</div>
<script>
async function updateStats() {
try {
const res = await fetch('/status');
const data = await res.json();
document.getElementById('model-id').textContent = data.model;
document.getElementById('uptime').textContent = data.uptime;
document.getElementById('wa-status').innerHTML = data.whatsapp
? '<div class="status-badge status-online"><div class="pulse"></div>Active</div>'
: '<div class="status-badge status-offline">Disabled</div>';
document.getElementById('tg-status').innerHTML = data.telegram
? '<div class="status-badge status-online"><div class="pulse"></div>Active</div>'
: '<div class="status-badge status-offline">Disabled</div>';
const syncData = data.sync;
let badgeClass = 'status-offline';
let pulseHtml = '';
if (syncData.status === 'success') {
badgeClass = 'status-online';
pulseHtml = '<div class="pulse"></div>';
} else if (syncData.status === 'syncing') {
badgeClass = 'status-syncing';
pulseHtml = '<div class="pulse" style="background:#3b82f6"></div>';
} else if (syncData.status === 'error') {
badgeClass = 'status-offline';
}
document.getElementById('sync-badge-container').innerHTML =
'<div class="status-badge ' + badgeClass + '">' + pulseHtml + syncData.status.toUpperCase() + '</div>';
document.getElementById('sync-time').textContent = syncData.timestamp || 'Never';
document.getElementById('sync-msg').textContent = syncData.message || 'Waiting for first sync...';
} catch (e) {
console.error("Failed to fetch status", e);
}
}
updateStats();
setInterval(updateStats, 10000);
</script>
</body>
</html>
`);
return;
}
res.writeHead(404);
res.end();
});
server.listen(PORT, "0.0.0.0", () => {
console.log(`πŸ₯ Health server & Dashboard listening on port ${PORT}`);
});