2026-04-28 17:55:09 +02:00
|
|
|
import type { CompetitionCategory, PrismaClient } from '@prisma/client'
|
|
|
|
|
import { signFinalistToken } from '@/lib/finalist-token'
|
2026-04-28 17:58:31 +02:00
|
|
|
import { sendFinalistConfirmationEmail } from '@/lib/email'
|
2026-04-28 17:55:09 +02:00
|
|
|
|
2026-04-28 17:58:31 +02:00
|
|
|
type AnyPrisma = Pick<PrismaClient, 'finalistConfirmation' | 'waitlistEntry' | 'project'>
|
2026-04-28 17:55:09 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Create a PENDING FinalistConfirmation row with a signed token. Caller is
|
|
|
|
|
* responsible for sending the notification email separately.
|
|
|
|
|
*/
|
|
|
|
|
export async function createPendingConfirmation(
|
2026-04-28 17:58:31 +02:00
|
|
|
prisma: Pick<PrismaClient, 'finalistConfirmation'>,
|
2026-04-28 17:55:09 +02:00
|
|
|
args: {
|
|
|
|
|
projectId: string
|
|
|
|
|
category: CompetitionCategory
|
|
|
|
|
windowHours: number
|
|
|
|
|
promotedFromWaitlistEntryId?: string
|
|
|
|
|
},
|
|
|
|
|
): Promise<{ id: string; token: string; deadline: Date }> {
|
|
|
|
|
const deadline = new Date(Date.now() + args.windowHours * 3_600_000)
|
|
|
|
|
// Generate the row ID up front so we can sign it into the token before
|
|
|
|
|
// writing the row (token is unique-indexed; embedding the ID gives the
|
|
|
|
|
// public verify path a stable lookup key).
|
|
|
|
|
const id = `cmfc_${Math.random().toString(36).slice(2, 10)}_${Date.now().toString(36)}`
|
|
|
|
|
const token = signFinalistToken({
|
|
|
|
|
confirmationId: id,
|
|
|
|
|
exp: Math.floor(deadline.getTime() / 1000),
|
|
|
|
|
})
|
|
|
|
|
await prisma.finalistConfirmation.create({
|
|
|
|
|
data: {
|
|
|
|
|
id,
|
|
|
|
|
projectId: args.projectId,
|
|
|
|
|
category: args.category,
|
|
|
|
|
status: 'PENDING',
|
|
|
|
|
deadline,
|
|
|
|
|
token,
|
|
|
|
|
promotedFromWaitlistEntryId: args.promotedFromWaitlistEntryId ?? null,
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
return { id, token, deadline }
|
|
|
|
|
}
|
2026-04-28 17:58:31 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Promote the lowest-ranked WAITING waitlist entry in the given category to
|
|
|
|
|
* PROMOTED, create a fresh PENDING confirmation for the project, and send
|
|
|
|
|
* the notification email. No-op if no WAITING entry exists.
|
|
|
|
|
*/
|
|
|
|
|
export async function promoteNextWaitlistEntry(
|
|
|
|
|
prisma: AnyPrisma,
|
|
|
|
|
args: { programId: string; category: CompetitionCategory; windowHours: number },
|
|
|
|
|
): Promise<{ promoted: boolean; entryId?: string; confirmationId?: string }> {
|
|
|
|
|
const entry = await prisma.waitlistEntry.findFirst({
|
|
|
|
|
where: {
|
|
|
|
|
programId: args.programId,
|
|
|
|
|
category: args.category,
|
|
|
|
|
status: 'WAITING',
|
|
|
|
|
},
|
|
|
|
|
orderBy: { rank: 'asc' },
|
|
|
|
|
})
|
|
|
|
|
if (!entry) return { promoted: false }
|
|
|
|
|
|
|
|
|
|
await prisma.waitlistEntry.update({
|
|
|
|
|
where: { id: entry.id },
|
|
|
|
|
data: { status: 'PROMOTED' },
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const { id: confirmationId, token, deadline } = await createPendingConfirmation(prisma, {
|
|
|
|
|
projectId: entry.projectId,
|
|
|
|
|
category: args.category,
|
|
|
|
|
windowHours: args.windowHours,
|
|
|
|
|
promotedFromWaitlistEntryId: entry.id,
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Send email — log and continue on failure.
|
|
|
|
|
const project = await prisma.project.findUnique({
|
|
|
|
|
where: { id: entry.projectId },
|
|
|
|
|
select: {
|
|
|
|
|
title: true,
|
|
|
|
|
teamMembers: {
|
|
|
|
|
where: { role: 'LEAD' },
|
|
|
|
|
take: 1,
|
|
|
|
|
select: { user: { select: { email: true, name: true } } },
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
const lead = project?.teamMembers[0]?.user
|
|
|
|
|
if (lead?.email && project) {
|
|
|
|
|
const baseUrl = (process.env.NEXTAUTH_URL ?? 'http://localhost:3000').replace(/\/$/, '')
|
|
|
|
|
const confirmUrl = `${baseUrl}/finalist/confirm/${token}`
|
|
|
|
|
try {
|
|
|
|
|
await sendFinalistConfirmationEmail(
|
|
|
|
|
lead.email,
|
|
|
|
|
lead.name ?? null,
|
|
|
|
|
project.title,
|
|
|
|
|
deadline,
|
|
|
|
|
confirmUrl,
|
|
|
|
|
)
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error(
|
|
|
|
|
`[promoteNextWaitlistEntry] failed to send email for project ${entry.projectId}:`,
|
|
|
|
|
err,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { promoted: true, entryId: entry.id, confirmationId }
|
|
|
|
|
}
|