warpgate/templates/web/dashboard.html
grabbit ba1cae7f75 Add daemon web UI, JSON API, and config hot-reload engine
- New: axum web server on port 8090 with htmx dashboard
- New: JSON API endpoints (/api/status, /api/config, /api/bwlimit)
- New: config diff engine with 4-tier change classification
- New: tiered config hot-reload (live/protocol/per-share/global)
- Refactor: supervisor loop uses mpsc command channel (recv_timeout)
- Refactor: supervisor updates shared DaemonStatus every poll cycle
- Dependencies: tokio, axum, askama, tower-http

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-18 14:18:20 +08:00

89 lines
4.6 KiB
HTML

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Warpgate Dashboard</title>
<script src="https://unpkg.com/htmx.org@2.0.4"></script>
<style>
:root {
--bg: #0f1117; --surface: #1a1d27; --border: #2a2d3a;
--text: #e1e4ed; --text-muted: #8b8fa3; --accent: #6c8aff;
--green: #4ade80; --red: #f87171; --yellow: #fbbf24;
--font: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, monospace;
}
* { box-sizing: border-box; margin: 0; padding: 0; }
body { background: var(--bg); color: var(--text); font-family: var(--font); padding: 24px; max-width: 960px; margin: 0 auto; }
h1 { font-size: 1.4em; margin-bottom: 4px; }
.header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 24px; border-bottom: 1px solid var(--border); padding-bottom: 16px; }
.header .status-dot { display: inline-block; width: 10px; height: 10px; border-radius: 50%; background: var(--green); margin-right: 8px; }
.meta { color: var(--text-muted); font-size: 0.85em; }
.cards { display: flex; flex-direction: column; gap: 12px; margin-bottom: 24px; }
.card { background: var(--surface); border: 1px solid var(--border); border-radius: 8px; padding: 16px; }
.card-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 8px; }
.card-header h2 { font-size: 1.1em; }
.card-header h2 a { color: var(--accent); text-decoration: none; }
.card-header h2 a:hover { text-decoration: underline; }
.badge { font-size: 0.75em; padding: 2px 8px; border-radius: 4px; font-weight: 600; }
.badge-ok { background: rgba(74,222,128,0.15); color: var(--green); }
.badge-error { background: rgba(248,113,113,0.15); color: var(--red); }
.badge-ro { background: rgba(251,191,36,0.15); color: var(--yellow); }
.stats { display: flex; gap: 24px; font-size: 0.9em; color: var(--text-muted); flex-wrap: wrap; }
.stats span { white-space: nowrap; }
.stats .label { color: var(--text-muted); }
.stats .value { color: var(--text); }
.protocols { display: flex; gap: 16px; margin-bottom: 20px; font-size: 0.9em; }
.proto-badge { padding: 4px 12px; border-radius: 4px; font-weight: 600; }
.proto-on { background: rgba(74,222,128,0.15); color: var(--green); }
.proto-off { background: rgba(248,113,113,0.1); color: var(--text-muted); }
.actions { display: flex; gap: 12px; }
.btn { display: inline-block; padding: 8px 16px; border-radius: 6px; text-decoration: none; font-size: 0.9em; font-weight: 500; border: 1px solid var(--border); color: var(--text); background: var(--surface); cursor: pointer; }
.btn:hover { border-color: var(--accent); color: var(--accent); }
</style>
</head>
<body>
<div class="header">
<div>
<h1><span class="status-dot"></span>Warpgate Dashboard</h1>
<div class="meta">Uptime: {{ uptime }} &nbsp;|&nbsp; Config: {{ config_path }}</div>
</div>
</div>
<div id="status-area" hx-get="/partials/status" hx-trigger="every 3s" hx-swap="innerHTML">
{% for share in shares %}
<div class="card">
<div class="card-header">
<h2><a href="/shares/{{ share.name }}">{{ share.name }}</a></h2>
<div>
{% if share.mounted %}
<span class="badge badge-ok">OK</span>
{% else %}
<span class="badge badge-error">DOWN</span>
{% endif %}
{% if share.read_only %}
<span class="badge badge-ro">RO</span>
{% endif %}
</div>
</div>
<div class="stats">
<span><span class="label">Mount:</span> <span class="value">{{ share.mount_point }}</span></span>
<span><span class="label">Cache:</span> <span class="value">{{ share.cache_display }}</span></span>
<span><span class="label">Dirty:</span> <span class="value">{{ share.dirty_count }}</span></span>
<span><span class="label">Speed:</span> <span class="value">{{ share.speed_display }}</span></span>
</div>
</div>
{% endfor %}
<div class="protocols">
<span class="proto-badge {% if smbd_running %}proto-on{% else %}proto-off{% endif %}">SMB: {% if smbd_running %}ON{% else %}OFF{% endif %}</span>
<span class="proto-badge {% if nfs_exported %}proto-on{% else %}proto-off{% endif %}">NFS: {% if nfs_exported %}ON{% else %}OFF{% endif %}</span>
<span class="proto-badge {% if webdav_running %}proto-on{% else %}proto-off{% endif %}">WebDAV: {% if webdav_running %}ON{% else %}OFF{% endif %}</span>
</div>
</div>
<div class="actions">
<a class="btn" href="/config">Edit Config</a>
</div>
</body>
</html>