import { NextRequest, NextResponse } from 'next/server' import bcrypt from 'bcryptjs' import { prisma } from '@/lib/prisma' import { generateSessionToken, createSession } from '@/lib/session' const loginAttempts = new Map() const MAX_ATTEMPTS = 5 const WINDOW_MS = 60_000 function isRateLimited(ip: string): boolean { const now = Date.now() const attempts = (loginAttempts.get(ip) ?? []).filter((t) => now - t < WINDOW_MS) attempts.push(now) loginAttempts.set(ip, attempts) return attempts.length > MAX_ATTEMPTS } export async function POST(request: NextRequest) { const ip = request.headers.get('x-forwarded-for')?.split(',')[0]?.trim() ?? 'unknown' if (isRateLimited(ip)) { return NextResponse.json({ error: 'Too many requests' }, { status: 429 }) } let email: string, password: string try { const body = await request.json() email = (body.email ?? '').trim() password = body.password ?? '' } catch { return NextResponse.json({ error: 'Invalid request body' }, { status: 400 }) } if (!email || !password) { return NextResponse.json( { error: 'Email and password are required' }, { status: 400 } ) } const user = await prisma.user.findUnique({ where: { email } }) const validPassword = user != null && (await bcrypt.compare(password, user.pwd_hash)) if (!user || !validPassword) { return NextResponse.json({ error: 'Invalid credentials' }, { status: 401 }) } const token = generateSessionToken() await createSession(user.id, token) const response = NextResponse.json({ success: true }) response.cookies.set('ops_session', token, { httpOnly: true, secure: process.env.NODE_ENV === 'production', sameSite: 'strict', maxAge: 60 * 60 * 24, path: '/', }) return response }