feat(observer): M-Obs 4-7 — widgets, dashboard, docs, integration example
Some checks failed
Test / test (push) Has been cancelled
Some checks failed
Test / test (push) Has been cancelled
M-Obs 4: @shade/widgets React library - ShadeProvider context with observer URL + token + theme - useShadeState (polling) + useShadeEvents (SSE) hooks - 7 widgets: IdentityCard, SessionList, PrekeyStock, RecentActivity, ServerStatus, FingerprintCompare, WidgetCatalog (meta-widget for user-selectable layout with localStorage persistence) - Self-contained CSS via inline styles, no external CSS conflicts - Light/dark/auto theme via tokens M-Obs 5: @shade/dashboard standalone SPA - Vite + React app composing all widgets into a full debugger layout - Login screen with token persistence to localStorage - Build script copies dist/ to @shade/observer/dist/ for embedded serving - 211 KB JS bundle (66 KB gzipped) M-Obs 6: Documentation + integration example - READMEs for @shade/observer and @shade/widgets - examples/06-observer-dashboard runnable demo: spins up prekey server + observer, runs Alice ↔ Bob conversation loop, dashboard at :3901 - Updated root README and docker-compose.yml with observer integration M-Obs 7: End-to-end verification - StateAggregator now drains buffered events on subscription, so identity.initialized fires before observer construction are still seen - Verified live: snapshot endpoint returns full state, dashboard serves, 401 without auth, sessions/messages/ratchet steps all tracked 220 tests passing, 0 failures. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
149
packages/shade-dashboard/src/App.tsx
Normal file
149
packages/shade-dashboard/src/App.tsx
Normal file
@@ -0,0 +1,149 @@
|
||||
import React, { useState, useEffect } from 'react';
|
||||
import {
|
||||
ShadeProvider,
|
||||
IdentityCard,
|
||||
SessionList,
|
||||
PrekeyStock,
|
||||
RecentActivity,
|
||||
ServerStatus,
|
||||
FingerprintCompare,
|
||||
} from '@shade/widgets';
|
||||
import { Login } from './Login.js';
|
||||
|
||||
const STORAGE_KEY = 'shade-dashboard-config';
|
||||
|
||||
interface DashboardConfig {
|
||||
observerUrl: string;
|
||||
token: string;
|
||||
}
|
||||
|
||||
export function App(): React.ReactElement {
|
||||
const [config, setConfig] = useState<DashboardConfig | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
// Determine the default observer URL: assume dashboard is served from
|
||||
// the same origin as the observer, so the API is at the parent path
|
||||
const defaultUrl = window.location.origin + window.location.pathname.replace(/\/dashboard\/?$/, '');
|
||||
|
||||
try {
|
||||
const stored = window.localStorage.getItem(STORAGE_KEY);
|
||||
if (stored) {
|
||||
const parsed = JSON.parse(stored) as DashboardConfig;
|
||||
if (parsed.observerUrl && parsed.token) {
|
||||
setConfig(parsed);
|
||||
return;
|
||||
}
|
||||
}
|
||||
} catch {}
|
||||
|
||||
// Show login screen
|
||||
setConfig({ observerUrl: defaultUrl, token: '' });
|
||||
}, []);
|
||||
|
||||
function handleLogin(observerUrl: string, token: string): void {
|
||||
const cfg = { observerUrl, token };
|
||||
window.localStorage.setItem(STORAGE_KEY, JSON.stringify(cfg));
|
||||
setConfig(cfg);
|
||||
}
|
||||
|
||||
function handleLogout(): void {
|
||||
window.localStorage.removeItem(STORAGE_KEY);
|
||||
setConfig({ observerUrl: window.location.origin, token: '' });
|
||||
}
|
||||
|
||||
if (!config) {
|
||||
return <div style={{ padding: 40, textAlign: 'center' }}>Loading…</div>;
|
||||
}
|
||||
|
||||
if (!config.token) {
|
||||
return <Login defaultUrl={config.observerUrl} onLogin={handleLogin} />;
|
||||
}
|
||||
|
||||
return (
|
||||
<ShadeProvider observerUrl={config.observerUrl} token={config.token}>
|
||||
<DashboardLayout onLogout={handleLogout} />
|
||||
</ShadeProvider>
|
||||
);
|
||||
}
|
||||
|
||||
function DashboardLayout({ onLogout }: { onLogout: () => void }): React.ReactElement {
|
||||
return (
|
||||
<div
|
||||
style={{
|
||||
maxWidth: 1400,
|
||||
margin: '0 auto',
|
||||
padding: 24,
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
gap: 24,
|
||||
}}
|
||||
>
|
||||
<header
|
||||
style={{
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
justifyContent: 'space-between',
|
||||
paddingBottom: 16,
|
||||
borderBottom: '1px solid #262626',
|
||||
}}
|
||||
>
|
||||
<div>
|
||||
<h1
|
||||
style={{
|
||||
margin: 0,
|
||||
fontSize: 22,
|
||||
fontWeight: 600,
|
||||
color: '#f7c948',
|
||||
letterSpacing: '-0.01em',
|
||||
}}
|
||||
>
|
||||
Shade Observer
|
||||
</h1>
|
||||
<div style={{ fontSize: 12, color: '#a3a3a3', marginTop: 2 }}>
|
||||
Live debugger for your Signal Protocol deployment
|
||||
</div>
|
||||
</div>
|
||||
<button
|
||||
onClick={onLogout}
|
||||
style={{
|
||||
background: 'transparent',
|
||||
border: '1px solid #262626',
|
||||
color: '#a3a3a3',
|
||||
padding: '6px 14px',
|
||||
borderRadius: 6,
|
||||
cursor: 'pointer',
|
||||
fontSize: 12,
|
||||
fontFamily: 'inherit',
|
||||
}}
|
||||
>
|
||||
Sign out
|
||||
</button>
|
||||
</header>
|
||||
|
||||
<IdentityCard />
|
||||
|
||||
<div
|
||||
style={{
|
||||
display: 'grid',
|
||||
gridTemplateColumns: 'minmax(280px, 1fr) minmax(280px, 1fr) minmax(280px, 1fr)',
|
||||
gap: 20,
|
||||
}}
|
||||
>
|
||||
<PrekeyStock />
|
||||
<ServerStatus />
|
||||
<FingerprintCompare />
|
||||
</div>
|
||||
|
||||
<div
|
||||
style={{
|
||||
display: 'grid',
|
||||
gridTemplateColumns: '1fr 1fr',
|
||||
gap: 20,
|
||||
}}
|
||||
>
|
||||
<SessionList />
|
||||
<RecentActivity limit={100} />
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user