warpgate/templates/web/tabs/shares.html
grabbit 74b0e72549 Add periodic dir-refresh and per-share refresh status display
Introduces a ScheduledTask mechanism that periodically calls rclone RC
vfs/refresh to keep directory listing caches warm (no file downloads),
with two-level config (global default + per-share override). Adds
dir-refresh status badges and timestamps to the web UI shares tab and
CLI status output, following the same pattern as warmup/warmed.

- src/scheduler.rs: New generic ScheduledTask runner with generation-based
  cancellation and parse_interval() helper
- src/rclone/rc.rs: Add vfs_refresh() RC API call
- src/config.rs: Add DirRefreshConfig, per-share dir_refresh_interval
  override, effective_dir_refresh_interval() resolution method
- src/config_diff.rs: Track dir_refresh_changed for hot-reload
- src/daemon.rs: Track per-share last_dir_refresh timestamps (HashMap),
  add dir_refresh_ago_for() helper and format_ago()
- src/supervisor.rs: spawn_dir_refresh() per-share background threads,
  called on startup and config reload
- src/web/api.rs: Expose dir_refresh_active + last_dir_refresh_ago in
  ShareStatusResponse
- src/web/pages.rs: Populate dir_refresh_active + last_dir_refresh_ago
  in ShareView and ShareDetailView
- templates/web/tabs/shares.html: DIR-REFRESH badge (yellow=pending,
  green=N ago) in health column; Dir Refresh row in detail panel
- templates/web/tabs/config.html: Dir Refresh section and per-share
  interval field in interactive config editor
- src/cli/status.rs: Append Dir-Refresh suffix to mount status lines

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-02-19 10:54:08 +08:00

118 lines
5.3 KiB
HTML

<div x-data="{ expanded: new URLSearchParams(location.search).get('expand') || '{{ expand }}' }">
<table class="share-table">
<thead>
<tr>
<th>Name</th>
<th>Health</th>
<th>Mount</th>
<th>Cache</th>
<th>Dirty</th>
<th>Speed</th>
<th>Transfers</th>
</tr>
</thead>
<tbody>
{% for share in shares %}
<tr class="share-row" @click="expanded = expanded === '{{ share.name }}' ? '' : '{{ share.name }}'">
<td><strong>{{ share.name }}</strong></td>
<td>
{% if share.health == "OK" %}
<span class="badge badge-ok">OK</span>
{% elif share.health == "FAILED" %}
<span class="badge badge-error">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 %}
{% if share.warmup_state == "running" %}
<span class="badge badge-warmup">WARMUP {{ share.warmup_done }}/{{ share.warmup_total }}</span>
{% elif share.warmup_state == "pending" %}
<span class="badge badge-warmup">WARMUP...</span>
{% elif share.warmup_state == "complete" %}
<span class="badge badge-ok">WARMED</span>
{% endif %}
{% if share.dir_refresh_active %}
{% if share.last_dir_refresh_ago.is_empty() %}
<span class="badge badge-warn">DIR-REFRESH...</span>
{% else %}
<span class="badge badge-ok">DIR-REFRESH {{ share.last_dir_refresh_ago }}</span>
{% endif %}
{% endif %}
</td>
<td class="mono">{{ share.mount_point }}</td>
<td>{{ share.cache_display }}</td>
<td>{{ share.dirty_count }}</td>
<td>{{ share.speed_display }}</td>
<td>{{ share.transfers }}</td>
</tr>
<tr x-show="expanded === '{{ share.name }}'" x-transition {% if share.name != expand %}x-cloak{% endif %} class="detail-row">
<td colspan="7">
<div class="detail-panel">
<div class="detail-grid">
<div class="detail-card">
<div class="label">Cache Used</div>
<div class="value">{{ share.cache_display }}</div>
</div>
<div class="detail-card">
<div class="label">Dirty Files</div>
<div class="value">{{ share.dirty_count }}</div>
</div>
<div class="detail-card">
<div class="label">Transfer Speed</div>
<div class="value">{{ share.speed_display }}</div>
</div>
<div class="detail-card">
<div class="label">Active Transfers</div>
<div class="value">{{ share.transfers }}</div>
</div>
</div>
<table class="info-table">
<tr><td>Health</td><td>{{ share.health }}</td></tr>
{% if share.health == "FAILED" %}
<tr><td>Probe Error</td><td class="error-text">{{ share.health_message }}</td></tr>
{% endif %}
<tr><td>Connection</td><td class="mono">{{ share.connection }}</td></tr>
<tr><td>Mount Point</td><td class="mono">{{ share.mount_point }}</td></tr>
<tr><td>Remote Path</td><td class="mono">{{ share.remote_path }}</td></tr>
<tr><td>RC Port</td><td>{{ share.rc_port }}</td></tr>
<tr><td>Errored Files</td><td>{{ share.errored_files }}</td></tr>
<tr><td>Total Errors</td><td>{{ share.errors }}</td></tr>
<tr><td>Mounted</td><td>{% if share.mounted %}Yes{% else %}No{% endif %}</td></tr>
<tr><td>Read-Only</td><td>{% if share.read_only %}Yes{% else %}No{% endif %}</td></tr>
{% if share.dir_refresh_active %}
<tr><td>Dir Refresh</td><td>{% if share.last_dir_refresh_ago.is_empty() %}pending{% else %}{{ share.last_dir_refresh_ago }}{% endif %}</td></tr>
{% endif %}
</table>
{% if !share.warmup_rules.is_empty() %}
<h4 style="margin-top:1rem;margin-bottom:0.5rem;font-size:0.95em">Warmup Rules</h4>
<table class="info-table">
<thead><tr>
<td style="font-weight:600;color:var(--text-muted)">Path</td>
<td style="font-weight:600;color:var(--text-muted)">Filter</td>
<td style="font-weight:600;color:var(--text-muted)">State</td>
<td style="font-weight:600;color:var(--text-muted)">Progress</td>
</tr></thead>
<tbody>
{% for rule in share.warmup_rules %}
<tr>
<td class="mono">{{ rule.path }}</td>
<td>{% if rule.newer_than.is_empty() %}-{% else %}{{ rule.newer_than }}{% endif %}</td>
<td><span class="badge badge-{{ rule.badge_class }}">{{ rule.state }}</span></td>
<td>{{ rule.cached + rule.skipped }}/{{ rule.total_files }}</td>
</tr>
{% endfor %}
</tbody>
</table>
{% endif %}
</div>
</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>