Compare commits

...

10 Commits

5 changed files with 287 additions and 40 deletions

115
README.md
View File

@@ -1,45 +1,100 @@
# OpenClaw Monitor # 🐾 OC Monitor — OpenClaw Mission Control
Multi-node monitoring dashboard for OpenClaw instances. Real-time monitoring dashboard for [OpenClaw](https://github.com/openclaw/openclaw) multi-node deployments.
## Architecture ![Dashboard](https://img.shields.io/badge/status-active-brightgreen) ![License](https://img.shields.io/badge/license-MIT-blue)
- **Server**: Node.js + SQLite + WebSocket (central dashboard) ## ✨ Features
- **Agent**: Bash script on each OpenClaw machine, reports heartbeat + metrics
## Quick Start - **Real-time metrics** — CPU / Memory / Disk / Swap with live jitter animation (10s refresh)
- **Provider health checks** — Auto-detect all configured AI providers, latency monitoring
- **Default model detection** — Auto-identifies most-used provider per node (green dot indicator)
- **Request logging** — Track API calls across all nodes with filtering by node / provider / result
- **Multi-node support** — Lightweight bash+python agent, works on macOS & Linux
- **Dark / Light theme** — Toggle with localStorage persistence
- **WebSocket push** — Instant updates, no polling
- **Admin panel** — Node management, token display, one-click agent install command generator
- **Docker deployment** — Single container, SQLite storage
## 🚀 Quick Start
### 1. Deploy Server (Docker)
### Server (Docker)
```bash ```bash
docker run -d --name oc-monitor -p 3800:3800 -v oc-monitor-data:/app/data ghcr.io/mango082888-bit/oc-monitor curl -fsSL https://cdn.jsdelivr.net/gh/xmg0828888/oc-monitor/install.sh | bash
# Get auth token
docker logs oc-monitor 2>&1 | grep "Auth token"
``` ```
### Agent This will:
- Pull the repo and build the Docker image
- Generate a random auth token
- Start the container on port **3800**
- Print the dashboard URL and token
### 2. Install Agent on Each Node
After server is running, install the agent on each OpenClaw node:
```bash ```bash
curl -sL https://raw.githubusercontent.com/mango082888-bit/oc-monitor/main/agent/agent.sh -o agent.sh curl -fsSL https://cdn.jsdelivr.net/gh/xmg0828888/oc-monitor/install-agent.sh | bash -s -- \
chmod +x agent.sh -s http://YOUR_SERVER_IP:3800 \
./agent.sh -s http://YOUR_SERVER:3800 -t YOUR_TOKEN -n "My Node" -r master -t YOUR_AUTH_TOKEN \
-n "Node Name"
``` ```
## Features The agent auto-detects:
- OpenClaw config, providers, and models
- System metrics (CPU, memory, disk, swap)
- Gateway & daemon status
- Session count and token usage
- Default/most-used provider
- Real-time node status (CPU/mem/disk/swap) ### 3. Open Dashboard
- Provider health matrix across all nodes
- API request logging with TTFT/latency tracking
- Auto-detect OpenClaw config changes
- WebSocket live updates
- Dark theme UI
## API Visit `http://YOUR_SERVER_IP:3800` in your browser.
| Endpoint | Method | Auth | Description | ## 📐 Architecture
|---|---|---|---|
| `/api/dashboard` | GET | No | Full dashboard data |
| `/api/heartbeat` | POST | Yes | Agent heartbeat report |
| `/api/request` | POST | Yes | Log API request |
| `/api/node/rename` | POST | Yes | Rename a node |
| `/api/node/:id` | DELETE | Yes | Remove a node |
Auth: `Authorization: Bearer <token>` ```
┌─────────────┐ ┌─────────────┐ ┌─────────────┐
│ Node Agent │ │ Node Agent │ │ Node Agent │
│ (bash+py) │ │ (bash+py) │ │ (bash+py) │
└──────┬───────┘ └──────┬───────┘ └──────┬───────┘
│ HTTP POST │ │
└────────────────────┼─────────────────────┘
┌─────────────────┐
│ OC Monitor │
│ Server │
│ (Node.js+SQLite│
│ +WebSocket) │
└────────┬────────┘
│ WS push
┌─────────────────┐
│ Browser │
│ Dashboard │
└─────────────────┘
```
## 🔧 Configuration
| Env Variable | Default | Description |
|---|---|---|
| `PORT` | `3800` | Server listen port |
| `AUTH_TOKEN` | (required) | Bearer token for API auth |
## 📋 API Endpoints
| Method | Path | Description |
|---|---|---|
| GET | `/api/dashboard` | Full dashboard data |
| POST | `/api/heartbeat` | Agent heartbeat report |
| POST | `/api/request` | API request log (single or batch) |
| POST | `/api/rename` | Rename a node |
| DELETE | `/api/node/:id` | Remove a node |
All POST/DELETE endpoints require `Authorization: Bearer <token>` header.
## License
MIT

View File

@@ -1,7 +1,7 @@
#!/bin/bash #!/bin/bash
# OpenClaw Monitor Agent # OpenClaw Monitor Agent
set -e set -e
SERVER="" TOKEN="" NAME="" ROLE="worker" INTERVAL=30 DEFAULT_PROV="" SERVER="" TOKEN="" NAME="" ROLE="worker" INTERVAL=10 DEFAULT_PROV=""
while getopts "s:t:n:r:i:d:" opt; do while getopts "s:t:n:r:i:d:" opt; do
case $opt in s)SERVER="$OPTARG";;t)TOKEN="$OPTARG";;n)NAME="$OPTARG";;r)ROLE="$OPTARG";;i)INTERVAL="$OPTARG";;d)DEFAULT_PROV="$OPTARG";;esac case $opt in s)SERVER="$OPTARG";;t)TOKEN="$OPTARG";;n)NAME="$OPTARG";;r)ROLE="$OPTARG";;i)INTERVAL="$OPTARG";;d)DEFAULT_PROV="$OPTARG";;esac
@@ -18,7 +18,10 @@ OC_VERSION=$(openclaw --version 2>/dev/null | head -1 || echo "unknown")
echo "OC Monitor Agent: node=$NODE_ID name=$NAME server=$SERVER" echo "OC Monitor Agent: node=$NODE_ID name=$NAME server=$SERVER"
TICK=0
while true; do while true; do
if [ $((TICK % 6)) -eq 0 ]; then
# Full run: collect everything
python3 - "$OC_CONFIG" "$NODE_ID" "$NAME" "$OC_VERSION" "$ROLE" "$DEFAULT_PROV" << 'PYEOF' > /tmp/.oc-agent-payload.json python3 - "$OC_CONFIG" "$NODE_ID" "$NAME" "$OC_VERSION" "$ROLE" "$DEFAULT_PROV" << 'PYEOF' > /tmp/.oc-agent-payload.json
import json,subprocess,os,platform,time,sys,glob import json,subprocess,os,platform,time,sys,glob
@@ -28,6 +31,7 @@ def run(cmd):
cfg,nid,name,ver,role = sys.argv[1],sys.argv[2],sys.argv[3],sys.argv[4],sys.argv[5] cfg,nid,name,ver,role = sys.argv[1],sys.argv[2],sys.argv[3],sys.argv[4],sys.argv[5]
default_prov = sys.argv[6] if len(sys.argv)>6 else '' default_prov = sys.argv[6] if len(sys.argv)>6 else ''
full_run = sys.argv[7]=='1' if len(sys.argv)>7 else True
mac = sys.platform=='darwin' mac = sys.platform=='darwin'
# Host IP - try multiple interfaces on macOS # Host IP - try multiple interfaces on macOS
@@ -215,6 +219,44 @@ print(json.dumps({
'tok_today':tok_today,'tok_week':tok_week,'tok_month':tok_month 'tok_today':tok_today,'tok_week':tok_week,'tok_month':tok_month
})) }))
PYEOF PYEOF
else
# Light run: only update metrics in cached payload
python3 -c "
import json,subprocess,os,platform
def run(cmd):
try: return subprocess.check_output(cmd,shell=True,stderr=subprocess.DEVNULL).decode().strip()
except: return ''
mac=platform.system()=='Darwin'
if mac:
raw=run('ps -A -o %cpu | tail -n +2')
try: cpu=round(sum(float(x) for x in raw.split() if x)/os.cpu_count(),1)
except: cpu=0
else:
cpu=float(run(\"top -bn1 | grep 'Cpu' | awk '{print 100-\$8}'\") or 0)
ds=run('df -h / | tail -1').split()
disk=int(ds[4].replace('%','')) if len(ds)>4 else 0
if mac:
try:
vm=run('vm_stat');d={}
for l in vm.split(chr(10))[1:]:
if ':' in l: k,v=l.split(':',1);d[k.strip()]=int(v.strip().rstrip('.'))
ps=16384;used=(d.get('Pages active',0)+d.get('Pages wired down',0))*ps;total=int(run('sysctl -n hw.memsize'))
mem=round(used/total*100,1)
except: mem=0
si=run('sysctl vm.swapusage');swap=0
if si:
import re;u=re.search(r'used\s*=\s*([\d.]+)M',si);t=re.search(r'total\s*=\s*([\d.]+)M',si)
if u and t and float(t.group(1))>0: swap=round(float(u.group(1))/float(t.group(1))*100,1)
else:
mi=run('free -b | grep Mem').split();mem=round(int(mi[2])/int(mi[1])*100,1) if len(mi)>2 else 0
si=run('free -b | grep Swap').split();swap=round(int(si[2])/int(si[1])*100,1) if len(si)>2 and int(si[1])>0 else 0
try:
p=json.load(open('/tmp/.oc-agent-payload.json'))
p['cpu']=cpu;p['mem']=mem;p['disk']=disk;p['swap']=swap
json.dump(p,open('/tmp/.oc-agent-payload.json','w'))
except: pass
" 2>/dev/null
fi
if [ -s /tmp/.oc-agent-payload.json ]; then if [ -s /tmp/.oc-agent-payload.json ]; then
curl -sS -X POST "$SERVER/api/heartbeat" \ curl -sS -X POST "$SERVER/api/heartbeat" \
@@ -225,7 +267,8 @@ PYEOF
|| echo "[$(date +%H:%M:%S)] heartbeat failed" || echo "[$(date +%H:%M:%S)] heartbeat failed"
fi fi
# Report new API requests from session jsonl files # Report new API requests only on full runs
if [ $((TICK % 6)) -eq 0 ]; then
python3 - "$OC_CONFIG" "$NODE_ID" "$SERVER" "$TOKEN" << 'REQEOF' 2>/dev/null python3 - "$OC_CONFIG" "$NODE_ID" "$SERVER" "$TOKEN" << 'REQEOF' 2>/dev/null
import json,os,sys,glob,time,urllib.request,urllib.error import json,os,sys,glob,time,urllib.request,urllib.error
cfg,nid,server,token = sys.argv[1],sys.argv[2],sys.argv[3],sys.argv[4] cfg,nid,server,token = sys.argv[1],sys.argv[2],sys.argv[3],sys.argv[4]
@@ -280,6 +323,8 @@ if reqs:
json.dump(state,open(state_f,'w')) json.dump(state,open(state_f,'w'))
if reqs: print(f'[{time.strftime("%H:%M:%S")}] reported {len(reqs)} requests') if reqs: print(f'[{time.strftime("%H:%M:%S")}] reported {len(reqs)} requests')
REQEOF REQEOF
fi
TICK=$((TICK + 1))
sleep "$INTERVAL" sleep "$INTERVAL"
done done

80
install-agent.sh Executable file
View File

@@ -0,0 +1,80 @@
#!/bin/bash
# OC Monitor - One-click agent install
set -e
SERVER="" TOKEN="" NAME="" ROLE="worker"
while getopts "s:t:n:r:" opt; do
case $opt in s)SERVER="$OPTARG";;t)TOKEN="$OPTARG";;n)NAME="$OPTARG";;r)ROLE="$OPTARG";;esac
done
if [ -z "$SERVER" ] || [ -z "$TOKEN" ]; then
echo "Usage: $0 -s SERVER_URL -t AUTH_TOKEN [-n NAME] [-r ROLE]"
echo " -s Server URL (e.g. http://1.2.3.4:3800)"
echo " -t Auth token"
echo " -n Node name (default: hostname)"
echo " -r Role: master|worker (default: worker)"
exit 1
fi
[ -z "$NAME" ] && NAME=$(hostname)
AGENT="/usr/local/bin/oc-monitor-agent.sh"
echo "🐾 OC Monitor Agent Installer"
echo "=============================="
# Check deps
for cmd in python3 curl; do
command -v $cmd &>/dev/null || { echo "$cmd not found"; exit 1; }
done
# Download agent
echo "📦 Downloading agent..."
curl -fsSL https://cdn.jsdelivr.net/gh/xmg0828888/oc-monitor/agent/agent.sh -o "$AGENT"
chmod +x "$AGENT"
# Detect init system
if [ "$(uname)" = "Darwin" ]; then
PLIST="$HOME/Library/LaunchAgents/com.oc-monitor.agent.plist"
echo "🍎 Setting up launchd service..."
cat > "$PLIST" <<EOF
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0"><dict>
<key>Label</key><string>com.oc-monitor.agent</string>
<key>ProgramArguments</key><array>
<string>/bin/bash</string><string>$AGENT</string>
<string>-s</string><string>$SERVER</string>
<string>-t</string><string>$TOKEN</string>
<string>-n</string><string>$NAME</string>
<string>-r</string><string>$ROLE</string>
</array>
<key>RunAtLoad</key><true/>
<key>KeepAlive</key><true/>
<key>StandardOutPath</key><string>/tmp/oc-monitor-agent.log</string>
<key>StandardErrorPath</key><string>/tmp/oc-monitor-agent.log</string>
</dict></plist>
EOF
launchctl unload "$PLIST" 2>/dev/null || true
launchctl load "$PLIST"
echo "✅ Agent running (launchd)"
else
echo "🐧 Setting up systemd service..."
cat > /etc/systemd/system/oc-monitor-agent.service <<EOF
[Unit]
Description=OC Monitor Agent
After=network.target
[Service]
ExecStart=/bin/bash $AGENT -s $SERVER -t $TOKEN -n $NAME -r $ROLE
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target
EOF
systemctl daemon-reload
systemctl enable --now oc-monitor-agent
echo "✅ Agent running (systemd)"
fi
echo ""
echo "📡 Node '$NAME' reporting to $SERVER"
echo "📋 Logs: journalctl -u oc-monitor-agent -f"

