Comprehensive admin + user dashboards (production-ready)

This commit is contained in:
Leon Serfaty
2026-06-07 17:54:30 -04:00
parent 155507f21a
commit f033f00379
122 changed files with 7878 additions and 805 deletions
+100 -73
View File
@@ -1,91 +1,118 @@
import type { Metadata } from "next";
import { prisma } from "@/lib/db";
import { PageHeader } from "@/components/app/page-header";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
import { Badge } from "@/components/ui/badge";
import { PLANS, type PlanKey } from "@/lib/billing/plans";
import { DollarSign, TrendingUp, CreditCard } from "lucide-react";
import { listSubscriptions, SUBS_PAGE_SIZE, type AdminSubRow } from "@/lib/admin/billing";
import { getOverview } from "@/lib/admin/metrics";
import { formatPrice } from "@/lib/utils";
import { PageHeader } from "@/components/app/page-header";
import { StatCard } from "@/components/admin/ui/stat-card";
import { DataTable, TableToolbar, type Column } from "@/components/admin/ui/data-table";
import { SearchInput, FilterSelect, Pagination } from "@/components/admin/ui/table-controls";
import { SubscriptionRowActions } from "@/components/admin/subscription-row-actions";
import { Badge } from "@/components/ui/badge";
export const metadata: Metadata = { title: "Admin · Subscriptions" };
export default async function AdminSubscriptionsPage() {
const subs = await prisma.subscription.findMany({ orderBy: { createdAt: "desc" }, take: 200 });
const refIds = subs.map((s) => s.referenceId);
const [users, orgs] = await Promise.all([
prisma.user.findMany({ where: { id: { in: refIds } }, select: { id: true, name: true, email: true } }),
prisma.organization.findMany({ where: { id: { in: refIds } }, select: { id: true, name: true } }),
export default async function AdminSubscriptionsPage({
searchParams,
}: {
searchParams: Promise<Record<string, string | undefined>>;
}) {
const sp = await searchParams;
const page = Math.max(1, Number(sp.page ?? "1"));
const [m, { rows, total }] = await Promise.all([
getOverview("30d"),
listSubscriptions({ status: sp.status, provider: sp.provider, search: sp.q, page }),
]);
const nameByRef = new Map<string, string>();
for (const u of users) nameByRef.set(u.id, u.email);
for (const o of orgs) nameByRef.set(o.id, o.name);
const active = subs.filter((s) => ["active", "trialing"].includes(s.status));
const mrr = active.reduce((sum, s) => sum + (PLANS[s.plan as PlanKey]?.priceMonthly ?? 0), 0);
const stripeCount = active.filter((s) => s.provider === "stripe").length;
const paypalCount = active.filter((s) => s.provider === "paypal").length;
const columns: Column<AdminSubRow>[] = [
{
key: "customer",
header: "Customer",
cell: (s) => <span className="truncate font-medium">{s.customer}</span>,
},
{ key: "plan", header: "Plan", cell: (s) => <span className="capitalize">{s.plan}</span> },
{ key: "provider", header: "Provider", cell: (s) => <span className="capitalize">{s.provider}</span> },
{
key: "status",
header: "Status",
cell: (s) => (
<Badge variant={["active", "trialing"].includes(s.status) ? "success" : "secondary"}>
{s.cancelAtPeriodEnd ? "cancels soon" : s.status}
</Badge>
),
},
{
key: "renews",
header: "Renews",
cell: (s) => (
<span className="text-muted-foreground">{s.periodEnd ? s.periodEnd.toLocaleDateString() : "—"}</span>
),
},
{
key: "actions",
header: "",
align: "right",
cell: (s) => (
<SubscriptionRowActions
sub={{
id: s.id,
referenceId: s.referenceId,
provider: s.provider,
status: s.status,
cancelAtPeriodEnd: s.cancelAtPeriodEnd,
}}
/>
),
},
];
return (
<>
<PageHeader title="Subscriptions" description="Revenue and active subscriptions." />
<div className="mb-6 grid gap-4 sm:grid-cols-4">
<Stat label="MRR" value={formatPrice(mrr)} />
<Stat label="ARR" value={formatPrice(mrr * 12)} />
<Stat label="Stripe" value={String(stripeCount)} />
<Stat label="PayPal" value={String(paypalCount)} />
<div className="mb-6 grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
<StatCard label="MRR" value={formatPrice(m.mrr)} icon={DollarSign} />
<StatCard label="ARR" value={formatPrice(m.arr)} icon={TrendingUp} />
<StatCard label="Stripe" value={String(m.providerSplit.stripe)} icon={CreditCard} />
<StatCard
label="PayPal"
value={String(m.providerSplit.paypal)}
icon={CreditCard}
hint={m.providerSplit.comp ? `${m.providerSplit.comp} comped` : undefined}
/>
</div>
<div className="overflow-x-auto rounded-lg border">
<table className="w-full text-sm">
<thead className="border-b bg-muted/40 text-left text-xs uppercase text-muted-foreground">
<tr>
<th className="p-3 font-medium">Customer</th>
<th className="p-3 font-medium">Plan</th>
<th className="p-3 font-medium">Provider</th>
<th className="p-3 font-medium">Status</th>
<th className="p-3 font-medium">Renews</th>
</tr>
</thead>
<tbody className="divide-y">
{subs.map((s) => (
<tr key={s.id} className="hover:bg-muted/20">
<td className="p-3">{nameByRef.get(s.referenceId) ?? s.referenceId}</td>
<td className="p-3 capitalize">{s.plan}</td>
<td className="p-3 capitalize">{s.provider}</td>
<td className="p-3">
<Badge variant={["active", "trialing"].includes(s.status) ? "success" : "secondary"}>
{s.cancelAtPeriodEnd ? "cancels soon" : s.status}
</Badge>
</td>
<td className="p-3 text-muted-foreground">
{s.periodEnd ? s.periodEnd.toLocaleDateString() : "—"}
</td>
</tr>
))}
{subs.length === 0 && (
<tr>
<td colSpan={5} className="p-8 text-center text-muted-foreground">
No subscriptions yet.
</td>
</tr>
)}
</tbody>
</table>
<TableToolbar>
<SearchInput placeholder="Search customer…" />
<div className="flex flex-wrap gap-2">
<FilterSelect
param="status"
placeholder="Status"
allLabel="All status"
options={[
{ value: "active", label: "Active" },
{ value: "trialing", label: "Trialing" },
{ value: "past_due", label: "Past due" },
{ value: "canceled", label: "Canceled" },
{ value: "paused", label: "Paused" },
]}
/>
<FilterSelect
param="provider"
placeholder="Provider"
allLabel="All providers"
options={[
{ value: "stripe", label: "Stripe" },
{ value: "paypal", label: "PayPal" },
{ value: "comp", label: "Comped" },
]}
/>
</div>
</TableToolbar>
<DataTable columns={columns} rows={rows} getRowKey={(s) => s.id} empty="No subscriptions match your filters." />
<div className="mt-4">
<Pagination page={page} pageSize={SUBS_PAGE_SIZE} total={total} />
</div>
</>
);
}
function Stat({ label, value }: { label: string; value: string }) {
return (
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm font-medium text-muted-foreground">{label}</CardTitle>
</CardHeader>
<CardContent>
<p className="font-display text-3xl font-extrabold tracking-tight">{value}</p>
</CardContent>
</Card>
);
}