Ops-dashboard/app/audit/page.tsx

100 lines
4.2 KiB
TypeScript

import Link from 'next/link'
import { redirect } from 'next/navigation'
import { getCurrentUser } from '@/lib/session'
import { prisma } from '@/lib/prisma'
export const dynamic = 'force-dynamic'
const STATUS_STYLES: Record<string, string> = {
pending: 'bg-zinc-100 text-zinc-600 dark:bg-zinc-800 dark:text-zinc-400',
running: 'bg-amber-100 text-amber-700 dark:bg-amber-900/30 dark:text-amber-400',
success: 'bg-green-100 text-green-800 dark:bg-green-900/30 dark:text-green-400',
failed: 'bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-400',
cancelled: 'bg-zinc-100 text-zinc-600 dark:bg-zinc-800 dark:text-zinc-400',
}
export default async function AuditPage() {
const user = await getCurrentUser()
if (!user) redirect('/login')
const runs = await prisma.flowRun.findMany({
where: { user_id: user.id },
orderBy: { started_at: 'desc' },
take: 100,
})
return (
<div className="min-h-screen bg-background p-6">
<div className="mx-auto max-w-6xl space-y-6">
<div className="flex items-center justify-between">
<div className="space-y-1">
<h1 className="text-2xl font-semibold tracking-tight">Audit Log</h1>
<p className="text-sm text-muted-foreground">Recent write actions executed on this server</p>
</div>
</div>
{runs.length === 0 ? (
<div className="rounded-lg border border-border p-8 text-center text-sm text-muted-foreground">
No actions have been run yet.
</div>
) : (
<div className="overflow-x-auto rounded-lg border border-border">
<table className="w-full text-sm">
<thead>
<tr className="border-b border-border bg-muted/50">
<th className="px-4 py-3 text-left font-medium text-muted-foreground">Command</th>
<th className="px-4 py-3 text-left font-medium text-muted-foreground">Status</th>
<th className="px-4 py-3 text-left font-medium text-muted-foreground">Exit</th>
<th className="px-4 py-3 text-left font-medium text-muted-foreground">Started</th>
<th className="px-4 py-3 text-left font-medium text-muted-foreground">Duration</th>
</tr>
</thead>
<tbody>
{runs.map((run) => {
const durationMs =
run.ended_at && run.started_at
? run.ended_at.getTime() - run.started_at.getTime()
: null
return (
<tr
key={run.id}
className="border-b border-border last:border-0 hover:bg-muted/30 transition-colors"
>
<td className="px-4 py-3 font-mono text-xs">
<Link
href={`/audit/${run.id}`}
className="font-medium text-foreground hover:underline"
>
{run.flow_key}
</Link>
</td>
<td className="px-4 py-3">
<span
className={
'inline-flex items-center rounded-full px-2 py-0.5 text-xs font-medium ' +
(STATUS_STYLES[run.status] ?? '')
}
>
{run.status}
</span>
</td>
<td className="px-4 py-3 font-mono text-xs text-muted-foreground">
{run.exit_code != null ? run.exit_code : '—'}
</td>
<td className="px-4 py-3 text-xs text-muted-foreground">
{run.started_at.toLocaleString()}
</td>
<td className="px-4 py-3 text-xs text-muted-foreground">
{durationMs != null ? `${(durationMs / 1000).toFixed(1)}s` : '—'}
</td>
</tr>
)
})}
</tbody>
</table>
</div>
)}
</div>
</div>
)
}