Before mounting, probe each share's remote path with `rclone lsf` (10s timeout, parallel execution). Failed shares are skipped — they never get mounted or exposed to SMB/NFS/WebDAV — preventing the silent hang that occurred when rclone mounted a nonexistent directory. - ShareHealth enum: Pending → Probing → Healthy / Failed(reason) - Supervisor: probe phase between preflight and mount, protocol configs generated after probe with only healthy shares - Web UI: health-aware badges (OK/FAILED/PROBING/PENDING) with error messages on dashboard, status partial, and share detail - JSON API: health + health_message fields on /api/status - CLI: `warpgate status` queries daemon API first for tri-state display (OK/FAILED/DOWN), falls back to direct mount checks Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
98 lines
5.2 KiB
HTML
98 lines
5.2 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); }
|
|
.badge-warn { background: rgba(251,191,36,0.15); color: var(--yellow); }
|
|
.error-msg { margin-top: 8px; padding: 8px 12px; background: rgba(248,113,113,0.08); border-radius: 4px; color: var(--red); font-size: 0.85em; }
|
|
.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 }} | 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.health == "OK" %}
|
|
<span class="badge badge-ok">OK</span>
|
|
{% elif share.health == "FAILED" %}
|
|
<span class="badge badge-error" title="{{ share.health_message }}">FAILED</span>
|
|
{% elif share.health == "PROBING" %}
|
|
<span class="badge badge-warn">PROBING</span>
|
|
{% else %}
|
|
<span class="badge badge-warn">PENDING</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>
|
|
{% if share.health == "FAILED" %}
|
|
<div class="error-msg">{{ share.health_message }}</div>
|
|
{% endif %}
|
|
</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>
|