feat(pitch-deck): admin UI for investor + financial-model management
Some checks failed
CI / Deploy (pull_request) Has been skipped
CI / go-lint (pull_request) Failing after 2s
CI / python-lint (pull_request) Failing after 11s
CI / nodejs-lint (pull_request) Failing after 2s
CI / test-go-consent (pull_request) Failing after 2s
CI / test-python-voice (pull_request) Failing after 9s
CI / test-bqas (pull_request) Failing after 8s
Some checks failed
CI / Deploy (pull_request) Has been skipped
CI / go-lint (pull_request) Failing after 2s
CI / python-lint (pull_request) Failing after 11s
CI / nodejs-lint (pull_request) Failing after 2s
CI / test-go-consent (pull_request) Failing after 2s
CI / test-python-voice (pull_request) Failing after 9s
CI / test-bqas (pull_request) Failing after 8s
Adds /pitch-admin dashboard with real admin accounts (bcrypt) and full
audit attribution for every state-changing action.
Backend:
- pitch_admins + pitch_admin_sessions tables (migration 002)
- pitch_audit_logs.admin_id + target_investor_id columns
- lib/admin-auth.ts: bcryptjs hashing, single-session enforcement,
jose JWT with 'pitch-admin' audience claim, requireAdmin guard
- logAudit extended to accept admin_id and target_investor_id
- middleware.ts: gates /pitch-admin/* and /api/admin/* on the admin
cookie (with bearer-secret fallback for CLI compatibility)
- 14 API routes under /api/admin-auth and /api/admin (login, logout,
me, dashboard, investors[id] CRUD + resend, admins CRUD,
fm scenarios + assumptions PATCH)
- Existing /api/admin/{invite,investors,revoke,audit-logs} migrated
to requireAdmin and now log with admin_id + target_investor_id
- scripts/create-admin.ts CLI bootstrap (npm run admin:create)
Frontend:
- /pitch-admin/login + /pitch-admin/(authed) route group
- AdminShell with sidebar nav + StatCard + AuditLogTable components
- Dashboard with KPIs, recent logins, recent activity
- Investors list with search/filter + resend/revoke inline actions
- Investor detail with inline edit + per-investor audit timeline
- Audit log viewer with actor/action/date filters + pagination
- Financial model scenario list + per-scenario assumption editor
(categorized, inline edit, before/after diff in audit)
- Admins management (add, deactivate, reset password)
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
153
pitch-deck/components/pitch-admin/AuditLogTable.tsx
Normal file
153
pitch-deck/components/pitch-admin/AuditLogTable.tsx
Normal file
@@ -0,0 +1,153 @@
|
||||
'use client'
|
||||
|
||||
export interface AuditLogRow {
|
||||
id: number | string
|
||||
action: string
|
||||
created_at: string
|
||||
details: Record<string, unknown> | null
|
||||
ip_address?: string | null
|
||||
slide_id?: string | null
|
||||
investor_email?: string | null
|
||||
investor_name?: string | null
|
||||
target_investor_email?: string | null
|
||||
target_investor_name?: string | null
|
||||
admin_email?: string | null
|
||||
admin_name?: string | null
|
||||
}
|
||||
|
||||
interface AuditLogTableProps {
|
||||
rows: AuditLogRow[]
|
||||
showActor?: boolean
|
||||
}
|
||||
|
||||
const ACTION_COLORS: Record<string, string> = {
|
||||
login_success: 'text-green-400 bg-green-500/10',
|
||||
login_failed: 'text-rose-400 bg-rose-500/10',
|
||||
admin_login_success: 'text-green-400 bg-green-500/10',
|
||||
admin_login_failed: 'text-rose-400 bg-rose-500/10',
|
||||
admin_logout: 'text-white/40 bg-white/[0.04]',
|
||||
logout: 'text-white/40 bg-white/[0.04]',
|
||||
slide_viewed: 'text-indigo-400 bg-indigo-500/10',
|
||||
assumption_changed: 'text-amber-400 bg-amber-500/10',
|
||||
assumption_edited: 'text-amber-400 bg-amber-500/10',
|
||||
scenario_edited: 'text-amber-400 bg-amber-500/10',
|
||||
investor_invited: 'text-purple-400 bg-purple-500/10',
|
||||
magic_link_resent: 'text-purple-400 bg-purple-500/10',
|
||||
investor_revoked: 'text-rose-400 bg-rose-500/10',
|
||||
investor_edited: 'text-blue-400 bg-blue-500/10',
|
||||
admin_created: 'text-green-400 bg-green-500/10',
|
||||
admin_edited: 'text-blue-400 bg-blue-500/10',
|
||||
admin_deactivated: 'text-rose-400 bg-rose-500/10',
|
||||
new_ip_detected: 'text-amber-400 bg-amber-500/10',
|
||||
}
|
||||
|
||||
function actorLabel(row: AuditLogRow): { label: string; sub: string; kind: 'admin' | 'investor' | 'system' } {
|
||||
if (row.admin_email) {
|
||||
return { label: row.admin_name || row.admin_email, sub: row.admin_email, kind: 'admin' }
|
||||
}
|
||||
if (row.investor_email) {
|
||||
return { label: row.investor_name || row.investor_email, sub: row.investor_email, kind: 'investor' }
|
||||
}
|
||||
return { label: 'system', sub: '', kind: 'system' }
|
||||
}
|
||||
|
||||
function targetLabel(row: AuditLogRow): string | null {
|
||||
if (row.target_investor_email) {
|
||||
return row.target_investor_name
|
||||
? `${row.target_investor_name} <${row.target_investor_email}>`
|
||||
: row.target_investor_email
|
||||
}
|
||||
return null
|
||||
}
|
||||
|
||||
function formatDate(iso: string): string {
|
||||
return new Date(iso).toLocaleString()
|
||||
}
|
||||
|
||||
function summarizeDetails(action: string, details: Record<string, unknown> | null): string {
|
||||
if (!details) return ''
|
||||
if (action === 'slide_viewed' && details.slide_id) return String(details.slide_id)
|
||||
if (action === 'assumption_edited' || action === 'scenario_edited') {
|
||||
const before = details.before as Record<string, unknown> | undefined
|
||||
const after = details.after as Record<string, unknown> | undefined
|
||||
if (before && after) {
|
||||
const keys = Object.keys(after).filter(k => JSON.stringify(before[k]) !== JSON.stringify(after[k]))
|
||||
return keys.map(k => `${k}: ${JSON.stringify(before[k])} → ${JSON.stringify(after[k])}`).join(', ')
|
||||
}
|
||||
}
|
||||
if (action === 'investor_invited' || action === 'magic_link_resent') {
|
||||
return String(details.email || '')
|
||||
}
|
||||
if (action === 'investor_edited') {
|
||||
const before = details.before as Record<string, unknown> | undefined
|
||||
const after = details.after as Record<string, unknown> | undefined
|
||||
if (before && after) {
|
||||
const keys = Object.keys(after).filter(k => before[k] !== after[k])
|
||||
return keys.map(k => `${k}: "${before[k] || ''}" → "${after[k] || ''}"`).join(', ')
|
||||
}
|
||||
}
|
||||
return JSON.stringify(details).slice(0, 80)
|
||||
}
|
||||
|
||||
export default function AuditLogTable({ rows, showActor = true }: AuditLogTableProps) {
|
||||
if (rows.length === 0) {
|
||||
return <div className="text-white/40 text-sm py-8 text-center">No audit events</div>
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="overflow-x-auto">
|
||||
<table className="w-full text-sm">
|
||||
<thead>
|
||||
<tr className="text-left text-xs uppercase tracking-wider text-white/40 border-b border-white/[0.06]">
|
||||
<th className="py-3 px-3 font-medium">When</th>
|
||||
{showActor && <th className="py-3 px-3 font-medium">Actor</th>}
|
||||
<th className="py-3 px-3 font-medium">Action</th>
|
||||
<th className="py-3 px-3 font-medium">Target / Details</th>
|
||||
<th className="py-3 px-3 font-medium">IP</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{rows.map((row) => {
|
||||
const actor = actorLabel(row)
|
||||
const target = targetLabel(row)
|
||||
const summary = summarizeDetails(row.action, row.details)
|
||||
const colorClass = ACTION_COLORS[row.action] || 'text-white/60 bg-white/[0.04]'
|
||||
return (
|
||||
<tr key={row.id} className="border-b border-white/[0.04] hover:bg-white/[0.02]">
|
||||
<td className="py-3 px-3 text-white/60 whitespace-nowrap">{formatDate(row.created_at)}</td>
|
||||
{showActor && (
|
||||
<td className="py-3 px-3">
|
||||
<div className="flex items-center gap-2">
|
||||
<span
|
||||
className={`text-[9px] px-1.5 py-0.5 rounded uppercase font-semibold ${
|
||||
actor.kind === 'admin'
|
||||
? 'bg-purple-500/20 text-purple-300'
|
||||
: actor.kind === 'investor'
|
||||
? 'bg-indigo-500/20 text-indigo-300'
|
||||
: 'bg-white/10 text-white/50'
|
||||
}`}
|
||||
>
|
||||
{actor.kind}
|
||||
</span>
|
||||
<div className="min-w-0">
|
||||
<div className="text-white/80 truncate max-w-[180px]">{actor.label}</div>
|
||||
</div>
|
||||
</div>
|
||||
</td>
|
||||
)}
|
||||
<td className="py-3 px-3">
|
||||
<span className={`text-xs px-2 py-1 rounded font-mono ${colorClass}`}>{row.action}</span>
|
||||
</td>
|
||||
<td className="py-3 px-3 text-white/60 max-w-md">
|
||||
{target && <div className="text-white/80 truncate">→ {target}</div>}
|
||||
{summary && <div className="text-xs text-white/40 truncate">{summary}</div>}
|
||||
</td>
|
||||
<td className="py-3 px-3 text-white/40 font-mono text-xs">{row.ip_address || '—'}</td>
|
||||
</tr>
|
||||
)
|
||||
})}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
Reference in New Issue
Block a user