All checks were successful
Build and Push Docker Image / build (push) Successful in 7m51s
External lunch attendees had no way to pick their own dish — an admin had to set
it inline and no email was ever sent. (Marine added herself as an external
expecting a dish-selection link and never received one.)
Adds:
- ExternalAttendee.inviteSentAt + additive migration
- HMAC-signed external lunch token (mirrors finalist-token)
- Public no-login picker page /lunch/pick/[token] — dish + allergens + notes,
gated by the lunch change deadline, read-only after
- tRPC getExternalByToken / setExternalPick (public) + sendExternalInvite (admin)
- Auto-send invite on createExternal when an email is present; per-row resend
button + status chip (Invited / Picked / no email) in the logistics screen
- Unpicked externals chased by the lunch reminder cron + manual "Send reminders"
- sendExternalDishInviteEmail (branded). Page + email title use the configurable
venue ("Lunch at {venue}") rather than "grand finale"
Tests: token roundtrip/tamper/expiry, selectUnpickedExternals filter,
get/set-by-token happy + deadline + bad-token, createExternal auto-send,
cron external reminders. Full suite 303 passing; build clean.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
83 lines
2.8 KiB
TypeScript
83 lines
2.8 KiB
TypeScript
import { NextResponse, type NextRequest } from 'next/server'
|
|
import { prisma } from '@/lib/prisma'
|
|
import { sendLunchReminderEmail } from '@/lib/email'
|
|
import {
|
|
selectUnpickedAttendees,
|
|
selectUnpickedExternals,
|
|
} from '@/server/services/lunch-reminders'
|
|
import { sendExternalDishInvite } from '@/server/services/lunch-external-invite'
|
|
|
|
/**
|
|
* Cron: send a single reminder email per attending member who hasn't picked
|
|
* a lunch dish yet, when we're inside the reminder window
|
|
* (deadline - reminderHoursBeforeDeadline) <= now < deadline.
|
|
*
|
|
* Idempotent — `LunchEvent.reminderSentAt` blocks repeat sends.
|
|
*/
|
|
export async function GET(request: NextRequest): Promise<NextResponse> {
|
|
const cronSecret = request.headers.get('x-cron-secret')
|
|
if (!cronSecret || cronSecret !== process.env.CRON_SECRET) {
|
|
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
}
|
|
const now = new Date()
|
|
const events = await prisma.lunchEvent.findMany({
|
|
where: {
|
|
enabled: true,
|
|
reminderSentAt: null,
|
|
reminderHoursBeforeDeadline: { not: null },
|
|
eventAt: { not: null },
|
|
},
|
|
})
|
|
let sent = 0
|
|
for (const event of events) {
|
|
try {
|
|
if (!event.eventAt || event.reminderHoursBeforeDeadline == null) continue
|
|
const deadline = new Date(
|
|
event.eventAt.getTime() - event.changeCutoffHours * 3_600_000,
|
|
)
|
|
const reminderAt = new Date(
|
|
deadline.getTime() - event.reminderHoursBeforeDeadline * 3_600_000,
|
|
)
|
|
if (now < reminderAt || now >= deadline) continue
|
|
|
|
const ams = await selectUnpickedAttendees(prisma, event)
|
|
for (const am of ams) {
|
|
if (!am.user.email) continue
|
|
try {
|
|
await sendLunchReminderEmail({
|
|
to: am.user.email,
|
|
memberName: am.user.name ?? am.user.email,
|
|
eventAt: event.eventAt,
|
|
venue: event.venue,
|
|
changeDeadline: deadline,
|
|
pickUrl: `${process.env.NEXTAUTH_URL ?? ''}/applicant`,
|
|
})
|
|
sent++
|
|
} catch (e) {
|
|
console.error('[lunch-reminders] send failed for', am.user.email, e)
|
|
}
|
|
}
|
|
|
|
// External attendees: emailed + no dish yet → their tokenized pick page.
|
|
const externals = await selectUnpickedExternals(prisma, { id: event.id })
|
|
for (const ext of externals) {
|
|
if (!ext.email) continue
|
|
try {
|
|
await sendExternalDishInvite(prisma, ext, event)
|
|
sent++
|
|
} catch (e) {
|
|
console.error('[lunch-reminders] external send failed for', ext.email, e)
|
|
}
|
|
}
|
|
|
|
await prisma.lunchEvent.update({
|
|
where: { id: event.id },
|
|
data: { reminderSentAt: new Date() },
|
|
})
|
|
} catch (e) {
|
|
console.error('[lunch-reminders] event failed', event.id, e)
|
|
}
|
|
}
|
|
return NextResponse.json({ ok: true, sent, processedEvents: events.length })
|
|
}
|