Files
PiCloud/dashboard/src/routes/apps/[slug]/dead-letters/+page.svelte
MechaCat02 1795dfc98a feat(v1.1.1-dead-letters): dashboard badge + list view
Design notes §4 makes the dashboard surface load-bearing — with no
default DL handler, users wouldn't know dead letters exist
otherwise.

New route: `apps/[slug]/dead-letters/+page.svelte` — list view
columns per the design notes:
- `created_at`, `source`, `op`, `script_id`, `attempt_count`,
  `first/last_attempt_at`, `last_error` (truncated; clickable)
- per-row Replay + Mark resolved buttons
- expandable row detail panel showing full payload (JSON) +
  full last_error
- unresolved-only filter (default on); refresh button

Per-app detail page (`apps/[slug]/+page.svelte`) grows a "Dead
letters" link in the tabs nav, with a red unresolved-count pill
when > 0. Loaded in parallel with the existing app loaders so it
doesn't slow the page.

Apps list (`apps/+page.svelte`) shows the same red pill next to
each app's name when its unresolved count > 0. Counts fetched in
parallel after the apps list lands; failures here are non-fatal
(just no badge).

API client wiring: `api.deadLetters.{count,list,get,replay,resolve}`
mirrors the v1.1.1 admin endpoints. `DeadLetterRow` type added to
the dashboard's API shape declarations.

dashboard's svelte-check passes (369 files, 0 errors, 0 warnings).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-01 22:21:20 +02:00

311 lines
6.6 KiB
Svelte

