- Drink Images: upload/display photos of bottles/cans on drink cards and detail pages - My Bar: inventory tracker for spirits, liqueurs, mixers, bitters, garnishes, tools - Bartender: AI-powered cocktail recipe generation, "what can I make" suggestions, saved recipes. Cross-references bar inventory for ingredient availability. - Recommend: AI flavor profile analysis, personalized drink recommendations, "find similar" drinks based on highly-rated favorites - Navigation: desktop sidebar with all 8 routes, mobile bottom nav with 4 primary items + "More" popup menu - New Prisma models: BarItem, Recipe, FlavorProfile - Backup/restore updated to include bar items Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
66 lines
1.7 KiB
TypeScript
66 lines
1.7 KiB
TypeScript
import { auth } from "@/lib/auth"
|
|
import { prisma } from "@/lib/prisma"
|
|
import { generateBackupCsv } from "@/lib/backup"
|
|
import { NextResponse } from "next/server"
|
|
|
|
export async function GET() {
|
|
const session = await auth()
|
|
if (!session?.user?.id) {
|
|
return NextResponse.json({ error: "Unauthorized" }, { status: 401 })
|
|
}
|
|
|
|
const userId = session.user.id
|
|
|
|
try {
|
|
const [drinks, ratings, wishlistItems, preferences, sharedLists, barItems] =
|
|
await Promise.all([
|
|
prisma.drink.findMany({
|
|
where: { userId },
|
|
orderBy: { createdAt: "asc" },
|
|
}),
|
|
prisma.rating.findMany({
|
|
where: { userId },
|
|
include: { drink: { select: { name: true } } },
|
|
orderBy: { createdAt: "asc" },
|
|
}),
|
|
prisma.wishlistItem.findMany({
|
|
where: { userId },
|
|
orderBy: { createdAt: "asc" },
|
|
}),
|
|
prisma.userPreference.findUnique({ where: { userId } }),
|
|
prisma.sharedList.findMany({
|
|
where: { userId },
|
|
orderBy: { createdAt: "asc" },
|
|
}),
|
|
prisma.barItem.findMany({
|
|
where: { userId },
|
|
orderBy: { createdAt: "asc" },
|
|
}),
|
|
])
|
|
|
|
const csv = generateBackupCsv(
|
|
drinks,
|
|
ratings,
|
|
wishlistItems,
|
|
preferences,
|
|
sharedLists,
|
|
barItems
|
|
)
|
|
|
|
const date = new Date().toISOString().split("T")[0]
|
|
|
|
return new Response(csv, {
|
|
headers: {
|
|
"Content-Type": "text/csv; charset=utf-8",
|
|
"Content-Disposition": `attachment; filename="drinktracker-backup-${date}.csv"`,
|
|
},
|
|
})
|
|
} catch (error) {
|
|
console.error("Backup export error:", error)
|
|
return NextResponse.json(
|
|
{ error: "Failed to generate backup" },
|
|
{ status: 500 }
|
|
)
|
|
}
|
|
}
|