Files
lms-sb/src/app/api/admin/lesson-files/route.ts
T
admins 444b9c0faf Apply tech debt fixes: middleware rename, React.cache, file size limits, remove dead deps
- Rename proxy.ts → middleware.ts, export proxy() → middleware() so Next.js edge
  protection actually activates (F001)
- Add PUBLIC_ROUTES entries for /forgot-password and /reset-password
- Wrap getSettings() in React.cache() to eliminate duplicate DB call in root layout (F003)
- Remove 4 console.log calls from saveLesson Server Action, keep console.error (F005)
- Add 50 MB file size guard to all 6 upload routes before arrayBuffer() read (F004)
- Remove unused deps: @tailwindcss/typography, shadcn, tw-animate-css (F008)
- Update CLAUDE.md: Prisma version 6.x → 7.x
- Add TECH_DEBT_AUDIT.md with 14 findings across 9 dimensions

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-09 19:35:05 +05:00

81 lines
2.8 KiB
TypeScript

import { NextRequest, NextResponse } from "next/server";
import { headers } from "next/headers";
import { auth } from "@/lib/auth";
import { prisma } from "@/lib/prisma";
import { uploadFile, deleteFile } from "@/lib/s3";
import { randomUUID } from "crypto";
async function requireAdmin() {
const session = await auth.api.getSession({ headers: await headers() });
if (!session || session.user.role !== "admin") return null;
return session;
}
export async function POST(req: NextRequest) {
if (!await requireAdmin()) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const form = await req.formData();
const file = form.get("file") as File | null;
const lessonId = form.get("lessonId") as string | null;
const label = (form.get("label") as string | null)?.trim() || null;
if (!file || !lessonId) return NextResponse.json({ error: "Missing fields" }, { status: 400 });
const MAX_BYTES = 50 * 1024 * 1024;
if (file.size > MAX_BYTES) return NextResponse.json({ error: "Файл слишком большой" }, { status: 413 });
const name = label ?? file.name;
const existing = await prisma.lessonFile.findFirst({ where: { lessonId, name } });
const ext = file.name.split(".").pop() ?? "bin";
const key = `lessons/${lessonId}/${randomUUID()}.${ext}`;
const buffer = Buffer.from(await file.arrayBuffer());
const url = await uploadFile(key, buffer, file.type);
if (existing) {
const oldKey = existing.url.split(`/${process.env.S3_BUCKET}/`)[1];
if (oldKey) await deleteFile(oldKey).catch(() => {});
const lessonFile = await prisma.lessonFile.update({
where: { id: existing.id },
data: { url, size: file.size },
});
return NextResponse.json(lessonFile);
}
const lessonFile = await prisma.lessonFile.create({
data: { lessonId, name, url, size: file.size },
});
return NextResponse.json(lessonFile);
}
export async function PATCH(req: NextRequest) {
if (!await requireAdmin()) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { fileId, label } = await req.json();
if (!fileId || typeof label !== "string") {
return NextResponse.json({ error: "Missing fields" }, { status: 400 });
}
const updated = await prisma.lessonFile.update({
where: { id: fileId },
data: { name: label.trim() || undefined },
});
return NextResponse.json(updated);
}
export async function DELETE(req: NextRequest) {
if (!await requireAdmin()) {
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
}
const { fileId, url } = await req.json();
if (url) {
const key = (url as string).split(`/${process.env.S3_BUCKET}/`)[1];
if (key) await deleteFile(key).catch(() => {});
}
await prisma.lessonFile.delete({ where: { id: fileId } });
return NextResponse.json({ ok: true });
}