First slice of Phase 5 (verification + appraisal). The office queue for verifying site-entered records, per Crewing-Implementation-Spec §8.11/R11. Stacks on 4c. Behind NEXT_PUBLIC_CREWING_ENABLED. What's in - Schema: CrewActionType += RECORD_VERIFIED/RECORD_REJECTED (migration crewing_verification_actions). No model changes — SeafarerDocument/BankDetail/ EpfDetail already carry verificationStatus + verifiedById (3b/4a). - Actions (crewing/verification/actions.ts): verifyDocument (verify_site_records — MPO/Manager) and verifyBankEpf (verify_bank_epf — Accounts) set verificationStatus VERIFIED/REJECTED + verifiedById; rejection requires remarks; each writes a CrewAction. Already-decided records are guarded. - Screen: /crewing/verification — role-aware (MPO: pending documents with expiry flags; Accounts: pending bank/EPF), Verify / Reject-with-remarks. Leave is not here (Manager approval, R11). Verification added to nav (MPO + Accounts + SU, §7). Tests & docs - Integration: verification.test.ts (6) — doc verify/reject + already-decided guard, bank/EPF verify, permission gating (Accounts can't verify docs, MPO can't verify bank/EPF). type-check clean; full unit (241) + integration (201) green (verified with RESEND_API_KEY unset, mimicking CI). - CLAUDE.md updated. Deferred (per decision): PPE / next-of-kin verification gates. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
64 lines
2.5 KiB
TypeScript
64 lines
2.5 KiB
TypeScript
import { auth } from "@/auth";
|
|
import { db } from "@/lib/db";
|
|
import { hasPermission } from "@/lib/permissions";
|
|
import { CREWING_ENABLED } from "@/lib/feature-flags";
|
|
import { redirect, notFound } from "next/navigation";
|
|
import { VerificationManager } from "./verification-manager";
|
|
import type { Metadata } from "next";
|
|
|
|
export const metadata: Metadata = { title: "Verification" };
|
|
|
|
export default async function VerificationPage() {
|
|
if (!CREWING_ENABLED) notFound();
|
|
|
|
const session = await auth();
|
|
if (!session?.user) redirect("/login");
|
|
const role = session.user.role;
|
|
const canDocs = hasPermission(role, "verify_site_records");
|
|
const canBankEpf = hasPermission(role, "verify_bank_epf");
|
|
if (!canDocs && !canBankEpf) redirect("/dashboard");
|
|
|
|
const [docs, bank, epf] = await Promise.all([
|
|
canDocs
|
|
? db.seafarerDocument.findMany({
|
|
where: { verificationStatus: "PENDING" },
|
|
orderBy: { createdAt: "asc" },
|
|
include: {
|
|
crewMember: {
|
|
select: {
|
|
name: true,
|
|
assignments: { where: { status: { not: "SIGNED_OFF" } }, take: 1, include: { vessel: { select: { name: true } }, site: { select: { name: true } } } },
|
|
},
|
|
},
|
|
},
|
|
})
|
|
: [],
|
|
canBankEpf
|
|
? db.bankDetail.findMany({ where: { verificationStatus: "PENDING" }, orderBy: { createdAt: "asc" }, include: { crewMember: { select: { name: true } } } })
|
|
: [],
|
|
canBankEpf
|
|
? db.epfDetail.findMany({ where: { verificationStatus: "PENDING" }, orderBy: { createdAt: "asc" }, include: { crewMember: { select: { name: true } } } })
|
|
: [],
|
|
]);
|
|
|
|
return (
|
|
<VerificationManager
|
|
docs={docs.map((d) => {
|
|
const a = d.crewMember.assignments[0];
|
|
return {
|
|
id: d.id,
|
|
crewName: d.crewMember.name,
|
|
location: a?.vessel?.name ?? a?.site?.name ?? "—",
|
|
docType: d.docType,
|
|
number: d.number,
|
|
expiryDate: d.expiryDate?.toISOString() ?? null,
|
|
submitted: d.createdAt.toISOString(),
|
|
};
|
|
})}
|
|
bank={bank.map((b) => ({ crewMemberId: b.crewMemberId, crewName: b.crewMember.name, accountName: b.accountName, accountNumber: b.accountNumber, ifsc: b.ifsc, bankName: b.bankName }))}
|
|
epf={epf.map((e) => ({ crewMemberId: e.crewMemberId, crewName: e.crewMember.name, uan: e.uan, aadhaarLast4: e.aadhaarLast4, pfNumber: e.pfNumber }))}
|
|
canDocs={canDocs}
|
|
canBankEpf={canBankEpf}
|
|
/>
|
|
);
|
|
}
|