feat(applicant): getMyLogistics (hotel+flight+visa) + submitter-match fix
- Fix getMyFinalistConfirmation to resolve project via OR [submittedByUserId, teamMembers] so a lead who submitted but has no TeamMember row can see their card. - Add getMyLogistics query: returns projectTitle, confirmationStatus, hotel (program 1:1), myFlight (caller's AttendingMember.flightDetail), visaVisible flag, and myVisa when visible. Returns null for non-confirmed or unrelated callers. - Tests: confirmed finalist sees hotel/flight/visa; non-finalist gets null; PENDING confirmation gets null. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -2747,7 +2747,13 @@ export const applicantRouter = router({
|
|||||||
*/
|
*/
|
||||||
getMyFinalistConfirmation: protectedProcedure.query(async ({ ctx }) => {
|
getMyFinalistConfirmation: protectedProcedure.query(async ({ ctx }) => {
|
||||||
const project = await ctx.prisma.project.findFirst({
|
const project = await ctx.prisma.project.findFirst({
|
||||||
where: { teamMembers: { some: { userId: ctx.user.id } } },
|
where: {
|
||||||
|
OR: [
|
||||||
|
{ submittedByUserId: ctx.user.id },
|
||||||
|
{ teamMembers: { some: { userId: ctx.user.id } } },
|
||||||
|
],
|
||||||
|
finalistConfirmation: { isNot: null },
|
||||||
|
},
|
||||||
include: {
|
include: {
|
||||||
program: { select: { id: true, defaultAttendeeCap: true } },
|
program: { select: { id: true, defaultAttendeeCap: true } },
|
||||||
teamMembers: {
|
teamMembers: {
|
||||||
@@ -2858,4 +2864,77 @@ export const applicantRouter = router({
|
|||||||
projectId: a.confirmation.project.id,
|
projectId: a.confirmation.project.id,
|
||||||
}))
|
}))
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns logistics info for the caller's confirmed finalist project:
|
||||||
|
* hotel details, the caller's own flight detail, and visa visibility + status.
|
||||||
|
* Returns null when the caller has no confirmed finalist project.
|
||||||
|
*/
|
||||||
|
getMyLogistics: protectedProcedure.query(async ({ ctx }) => {
|
||||||
|
const project = await ctx.prisma.project.findFirst({
|
||||||
|
where: {
|
||||||
|
OR: [
|
||||||
|
{ submittedByUserId: ctx.user.id },
|
||||||
|
{ teamMembers: { some: { userId: ctx.user.id } } },
|
||||||
|
],
|
||||||
|
finalistConfirmation: { isNot: null },
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
program: {
|
||||||
|
select: { id: true, visaStatusVisibleToMembers: true },
|
||||||
|
},
|
||||||
|
finalistConfirmation: true,
|
||||||
|
},
|
||||||
|
orderBy: { createdAt: 'desc' },
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!project || !project.finalistConfirmation) return null
|
||||||
|
if (project.finalistConfirmation.status !== 'CONFIRMED') return null
|
||||||
|
|
||||||
|
const programId = project.program.id
|
||||||
|
const confirmationId = project.finalistConfirmation.id
|
||||||
|
const visaVisible = project.program.visaStatusVisibleToMembers
|
||||||
|
|
||||||
|
// Hotel (1:1 per program)
|
||||||
|
const hotelRow = await ctx.prisma.hotel.findUnique({
|
||||||
|
where: { programId },
|
||||||
|
select: { name: true, address: true, link: true, notes: true },
|
||||||
|
})
|
||||||
|
const hotel = hotelRow ?? null
|
||||||
|
|
||||||
|
// Caller's own AttendingMember + flight + visa
|
||||||
|
const attendee = await ctx.prisma.attendingMember.findFirst({
|
||||||
|
where: { confirmationId, userId: ctx.user.id },
|
||||||
|
include: {
|
||||||
|
flightDetail: true,
|
||||||
|
visaApplication: visaVisible,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const fd = attendee?.flightDetail ?? null
|
||||||
|
const myFlight = fd
|
||||||
|
? {
|
||||||
|
arrivalAt: fd.arrivalAt,
|
||||||
|
arrivalFlightNumber: fd.arrivalFlightNumber,
|
||||||
|
arrivalAirport: fd.arrivalAirport,
|
||||||
|
departureAt: fd.departureAt,
|
||||||
|
departureFlightNumber: fd.departureFlightNumber,
|
||||||
|
departureAirport: fd.departureAirport,
|
||||||
|
status: fd.status,
|
||||||
|
}
|
||||||
|
: null
|
||||||
|
|
||||||
|
const va = visaVisible ? (attendee?.visaApplication ?? null) : null
|
||||||
|
const myVisa = va ? { status: va.status, nationality: va.nationality } : null
|
||||||
|
|
||||||
|
return {
|
||||||
|
projectTitle: project.title,
|
||||||
|
confirmationStatus: project.finalistConfirmation.status,
|
||||||
|
hotel,
|
||||||
|
myFlight,
|
||||||
|
visaVisible,
|
||||||
|
myVisa,
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|||||||
214
tests/unit/applicant-my-logistics.test.ts
Normal file
214
tests/unit/applicant-my-logistics.test.ts
Normal file
@@ -0,0 +1,214 @@
|
|||||||
|
import { afterAll, describe, expect, it } from 'vitest'
|
||||||
|
import { prisma, createCaller } from '../setup'
|
||||||
|
import {
|
||||||
|
createTestProgram,
|
||||||
|
createTestProject,
|
||||||
|
cleanupTestData,
|
||||||
|
uid,
|
||||||
|
} from '../helpers'
|
||||||
|
import { applicantRouter } from '../../src/server/routers/applicant'
|
||||||
|
|
||||||
|
// ─── shared helpers ────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
async function createApplicant() {
|
||||||
|
const id = uid('user')
|
||||||
|
return prisma.user.create({
|
||||||
|
data: {
|
||||||
|
id,
|
||||||
|
email: `${id}@test.local`,
|
||||||
|
name: 'Test Applicant',
|
||||||
|
role: 'APPLICANT',
|
||||||
|
roles: ['APPLICANT'],
|
||||||
|
status: 'ACTIVE',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a fully-confirmed finalist setup:
|
||||||
|
* - Program with visaStatusVisibleToMembers=true
|
||||||
|
* - Project with a TeamMember (LEAD)
|
||||||
|
* - CONFIRMED FinalistConfirmation
|
||||||
|
* - Hotel for the program
|
||||||
|
* - AttendingMember for the caller
|
||||||
|
* - FlightDetail for that attendee
|
||||||
|
* - VisaApplication GRANTED for that attendee
|
||||||
|
*/
|
||||||
|
async function buildConfirmedFinalist() {
|
||||||
|
const program = await createTestProgram({
|
||||||
|
name: `logistics-${uid()}`,
|
||||||
|
defaultAttendeeCap: 3,
|
||||||
|
})
|
||||||
|
await prisma.program.update({
|
||||||
|
where: { id: program.id },
|
||||||
|
data: { visaStatusVisibleToMembers: true },
|
||||||
|
})
|
||||||
|
|
||||||
|
const project = await createTestProject(program.id, {
|
||||||
|
title: 'Ocean Wave Project',
|
||||||
|
competitionCategory: 'STARTUP',
|
||||||
|
})
|
||||||
|
|
||||||
|
const user = await createApplicant()
|
||||||
|
|
||||||
|
await prisma.teamMember.create({
|
||||||
|
data: { projectId: project.id, userId: user.id, role: 'LEAD' },
|
||||||
|
})
|
||||||
|
|
||||||
|
const confirmation = await prisma.finalistConfirmation.create({
|
||||||
|
data: {
|
||||||
|
projectId: project.id,
|
||||||
|
category: 'STARTUP',
|
||||||
|
status: 'CONFIRMED',
|
||||||
|
deadline: new Date(Date.now() + 86_400_000),
|
||||||
|
token: `tok_${uid()}`,
|
||||||
|
confirmedAt: new Date(),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// Hotel for the program (1:1)
|
||||||
|
await prisma.hotel.create({
|
||||||
|
data: {
|
||||||
|
programId: program.id,
|
||||||
|
name: 'Hotel Hermitage',
|
||||||
|
address: '11 Avenue de la Costa, Monaco',
|
||||||
|
link: 'https://example.com/hotel',
|
||||||
|
notes: 'Breakfast included.',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const attendee = await prisma.attendingMember.create({
|
||||||
|
data: {
|
||||||
|
confirmationId: confirmation.id,
|
||||||
|
userId: user.id,
|
||||||
|
needsVisa: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// FlightDetail
|
||||||
|
await prisma.flightDetail.create({
|
||||||
|
data: {
|
||||||
|
attendingMemberId: attendee.id,
|
||||||
|
arrivalFlightNumber: 'AF1234',
|
||||||
|
arrivalAirport: 'NCE',
|
||||||
|
arrivalAt: new Date('2026-06-20T10:00:00Z'),
|
||||||
|
departureFlightNumber: 'AF5678',
|
||||||
|
departureAirport: 'NCE',
|
||||||
|
departureAt: new Date('2026-06-23T15:00:00Z'),
|
||||||
|
status: 'CONFIRMED',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// VisaApplication GRANTED
|
||||||
|
await prisma.visaApplication.create({
|
||||||
|
data: {
|
||||||
|
attendingMemberId: attendee.id,
|
||||||
|
status: 'GRANTED',
|
||||||
|
nationality: 'French',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return { program, project, user, confirmation, attendee }
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── Task 1 tests ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
describe('applicant.getMyLogistics', () => {
|
||||||
|
const programIds: string[] = []
|
||||||
|
const userIds: string[] = []
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
for (const programId of programIds) {
|
||||||
|
await prisma.visaApplication.deleteMany({
|
||||||
|
where: { attendingMember: { confirmation: { project: { programId } } } },
|
||||||
|
})
|
||||||
|
await prisma.flightDetail.deleteMany({
|
||||||
|
where: { attendingMember: { confirmation: { project: { programId } } } },
|
||||||
|
})
|
||||||
|
await prisma.hotel.deleteMany({ where: { programId } })
|
||||||
|
await prisma.attendingMember.deleteMany({
|
||||||
|
where: { confirmation: { project: { programId } } },
|
||||||
|
})
|
||||||
|
await prisma.finalistConfirmation.deleteMany({ where: { project: { programId } } })
|
||||||
|
await cleanupTestData(programId, [])
|
||||||
|
}
|
||||||
|
if (userIds.length > 0) {
|
||||||
|
await prisma.user.deleteMany({ where: { id: { in: userIds } } })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns hotel, flight, and visa for a confirmed finalist', async () => {
|
||||||
|
const { program, user } = await buildConfirmedFinalist()
|
||||||
|
programIds.push(program.id)
|
||||||
|
userIds.push(user.id)
|
||||||
|
|
||||||
|
const caller = createCaller(applicantRouter, {
|
||||||
|
id: user.id,
|
||||||
|
email: user.email,
|
||||||
|
role: 'APPLICANT',
|
||||||
|
})
|
||||||
|
|
||||||
|
const result = await caller.getMyLogistics()
|
||||||
|
expect(result).not.toBeNull()
|
||||||
|
expect(result!.projectTitle).toBe('Ocean Wave Project')
|
||||||
|
expect(result!.confirmationStatus).toBe('CONFIRMED')
|
||||||
|
expect(result!.hotel).not.toBeNull()
|
||||||
|
expect(result!.hotel!.name).toBe('Hotel Hermitage')
|
||||||
|
expect(result!.myFlight).not.toBeNull()
|
||||||
|
expect(result!.myFlight!.arrivalFlightNumber).toBe('AF1234')
|
||||||
|
expect(result!.myFlight!.arrivalAirport).toBe('NCE')
|
||||||
|
expect(result!.visaVisible).toBe(true)
|
||||||
|
expect(result!.myVisa).not.toBeNull()
|
||||||
|
expect(result!.myVisa!.status).toBe('GRANTED')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns null for a user who has no confirmed finalist project', async () => {
|
||||||
|
// user with no project at all
|
||||||
|
const nonFinalist = await createApplicant()
|
||||||
|
userIds.push(nonFinalist.id)
|
||||||
|
|
||||||
|
const caller = createCaller(applicantRouter, {
|
||||||
|
id: nonFinalist.id,
|
||||||
|
email: nonFinalist.email,
|
||||||
|
role: 'APPLICANT',
|
||||||
|
})
|
||||||
|
|
||||||
|
const result = await caller.getMyLogistics()
|
||||||
|
expect(result).toBeNull()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('returns null when the confirmation is PENDING (not CONFIRMED)', async () => {
|
||||||
|
const program = await createTestProgram({
|
||||||
|
name: `logistics-pending-${uid()}`,
|
||||||
|
defaultAttendeeCap: 2,
|
||||||
|
})
|
||||||
|
programIds.push(program.id)
|
||||||
|
const project = await createTestProject(program.id, {
|
||||||
|
title: 'Pending Project',
|
||||||
|
competitionCategory: 'STARTUP',
|
||||||
|
})
|
||||||
|
const user = await createApplicant()
|
||||||
|
userIds.push(user.id)
|
||||||
|
await prisma.teamMember.create({
|
||||||
|
data: { projectId: project.id, userId: user.id, role: 'LEAD' },
|
||||||
|
})
|
||||||
|
await prisma.finalistConfirmation.create({
|
||||||
|
data: {
|
||||||
|
projectId: project.id,
|
||||||
|
category: 'STARTUP',
|
||||||
|
status: 'PENDING',
|
||||||
|
deadline: new Date(Date.now() + 86_400_000),
|
||||||
|
token: `tok_${uid()}`,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const caller = createCaller(applicantRouter, {
|
||||||
|
id: user.id,
|
||||||
|
email: user.email,
|
||||||
|
role: 'APPLICANT',
|
||||||
|
})
|
||||||
|
|
||||||
|
const result = await caller.getMyLogistics()
|
||||||
|
expect(result).toBeNull()
|
||||||
|
})
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user