980d7d905d
Stack
- Next.js 15 (App Router) + TypeScript + Drizzle ORM + postgres-js
- Node scrypt for password hashing; PBKDF2 verifier for legacy gnuboard5 hashes
- pnpm workspace monorepo: apps/web + packages/{db,auth,themes,...}
Themes (admin-selectable at /admin/themes)
- basic : 그누보드 default reproduction (light, blue accent)
- eyoom : eb4_maga_005 매거진 reproduction (dark, orange accent, ranking sidebar)
- amina : Aminam Builder reproduction (light, violet gradient, card grid)
- youngcart : 영카트 shop reproduction (red accent, search bar, category nav)
DB
- New schema (12 tables) pushed to PG via drizzle-kit: members, sessions, boards,
posts, point_ledger, app_settings, bacara_*, lottery_tickets, roulette_spins,
game_points, board_groups
- Legacy data still readable from inspection2 schema via @slot/db/legacy
Verified end-to-end against the migrated DB on localhost:3000:
- Home renders with active theme tokens injected as CSS variables
- /free lists 442K real posts from inspection2.g5_write_free
- Login (testlogin/test1234) issues session cookie, header switches to
"테스트님 환영합니다 / 로그아웃"
- Switching app_settings.theme.global from eyoom → amina swaps colors,
layout, and Korean nav labels site-wide on next request
Migration docs added: 03-migration-plan, 04-theme-architecture,
05-local-dev-setup, 06-feature-inventory.
26 lines
984 B
TypeScript
26 lines
984 B
TypeScript
import { NextRequest, NextResponse } from 'next/server';
|
|
import { login, SESSION_COOKIE } from '@slot/auth';
|
|
import { getRequestIp } from '@/lib/session';
|
|
|
|
export async function POST(req: NextRequest) {
|
|
const form = await req.formData();
|
|
const loginId = String(form.get('loginId') ?? '').trim();
|
|
const password = String(form.get('password') ?? '');
|
|
if (!loginId || !password) {
|
|
return NextResponse.redirect(new URL('/login?error=1', req.url), { status: 303 });
|
|
}
|
|
const ip = await getRequestIp();
|
|
const result = await login(loginId, password, { ip, userAgent: req.headers.get('user-agent') ?? undefined });
|
|
if (!result.ok) {
|
|
return NextResponse.redirect(new URL('/login?error=1', req.url), { status: 303 });
|
|
}
|
|
const res = NextResponse.redirect(new URL('/', req.url), { status: 303 });
|
|
res.cookies.set(SESSION_COOKIE, result.session.id, {
|
|
httpOnly: true,
|
|
sameSite: 'lax',
|
|
path: '/',
|
|
expires: result.session.expiresAt,
|
|
});
|
|
return res;
|
|
}
|