feat(applicant): self-service visa nationality entry
Add updateMyVisaNationality mutation: finds the caller's AttendingMember where the program has visaStatusVisibleToMembers=true and a VisaApplication exists, updates VisaApplication.nationality, and emits a VISA_NATIONALITY_SELF_SET audit log. Throws NOT_FOUND when no eligible application exists. Tests: persists update; rejects caller without a visible visa app. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -2937,4 +2937,51 @@ export const applicantRouter = router({
|
|||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows the authenticated user to self-declare their passport nationality
|
||||||
|
* on their own VisaApplication when visaStatusVisibleToMembers is true.
|
||||||
|
*/
|
||||||
|
updateMyVisaNationality: protectedProcedure
|
||||||
|
.input(z.object({ nationality: z.string().max(100) }))
|
||||||
|
.mutation(async ({ ctx, input }) => {
|
||||||
|
// Find the caller's AttendingMember whose program has visaStatusVisibleToMembers=true
|
||||||
|
// and which has a visaApplication.
|
||||||
|
const attendee = await ctx.prisma.attendingMember.findFirst({
|
||||||
|
where: {
|
||||||
|
userId: ctx.user.id,
|
||||||
|
confirmation: {
|
||||||
|
project: {
|
||||||
|
program: { visaStatusVisibleToMembers: true },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
visaApplication: { isNot: null },
|
||||||
|
},
|
||||||
|
include: { visaApplication: true },
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!attendee || !attendee.visaApplication) {
|
||||||
|
throw new TRPCError({
|
||||||
|
code: 'NOT_FOUND',
|
||||||
|
message: 'No visa application to update',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const updated = await ctx.prisma.visaApplication.update({
|
||||||
|
where: { id: attendee.visaApplication.id },
|
||||||
|
data: { nationality: input.nationality },
|
||||||
|
})
|
||||||
|
|
||||||
|
await logAudit({
|
||||||
|
prisma: ctx.prisma,
|
||||||
|
userId: ctx.user.id,
|
||||||
|
action: 'VISA_NATIONALITY_SELF_SET',
|
||||||
|
entityType: 'VisaApplication',
|
||||||
|
entityId: updated.id,
|
||||||
|
detailsJson: { nationality: input.nationality },
|
||||||
|
ipAddress: ctx.ip,
|
||||||
|
userAgent: ctx.userAgent,
|
||||||
|
})
|
||||||
|
|
||||||
|
return { ok: true }
|
||||||
|
}),
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -212,3 +212,65 @@ describe('applicant.getMyLogistics', () => {
|
|||||||
expect(result).toBeNull()
|
expect(result).toBeNull()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// ─── Task 2 tests ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
describe('applicant.updateMyVisaNationality', () => {
|
||||||
|
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('updates VisaApplication.nationality and persists it', async () => {
|
||||||
|
const { program, user, attendee } = 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.updateMyVisaNationality({ nationality: 'Kenya' })
|
||||||
|
expect(result).toEqual({ ok: true })
|
||||||
|
|
||||||
|
const updated = await prisma.visaApplication.findUnique({
|
||||||
|
where: { attendingMemberId: attendee.id },
|
||||||
|
})
|
||||||
|
expect(updated!.nationality).toBe('Kenya')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws NOT_FOUND when the caller has no visible visa application', async () => {
|
||||||
|
const nonVisa = await createApplicant()
|
||||||
|
userIds.push(nonVisa.id)
|
||||||
|
|
||||||
|
const caller = createCaller(applicantRouter, {
|
||||||
|
id: nonVisa.id,
|
||||||
|
email: nonVisa.email,
|
||||||
|
role: 'APPLICANT',
|
||||||
|
})
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
caller.updateMyVisaNationality({ nationality: 'Kenya' }),
|
||||||
|
).rejects.toMatchObject({ code: 'NOT_FOUND' })
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|||||||
Reference in New Issue
Block a user