49
install.sh Executable file
View File

@@ -0,0 +1,49 @@
#!/bin/bash
# OC Monitor - One-click server install
set -e
PORT=${PORT:-3800}
TOKEN=${TOKEN:-$(head -c 16 /dev/urandom | xxd -p | tr '[:lower:]' '[:upper:]')}
DIR="/opt/oc-monitor"
echo "🐾 OC Monitor Installer"
echo "========================"
# Check docker
if ! command -v docker &>/dev/null; then
echo "❌ Docker not found. Install docker first."
exit 1
fi
# Clone or update
if [ -d "$DIR" ]; then
echo "📦 Updating existing installation..."
cd "$DIR" && git pull
else
echo "📦 Cloning repository..."
git clone https://github.com/xmg0828888/oc-monitor.git "$DIR"
cd "$DIR"
fi
# Build and run
echo "🔨 Building Docker image..."
docker build -t oc-monitor .
docker rm -f oc-monitor 2>/dev/null || true
echo "🚀 Starting container..."
docker run -d --name oc-monitor --restart always \
-p "$PORT:3800" \
-v oc-monitor-data:/app/data \
-e "AUTH_TOKEN=$TOKEN" \
oc-monitor
IP=$(hostname -I 2>/dev/null | awk '{print $1}' || curl -s ifconfig.me)
echo ""
echo "✅ OC Monitor is running!"
echo "========================"
echo "🌐 Dashboard: http://$IP:$PORT"
echo "🔑 Auth Token: $TOKEN"
echo ""
echo "📡 Install agent on each node:"
echo " curl -fsSL https://cdn.jsdelivr.net/gh/xmg0828888/oc-monitor/install-agent.sh | bash -s -- -s http://$IP:$PORT -t $TOKEN -n \"NodeName\""

