e77588deb8
- Settings key-value table in Prisma with migration - getSettings() / getSetting() helpers in lib/settings.ts - Admin UI at /admin/settings with 6 sections: General, Notifications, Student profile, Legal docs, Curator permissions, Code injection - saveSettings() server action with admin-only guard - Maintenance mode: non-admin users redirected to /maintenance page - schoolName propagated to page metadata and all email templates - headCode / bodyCode injected into root layout <head> and <body> Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
28 lines
875 B
TypeScript
28 lines
875 B
TypeScript
"use server";
|
|
|
|
import { headers } from "next/headers";
|
|
import { auth } from "@/lib/auth";
|
|
import { prisma } from "@/lib/prisma";
|
|
import { revalidatePath } from "next/cache";
|
|
import { SETTINGS_DEFAULTS, type SettingsKey } from "@/lib/settings";
|
|
|
|
export async function saveSettings(data: Record<string, string>) {
|
|
const session = await auth.api.getSession({ headers: await headers() });
|
|
if (!session || session.user.role !== "admin") throw new Error("Нет доступа");
|
|
|
|
const validKeys = Object.keys(SETTINGS_DEFAULTS) as SettingsKey[];
|
|
const ops = validKeys
|
|
.filter((key) => key in data)
|
|
.map((key) =>
|
|
prisma.settings.upsert({
|
|
where: { key },
|
|
update: { value: data[key] },
|
|
create: { key, value: data[key] },
|
|
})
|
|
);
|
|
|
|
await Promise.all(ops);
|
|
revalidatePath("/admin/settings");
|
|
revalidatePath("/", "layout");
|
|
}
|