All checks were successful
Deploy to Production / deploy (push) Successful in 50s
- login: SMS step now has a 60-country dial-code <select> (CH default) and a national-number input, combined into strict E.164 client-side - marketing header: probe /v1/auth/me, show "Dashboard" when signed in instead of the Sign in / Start building CTAs - dashboard overview: drop the duplicate "+ New server" button, the navbar one is the single source Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
131 lines
4.6 KiB
TypeScript
131 lines
4.6 KiB
TypeScript
'use client';
|
|
|
|
import { StatusPill } from '@/components/status-pill';
|
|
import { Button } from '@/components/ui/button';
|
|
import { apiFetch } from '@/lib/api';
|
|
import Link from 'next/link';
|
|
import { useEffect, useState } from 'react';
|
|
|
|
interface ServerRow {
|
|
id: string;
|
|
name: string;
|
|
slug: string;
|
|
status: string;
|
|
publicUrl: string | null;
|
|
createdAt: string;
|
|
}
|
|
|
|
export default function Overview() {
|
|
const [servers, setServers] = useState<ServerRow[] | null>(null);
|
|
const [err, setErr] = useState<string | null>(null);
|
|
|
|
useEffect(() => {
|
|
apiFetch<{ servers: ServerRow[] }>('/v1/servers')
|
|
.then((r) => setServers(r.servers))
|
|
.catch((e) => setErr((e as Error).message));
|
|
}, []);
|
|
|
|
if (err?.includes('401')) {
|
|
if (typeof window !== 'undefined') {
|
|
window.location.href = '/login';
|
|
}
|
|
return null;
|
|
}
|
|
|
|
const total = servers?.length ?? 0;
|
|
const live = servers?.filter((s) => s.status === 'live').length ?? 0;
|
|
|
|
return (
|
|
<div className="mx-auto max-w-7xl px-6 py-8">
|
|
<div>
|
|
<h1 className="text-[22px] font-semibold tracking-tight">Overview</h1>
|
|
<p className="mt-1 text-[13px] text-[--color-fg-muted]">
|
|
Your MCP servers, calls and recent builds.
|
|
</p>
|
|
</div>
|
|
|
|
<div className="mt-6 grid gap-3 md:grid-cols-3">
|
|
<Card label="Servers" value={total.toString()} sub={`${live} live`} />
|
|
<Card label="Calls this period" value="0" sub="of 100,000" />
|
|
<Card label="Plan" value="Hobby" sub="Upgrade in Settings" />
|
|
</div>
|
|
|
|
<div className="mt-10">
|
|
<div className="flex items-center justify-between">
|
|
<h2 className="text-[14px] font-semibold tracking-tight">Recent servers</h2>
|
|
<Link
|
|
href="/servers"
|
|
className="text-[12px] text-[--color-fg-muted] hover:text-[--color-fg]"
|
|
>
|
|
View all →
|
|
</Link>
|
|
</div>
|
|
<div className="panel mt-3">
|
|
{servers === null && (
|
|
<div className="px-4 py-3 text-[12.5px] text-[--color-fg-muted]">Loading…</div>
|
|
)}
|
|
{servers && servers.length === 0 && (
|
|
<div className="px-4 py-12 text-center">
|
|
<p className="text-[14px] text-[--color-fg]">No servers yet.</p>
|
|
<p className="mt-1 text-[12.5px] text-[--color-fg-muted]">
|
|
Describe the tool you want — we host the server.
|
|
</p>
|
|
<Link href="/servers/new" className="mt-4 inline-block">
|
|
<Button variant="primary" size="md">
|
|
Create your first server
|
|
</Button>
|
|
</Link>
|
|
</div>
|
|
)}
|
|
{servers && servers.length > 0 && (
|
|
<table className="w-full text-[12.5px]">
|
|
<thead className="border-b border-[--color-border] text-[--color-fg-subtle]">
|
|
<tr>
|
|
<th className="px-4 py-2 text-left font-medium">Name</th>
|
|
<th className="px-4 py-2 text-left font-medium">Slug</th>
|
|
<th className="px-4 py-2 text-left font-medium">Status</th>
|
|
<th className="px-4 py-2 text-left font-medium">URL</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{servers.slice(0, 5).map((s) => (
|
|
<tr
|
|
key={s.id}
|
|
className="border-b border-[--color-border] last:border-0 hover:bg-[--color-bg-subtle]"
|
|
>
|
|
<td className="px-4 py-2.5">
|
|
<Link
|
|
href={`/servers/${s.id}`}
|
|
className="font-medium hover:text-[--color-accent]"
|
|
>
|
|
{s.name}
|
|
</Link>
|
|
</td>
|
|
<td className="mono px-4 py-2.5 text-[--color-fg-muted]">{s.slug}</td>
|
|
<td className="px-4 py-2.5">
|
|
<StatusPill status={s.status as never} />
|
|
</td>
|
|
<td className="mono px-4 py-2.5 text-[--color-fg-muted]">
|
|
{s.publicUrl ?? '—'}
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</table>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function Card({ label, value, sub }: { label: string; value: string; sub: string }) {
|
|
return (
|
|
<div className="panel p-4">
|
|
<div className="text-[11px] uppercase tracking-wider text-[--color-fg-subtle]">{label}</div>
|
|
<div className="mt-1.5 text-[24px] font-semibold tracking-tight">{value}</div>
|
|
<div className="mt-1 text-[12px] text-[--color-fg-muted]">{sub}</div>
|
|
</div>
|
|
);
|
|
}
|