View File

@@ -7,6 +7,7 @@
<style> <style>
*{margin:0;padding:0;box-sizing:border-box} *{margin:0;padding:0;box-sizing:border-box}
:root{--bg:#060a10;--card:#0d1520;--card2:#111c2e;--border:#1a2740;--txt:#c8d6e5;--dim:#5a6f88;--neon:#00e5ff;--green:#00ff88;--warn:#ffb020;--err:#ff4466;--purple:#b8a9ff;--peach:#ffb088} :root{--bg:#060a10;--card:#0d1520;--card2:#111c2e;--border:#1a2740;--txt:#c8d6e5;--dim:#5a6f88;--neon:#00e5ff;--green:#00ff88;--warn:#ffb020;--err:#ff4466;--purple:#b8a9ff;--peach:#ffb088}
[data-theme="light"]{--bg:#f0f2f5;--card:#fff;--card2:#f5f7fa;--border:#d8dde6;--txt:#1a1a2e;--dim:#6b7b8d;--neon:#0088cc;--green:#00a856;--warn:#e69500;--err:#d63050;--purple:#7c5cbf;--peach:#d07040}
body{font-family:'SF Mono',Menlo,'Courier New',monospace;background:var(--bg);color:var(--txt);padding:16px} body{font-family:'SF Mono',Menlo,'Courier New',monospace;background:var(--bg);color:var(--txt);padding:16px}
.wrap{max-width:1200px;margin:0 auto} .wrap{max-width:1200px;margin:0 auto}
h1{font-size:1.3em;color:var(--neon);margin-bottom:4px} h1{font-size:1.3em;color:var(--neon);margin-bottom:4px}
@@ -41,7 +42,7 @@ h1{font-size:1.3em;color:var(--neon);margin-bottom:4px}
.ok{color:var(--green)}.er{color:var(--err)} .ok{color:var(--green)}.er{color:var(--err)}
.gs{display:grid;grid-template-columns:repeat(4,1fr);gap:8px;margin:8px 0} .gs{display:grid;grid-template-columns:repeat(4,1fr);gap:8px;margin:8px 0}
.g{font-size:.7em}.g-l{color:var(--dim)}.g-t{height:4px;background:var(--card2);border-radius:2px;margin:2px 0} .g{font-size:.7em}.g-l{color:var(--dim)}.g-t{height:4px;background:var(--card2);border-radius:2px;margin:2px 0}
.g-f{height:100%;border-radius:2px}.fg{background:var(--green)}.fb{background:var(--neon)}.fw{background:var(--warn)}.fr{background:var(--err)} .g-f{height:100%;border-radius:2px;transition:width .8s ease,background .5s}.fg{background:var(--green)}.fb{background:var(--neon)}.fw{background:var(--warn)}.fr{background:var(--err)}
.g-n{font-weight:600} .g-n{font-weight:600}
.tks{display:flex;gap:10px;margin:8px 0} .tks{display:flex;gap:10px;margin:8px 0}
.tk{flex:1;text-align:center;background:var(--card2);border-radius:6px;padding:5px} .tk{flex:1;text-align:center;background:var(--card2);border-radius:6px;padding:5px}
@@ -61,7 +62,7 @@ td{padding:6px 10px;border-bottom:1px solid rgba(26,39,64,.5)}
</head> </head>
<body> <body>
<div class="wrap"> <div class="wrap">
<h1>🐾 OpenClaw Mission Control <a href="/admin.html" style="font-size:.5em;color:var(--dim);text-decoration:none;margin-left:8px">⚙️ Admin</a></h1> <h1>🐾 OpenClaw Mission Control <a href="/admin.html" style="font-size:.5em;color:var(--dim);text-decoration:none;margin-left:8px">⚙️ Admin</a> <span id="themeBtn" onclick="toggleTheme()" style="font-size:.5em;cursor:pointer;margin-left:8px">🌙</span></h1>
<div class="sub" id="subtitle">Loading...</div> <div class="sub" id="subtitle">Loading...</div>
<div class="stats" id="stats"></div> <div class="stats" id="stats"></div>
@@ -122,7 +123,7 @@ function renderNodes(){
<div class="tg"><span class="${n.role==='master'?'ms':'wk'}">${n.role}</span><span>OC ${n.oc_version}</span><span>${n.os}</span></div> <div class="tg"><span class="${n.role==='master'?'ms':'wk'}">${n.role}</span><span>OC ${n.oc_version}</span><span>${n.os}</span></div>
<div class="hb">${hbBars}</div> <div class="hb">${hbBars}</div>
<div class="sec">供应商</div> <div class="sec">供应商</div>
${provs.sort((a,b)=>b.default-a.default||(a.name>b.name?1:-1)).map(p=>`<div class="pv"><div class="pv-l">${p.default?'<span class="dot on" style="width:6px;height:6px"></span>':''}<span${p.default?' class="pv-default"':''}>${p.name}</span> <span class="pm">${p.model}</span></div><div>${p.status==='ok'?'<span class="ok">✓</span> <span class="pm">'+p.ms+'ms</span>':'<span class="er">✗</span> <span class="pm">'+(p.err||'离线')+'</span>'}</div></div>`).join('')} ${provs.sort((a,b)=>b.default-a.default||(a.name>b.name?1:-1)).map(p=>`<div class="pv"><div class="pv-l">${p.default?'<span class="dot on" style="width:6px;height:6px"></span>':''}<span${p.default?' class="pv-default"':''}>${p.name}</span><div>${p.model.split(' | ').map(m=>'<span class="pm">'+m+'</span>').join('<br>')}</div></div><div>${p.status==='ok'?'<span class="ok">✓</span> <span class="pm">'+p.ms+'ms</span>':'<span class="er">✗</span> <span class="pm">'+(p.err||'离线')+'</span>'}</div></div>`).join('')}
<div class="gs"> <div class="gs">
${[['cpu',n.cpu],['mem',n.mem],['disk',n.disk],['swap',n.swap]].map(([l,v])=>`<div class="g" data-g="${l}"><span class="g-l">${l}</span><div class="g-t"><div class="g-f ${gaugeColor(v)}" style="width:${v}%"></div></div><span class="g-n">${on?v+'%':'—'}</span></div>`).join('')} ${[['cpu',n.cpu],['mem',n.mem],['disk',n.disk],['swap',n.swap]].map(([l,v])=>`<div class="g" data-g="${l}"><span class="g-l">${l}</span><div class="g-t"><div class="g-f ${gaugeColor(v)}" style="width:${v}%"></div></div><span class="g-n">${on?v+'%':'—'}</span></div>`).join('')}
</div> </div>
@@ -178,6 +179,18 @@ function renderLogs(){
$('#logTable').innerHTML=h+'</tbody>'; $('#logTable').innerHTML=h+'</tbody>';
} }
function animateNum(el,to,suffix=''){
el._real=to;
el.textContent=to+suffix;
}
// Live jitter: every 1s, all gauge numbers wobble around real value
setInterval(()=>{
document.querySelectorAll('.g-n').forEach(el=>{
if(el._real==null||el.textContent==='—')return;
const r=el._real,jit=r*(0.02*Math.random()-0.01);
el.textContent=Math.max(0,r+jit).toFixed(1)+'%';
});
},1000);
function updateNodeCard(n){ function updateNodeCard(n){
const el=document.querySelector(`.nd[data-id="${n.id}"]`); const el=document.querySelector(`.nd[data-id="${n.id}"]`);
if(!el)return false; if(!el)return false;
@@ -186,13 +199,11 @@ function updateNodeCard(n){
// gauges // gauges
[['cpu',n.cpu],['mem',n.mem],['disk',n.disk],['swap',n.swap]].forEach(([k,v])=>{ [['cpu',n.cpu],['mem',n.mem],['disk',n.disk],['swap',n.swap]].forEach(([k,v])=>{
const g=el.querySelector(`[data-g="${k}"]`); const g=el.querySelector(`[data-g="${k}"]`);
if(g){g.querySelector('.g-f').style.width=v+'%';g.querySelector('.g-f').className='g-f '+gaugeColor(v);g.querySelector('.g-n').textContent=on?v+'%':'—';} if(g){g.querySelector('.g-f').style.width=v+'%';g.querySelector('.g-f').className='g-f '+gaugeColor(v);const gn=g.querySelector('.g-n');if(on)animateNum(gn,v,'%');else gn.textContent='—';}
}); });
// tokens // tokens
const tv=el.querySelectorAll('.tk-v'); const tv=el.querySelectorAll('.tk-v');
if(tv[0])tv[0].textContent=fmtTok(n.tok_today); [[tv[0],n.tok_today],[tv[1],n.tok_week],[tv[2],n.tok_month]].forEach(([e,v])=>{if(e){const s=fmtTok(v);if(e.textContent!==s)e.textContent=s;}});
if(tv[1])tv[1].textContent=fmtTok(n.tok_week);
if(tv[2])tv[2].textContent=fmtTok(n.tok_month);
// footer // footer
const fs=el.querySelectorAll('.nd-f span'); const fs=el.querySelectorAll('.nd-f span');
if(fs[0])fs[0].textContent='⏱ '+fmtAge(Date.now()/1000-n.uptime); if(fs[0])fs[0].textContent='⏱ '+fmtAge(Date.now()/1000-n.uptime);
@@ -227,6 +238,13 @@ function connectWS(){
ws.onclose=function(){setTimeout(connectWS,3000)}; ws.onclose=function(){setTimeout(connectWS,3000)};
} }
function toggleTheme(){
const d=document.documentElement,light=d.getAttribute('data-theme')==='light';
d.setAttribute('data-theme',light?'':'light');
$('#themeBtn').textContent=light?'🌙':'☀️';
localStorage.setItem('theme',light?'dark':'light');
}
(function(){const t=localStorage.getItem('theme');if(t==='light'){document.documentElement.setAttribute('data-theme','light');document.getElementById('themeBtn').textContent='☀️';}})();
load();connectWS();setInterval(load,60000); load();connectWS();setInterval(load,60000);
</script> </script>
</body></html> </body></html>