Fresh-DB detection on the login screen — if no super-admin row exists, /api/v1/bootstrap/status reports needsBootstrap and login redirects to /setup, which mints the first super-admin via /api/v1/bootstrap/super-admin. Endpoint refuses once any user already exists. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
141 lines
4.7 KiB
TypeScript
141 lines
4.7 KiB
TypeScript
'use client';
|
||
|
||
import { useEffect, useState } from 'react';
|
||
import { useRouter } from 'next/navigation';
|
||
import Link from 'next/link';
|
||
import { useForm } from 'react-hook-form';
|
||
import { zodResolver } from '@hookform/resolvers/zod';
|
||
import { z } from 'zod';
|
||
import { toast } from 'sonner';
|
||
import { cn } from '@/lib/utils';
|
||
import { Button } from '@/components/ui/button';
|
||
import { Input } from '@/components/ui/input';
|
||
import { Label } from '@/components/ui/label';
|
||
import { BrandedAuthShell } from '@/components/shared/branded-auth-shell';
|
||
|
||
// `identifier` accepts either an email address or a username (3–30 lowercase
|
||
// letters / digits / dot / underscore / hyphen). The server endpoint
|
||
// /api/auth/sign-in-by-identifier resolves the username server-side and
|
||
// forwards to better-auth in one round-trip — the canonical email is never
|
||
// returned to the browser, which closes the username-enumeration vector.
|
||
const loginSchema = z.object({
|
||
identifier: z.string().min(1, 'Email or username is required'),
|
||
password: z.string().min(1, 'Password is required'),
|
||
});
|
||
|
||
type LoginFormData = z.infer<typeof loginSchema>;
|
||
|
||
export default function LoginPage() {
|
||
const router = useRouter();
|
||
const [isLoading, setIsLoading] = useState(false);
|
||
|
||
// Fresh-DB bootstrap detection: if no super-admin exists yet, /setup
|
||
// owns the first-run flow. Failure of the status endpoint is silent
|
||
// (login still works for everyone else).
|
||
useEffect(() => {
|
||
let cancelled = false;
|
||
fetch('/api/v1/bootstrap/status')
|
||
.then((r) => (r.ok ? (r.json() as Promise<{ data?: { needsBootstrap?: boolean } }>) : null))
|
||
.then((payload) => {
|
||
if (cancelled || !payload) return;
|
||
if (payload.data?.needsBootstrap) router.replace('/setup');
|
||
})
|
||
.catch(() => {
|
||
/* silent — login UX must still work even if status check fails */
|
||
});
|
||
return () => {
|
||
cancelled = true;
|
||
};
|
||
}, [router]);
|
||
|
||
const {
|
||
register,
|
||
handleSubmit,
|
||
formState: { errors },
|
||
} = useForm<LoginFormData>({
|
||
resolver: zodResolver(loginSchema),
|
||
});
|
||
|
||
async function onSubmit(data: LoginFormData) {
|
||
setIsLoading(true);
|
||
try {
|
||
const res = await fetch('/api/auth/sign-in-by-identifier', {
|
||
method: 'POST',
|
||
headers: { 'Content-Type': 'application/json' },
|
||
body: JSON.stringify({
|
||
identifier: data.identifier.trim(),
|
||
password: data.password,
|
||
}),
|
||
});
|
||
|
||
if (!res.ok) {
|
||
const payload = (await res.json().catch(() => ({}))) as {
|
||
error?: { message?: string };
|
||
};
|
||
toast.error(payload.error?.message ?? 'Invalid credentials');
|
||
return;
|
||
}
|
||
|
||
router.push('/dashboard');
|
||
} catch {
|
||
toast.error('Something went wrong. Please try again.');
|
||
} finally {
|
||
setIsLoading(false);
|
||
}
|
||
}
|
||
|
||
return (
|
||
<BrandedAuthShell>
|
||
<div className="text-center mb-6">
|
||
<h1 className="text-xl font-semibold text-gray-900">Port Nimara CRM</h1>
|
||
<p className="text-sm text-gray-500 mt-1">Sign in to continue</p>
|
||
</div>
|
||
|
||
<form onSubmit={handleSubmit(onSubmit)} className="space-y-4" noValidate>
|
||
<div className="space-y-1.5">
|
||
<Label htmlFor="identifier">Email or username</Label>
|
||
<Input
|
||
id="identifier"
|
||
type="text"
|
||
autoComplete="username"
|
||
autoCapitalize="none"
|
||
spellCheck={false}
|
||
disabled={isLoading}
|
||
className={cn(errors.identifier && 'border-destructive focus-visible:ring-destructive')}
|
||
{...register('identifier')}
|
||
/>
|
||
{errors.identifier && (
|
||
<p className="text-sm text-destructive">{errors.identifier.message}</p>
|
||
)}
|
||
</div>
|
||
|
||
<div className="space-y-1.5">
|
||
<div className="flex items-center justify-between">
|
||
<Label htmlFor="password">Password</Label>
|
||
<Link href="/reset-password" className="text-xs text-[#007bff] hover:underline">
|
||
Forgot password?
|
||
</Link>
|
||
</div>
|
||
<Input
|
||
id="password"
|
||
type="password"
|
||
autoComplete="current-password"
|
||
disabled={isLoading}
|
||
className={cn(errors.password && 'border-destructive focus-visible:ring-destructive')}
|
||
{...register('password')}
|
||
/>
|
||
{errors.password && <p className="text-sm text-destructive">{errors.password.message}</p>}
|
||
</div>
|
||
|
||
<Button
|
||
type="submit"
|
||
className="w-full bg-[#007bff] hover:bg-[#0069d9] text-white"
|
||
disabled={isLoading}
|
||
>
|
||
{isLoading ? 'Signing in…' : 'Sign in'}
|
||
</Button>
|
||
</form>
|
||
</BrandedAuthShell>
|
||
);
|
||
}
|