<script lang="ts">
import { base } from '$app/paths';
import { page } from '$app/state';
import { api, ApiError, type App, type DeadLetterRow } from '$lib/api';
let slug = $derived(page.params.slug ?? '');
let app = $state<App | null>(null);
let rows = $state<DeadLetterRow[]>([]);
let unresolved = $state<number>(0);
let loading = $state(true);
let error = $state<string | null>(null);
let unresolvedOnly = $state(true);
let expandedId = $state<string | null>(null);
async function load() {
loading = true;
error = null;
try {
const a = await api.apps.get(slug);
app = a;
const c = await api.deadLetters.count(slug);
unresolved = c.unresolved;
const r = await api.deadLetters.list(slug, { unresolved: unresolvedOnly, limit: 100 });
rows = r.dead_letters;
} catch (e) {
error = e instanceof ApiError ? e.message : String(e);
} finally {
loading = false;
}
}
$effect(() => {
// Re-load whenever the slug or filter changes.
void slug;
void unresolvedOnly;
void load();
});
async function replay(dlId: string) {
try {
await api.deadLetters.replay(slug, dlId);
await load();
} catch (e) {
error = e instanceof ApiError ? e.message : String(e);
}
}
async function markIgnored(dlId: string) {
try {
await api.deadLetters.resolve(slug, dlId, 'ignored');
await load();
} catch (e) {
error = e instanceof ApiError ? e.message : String(e);
}
}
function toggleExpanded(id: string) {
expandedId = expandedId === id ? null : id;
}
function fmtTime(iso: string): string {
return new Date(iso).toLocaleString();
}
function truncate(s: string, n: number): string {
if (s.length <= n) return s;
return s.slice(0, n) + '…';
}
</script>
<svelte:head>
<title>Dead letters · {slug} · PiCloud</title>
</svelte:head>
<div class="container">
<header>
<div>
<a href="{base}/apps/{slug}" class="back">&larr; back to {app?.name ?? slug}</a>
<h1>Dead letters</h1>
<p class="subtitle">
{#if unresolved > 0}
<strong class="badge">{unresolved}</strong> unresolved
{:else}
No unresolved dead letters
{/if}
</p>
</div>
<div class="controls">
<label>
<input type="checkbox" bind:checked={unresolvedOnly} />
Show unresolved only
</label>
<button onclick={load} disabled={loading}>Refresh</button>
</div>
</header>
{#if error}
<div class="error">{error}</div>
{/if}
{#if loading}
<p>Loading…</p>
{:else if rows.length === 0}
<p class="empty">
{#if unresolvedOnly}
No unresolved dead letters for this app. 🎉
{:else}
No dead letters recorded yet.
{/if}
</p>
{:else}
<table>
<thead>
<tr>
<th>Created</th>
<th>Source</th>
<th>Op</th>
<th>Script</th>
<th>Attempts</th>
<th>First / Last attempt</th>
<th>Last error</th>
<th>Actions</th>
</tr>
</thead>
<tbody>
{#each rows as row (row.id)}
<tr class:resolved={row.resolved_at !== null}>
<td>{fmtTime(row.created_at)}</td>
<td><code>{row.source}</code></td>
<td><code>{row.op}</code></td>
<td>{row.script_id ? row.script_id.slice(0, 8) : '—'}</td>
<td>{row.attempt_count}</td>
<td class="times">
<div>{fmtTime(row.first_attempt_at)}</div>
<div>{fmtTime(row.last_attempt_at)}</div>
</td>
<td class="err">
<button class="link" onclick={() => toggleExpanded(row.id)}>
{truncate(row.last_error, 60)}
</button>
</td>
<td class="actions">
{#if row.resolved_at === null}
<button onclick={() => replay(row.id)}>Replay</button>
<button class="secondary" onclick={() => markIgnored(row.id)}>
Mark resolved
</button>
{:else}
<span class="resolution">{row.resolution ?? 'resolved'}</span>
{/if}
</td>
</tr>
{#if expandedId === row.id}
<tr class="detail">
<td colspan="8">
<div class="detail-grid">
<section>
<h3>Payload</h3>
<pre>{JSON.stringify(row.payload, null, 2)}</pre>
</section>
<section>
<h3>Last error</h3>
<pre>{row.last_error}</pre>
</section>
</div>
</td>
</tr>
{/if}
{/each}
</tbody>
</table>
{/if}
</div>
<style>
.container {
max-width: 1200px;
margin: 0 auto;
padding: 2rem;
}
header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 1rem;
gap: 1rem;
}
.back {
font-size: 0.85rem;
color: var(--text-muted, #666);
text-decoration: none;
}
.back:hover {
text-decoration: underline;
}
h1 {
margin: 0.25rem 0;
}
.subtitle {
color: var(--text-muted, #666);
margin: 0;
}
.badge {
display: inline-block;
min-width: 1.5rem;
padding: 0.1rem 0.4rem;
background: #c00;
color: #fff;
border-radius: 999px;
text-align: center;
font-weight: 600;
}
.controls {
display: flex;
gap: 0.75rem;
align-items: center;
}
.error {
background: #fee;
border: 1px solid #fbb;
color: #900;
padding: 0.75rem 1rem;
border-radius: 4px;
margin-bottom: 1rem;
}
.empty {
color: var(--text-muted, #666);
text-align: center;
padding: 2rem;
}
table {
width: 100%;
border-collapse: collapse;
font-size: 0.9rem;
}
th,
td {
text-align: left;
padding: 0.5rem 0.75rem;
border-bottom: 1px solid var(--border, #e0e0e0);
vertical-align: top;
}
th {
background: var(--bg-secondary, #f5f5f5);
font-weight: 600;
}
tr.resolved {
opacity: 0.6;
}
.times div {
font-size: 0.8rem;
white-space: nowrap;
}
.err button.link {
background: none;
border: none;
color: var(--link, #06c);
text-decoration: underline;
cursor: pointer;
padding: 0;
font-family: monospace;
font-size: 0.85rem;
text-align: left;
}
.actions {
white-space: nowrap;
display: flex;
gap: 0.4rem;
}
.actions button.secondary {
background: transparent;
color: var(--text, #333);
border: 1px solid var(--border, #ccc);
}
.resolution {
font-style: italic;
color: var(--text-muted, #666);
font-size: 0.85rem;
}
tr.detail td {
background: var(--bg-secondary, #fafafa);
padding: 0;
}
.detail-grid {
display: grid;
grid-template-columns: 2fr 1fr;
gap: 1rem;
padding: 1rem;
}
.detail-grid section h3 {
margin: 0 0 0.5rem 0;
font-size: 0.85rem;
text-transform: uppercase;
color: var(--text-muted, #666);
}
.detail-grid pre {
background: #fff;
border: 1px solid var(--border, #e0e0e0);
padding: 0.75rem;
border-radius: 4px;
font-size: 0.8rem;
overflow: auto;
max-height: 300px;
margin: 0;
}
code {
font-family: monospace;
font-size: 0.85rem;
}
</style>