Grapevine_Disease_Detection/vineye-admin/app/api/alerts/[id]/route.ts
Yanis 720dd34fdd add MyPlantsScreen + ScanDetailScreen + enriched admin + API mobile + project summary
Mobile:
- Replace LibraryScreen with MyPlantsScreen (date-grouped scan list, swipe actions, search, pull-to-refresh)
- Add ScanDetailScreen (immersive hero, confidence bar, cepage card, share/delete)
- Add DiseaseDetailScreen + GuideDetailScreen (hero pattern, animated entry)
- Add useScanDetail, useHistory (useCallback fix), dateGrouping utility
- Connect diseases/guides to admin API with cache + offline fallback
- Add NetworkContext, ToastContext, Skeleton loading components
- Extend ScanRecord type (isFavorite, location)
- Full i18n FR/EN for all new screens

Admin (vineye-admin):
- Enrich Disease/Guide Prisma schema (timeline, conditions, actions, sections)
- Enriched disease-form (7 sections) + guide-form (structured sections editor)
- Add mobile public API endpoints (diseases, guides by slug)
- Add Prisma migration + enriched seed data
- UI polish: sidebar, login, layout updates

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-04-09 03:19:39 +02:00

100 lines
2.6 KiB
TypeScript

import { NextRequest } from "next/server";
import { prisma } from "@/lib/prisma";
import { requireAdmin } from "@/lib/auth-guard";
import { alertSchema } from "@/lib/validations";
export async function GET(
_request: NextRequest,
{ params }: { params: Promise<{ id: string }> }
) {
const { id } = await params;
const alert = await prisma.seasonAlert.findUnique({ where: { id } });
if (!alert) {
return Response.json({ error: "Alerte introuvable" }, { status: 404 });
}
return Response.json({ data: alert });
}
export async function PUT(
request: NextRequest,
{ params }: { params: Promise<{ id: string }> }
) {
const auth = await requireAdmin();
if ("error" in auth) {
return Response.json({ error: auth.error }, { status: auth.status });
}
const { id } = await params;
const existing = await prisma.seasonAlert.findUnique({ where: { id } });
if (!existing) {
return Response.json({ error: "Alerte introuvable" }, { status: 404 });
}
const body = await request.json();
const result = alertSchema.safeParse(body);
if (!result.success) {
return Response.json(
{ error: "Validation failed", details: result.error.flatten() },
{ status: 400 }
);
}
const alert = await prisma.seasonAlert.update({
where: { id },
data: result.data,
});
return Response.json({ data: alert });
}
export async function PATCH(
request: NextRequest,
{ params }: { params: Promise<{ id: string }> }
) {
const auth = await requireAdmin();
if ("error" in auth) {
return Response.json({ error: auth.error }, { status: auth.status });
}
const { id } = await params;
const existing = await prisma.seasonAlert.findUnique({ where: { id } });
if (!existing) {
return Response.json({ error: "Alerte introuvable" }, { status: 404 });
}
const body = await request.json();
if (typeof body.active !== "boolean") {
return Response.json({ error: "Champ 'active' requis" }, { status: 400 });
}
const alert = await prisma.seasonAlert.update({
where: { id },
data: { active: body.active },
});
return Response.json({ data: alert });
}
export async function DELETE(
_request: NextRequest,
{ params }: { params: Promise<{ id: string }> }
) {
const auth = await requireAdmin();
if ("error" in auth) {
return Response.json({ error: auth.error }, { status: auth.status });
}
const { id } = await params;
const existing = await prisma.seasonAlert.findUnique({ where: { id } });
if (!existing) {
return Response.json({ error: "Alerte introuvable" }, { status: 404 });
}
await prisma.seasonAlert.delete({ where: { id } });
return Response.json({ message: "Alerte supprimee" });
}