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>
154 lines
6.2 KiB
TypeScript
154 lines
6.2 KiB
TypeScript
'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>
|
|
)
|
|
}
|