73-file atomic rename per docs/tenancies-design.md:
- Migration 0085: rename table + indexes + FK constraints; rename
documents.reservation_id → tenancy_id; migrate jsonb permission maps
(reservations resource → tenancies; collapse create+activate → manage);
rewrite historical audit_logs.entity_type='berth_reservation' →
'berth_tenancy'. FK renames wrapped in DO blocks so dev DBs that pre-date
the FK additions don't abort.
- Schema: berthReservations → berthTenancies; BerthReservation type →
BerthTenancy; indexes idx_br_* / idx_brr_* → idx_bt_*.
- RolePermissions: resource { view, create, activate, cancel } collapses to
{ view, manage, cancel }; all 8 default seed bundles + role-form + matrix
updated.
- Service: berth-reservations.service.ts → berth-tenancies.service.ts;
endReservation → endTenancy; listReservations → listTenancies.
- API: /api/v1/berth-reservations → /api/v1/tenancies (+ nested [id]);
/api/v1/berths/[id]/reservations → /api/v1/berths/[id]/tenancies.
- Validators: reservations.ts → tenancies.ts; RESERVATION_STATUSES →
TENANCY_STATUSES; endReservationSchema → endTenancySchema.
- Routes: /{portSlug}/berth-reservations → /{portSlug}/tenancies;
/portal/my-reservations → /portal/my-tenancies.
- Components: src/components/reservations/* → src/components/tenancies/*;
BerthReservationsTab → BerthTenanciesTab; ClientReservationsTab →
ClientTenanciesTab; ReservationList → TenancyList.
- Socket events: berth_reservation:* → berth_tenancy:*; payload
reservationId → tenancyId.
- Webhook events: berth_reservation.* → berth_tenancy.*.
- Portal: getPortalUserReservations → getPortalUserTenancies;
PortalReservation → PortalTenancy; PortalDashboard.counts.activeReservations
→ activeTenancies; PortalNav label "Reservations" → "Tenancies".
- Dossier: DossierReservation → DossierTenancy; reservationDecisions →
tenancyDecisions across smart-archive-dialog + bulk-archive routes.
- Documents schema: documents.reservationId → documents.tenancyId
(TS + DB column + index + FK constraint).
- Activity feed label berth_reservation → berth_tenancy (matched against
migrated historical audit rows).
KEPT (separate concepts):
- Reservation Agreement document type (the contract sent to clients).
- "Reservation" pipeline stage name.
- {{reservation.*}} merge tokens in template authoring.
- interest.reservationStatus / reservationDocStatus / dateReservationSent
fields (track agreement signing on the deal).
- reservation-agreement-context.ts service (builds merge context for the
Reservation Agreement doc; only its DB imports were renamed).
Verified: tsc clean, 1480/1480 vitest passing, migration applied.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
129 lines
4.4 KiB
TypeScript
129 lines
4.4 KiB
TypeScript
import { z } from 'zod';
|
|
|
|
import { baseListQuerySchema } from '@/lib/api/list-query';
|
|
import { DOCUMENT_TYPES, DOCUMENT_STATUSES } from '@/lib/constants';
|
|
|
|
export const createDocumentSchema = z.object({
|
|
interestId: z.string().optional(),
|
|
clientId: z.string().optional(),
|
|
folderId: z.string().nullable().optional(),
|
|
documentType: z.enum(DOCUMENT_TYPES),
|
|
title: z.string().min(1).max(200),
|
|
notes: z.string().optional(),
|
|
});
|
|
|
|
export const updateDocumentSchema = z.object({
|
|
title: z.string().min(1).max(200).optional(),
|
|
notes: z.string().optional(),
|
|
status: z.enum(DOCUMENT_STATUSES).optional(),
|
|
});
|
|
|
|
const wizardSignerSchema = z.object({
|
|
signerName: z.string().min(1),
|
|
signerEmail: z.string().email(),
|
|
signerRole: z.enum(['client', 'sales', 'approver', 'developer', 'other']),
|
|
signingOrder: z.number().int().min(1),
|
|
});
|
|
|
|
export const createDocumentWizardSchema = z
|
|
.object({
|
|
source: z.enum(['template', 'upload']).default('template'),
|
|
templateId: z.string().optional(),
|
|
uploadedFileId: z.string().optional(),
|
|
|
|
documentType: z.enum(DOCUMENT_TYPES),
|
|
title: z.string().min(1).max(200),
|
|
notes: z.string().optional(),
|
|
|
|
interestId: z.string().optional(),
|
|
tenancyId: z.string().optional(),
|
|
clientId: z.string().optional(),
|
|
companyId: z.string().optional(),
|
|
yachtId: z.string().optional(),
|
|
|
|
signers: z.array(wizardSignerSchema).optional(),
|
|
signingMode: z.enum(['sequential', 'parallel']).default('sequential'),
|
|
pathway: z.enum(['documenso-template', 'inapp', 'upload']).default('documenso-template'),
|
|
|
|
watchers: z.array(z.string()).default([]),
|
|
|
|
reminderCadenceOverride: z.number().int().min(1).max(365).nullable().optional(),
|
|
remindersDisabled: z.boolean().default(false),
|
|
|
|
autoPlaceFields: z.boolean().default(true),
|
|
sendImmediately: z.boolean().default(true),
|
|
})
|
|
.refine(
|
|
(d) =>
|
|
[d.interestId, d.tenancyId, d.clientId, d.companyId, d.yachtId].filter(Boolean).length === 1,
|
|
{ message: 'Exactly one subject (interest/tenancy/client/company/yacht) is required' },
|
|
)
|
|
.refine((d) => d.source !== 'template' || Boolean(d.templateId), {
|
|
path: ['templateId'],
|
|
message: 'templateId is required when source=template',
|
|
})
|
|
.refine((d) => d.source !== 'upload' || Boolean(d.uploadedFileId), {
|
|
path: ['uploadedFileId'],
|
|
message: 'uploadedFileId is required when source=upload',
|
|
});
|
|
|
|
export type CreateDocumentWizardInput = z.infer<typeof createDocumentWizardSchema>;
|
|
|
|
export const documentsHubTabs = [
|
|
'all',
|
|
'in_progress',
|
|
'eoi_queue',
|
|
'awaiting_them',
|
|
'awaiting_me',
|
|
'completed',
|
|
'expired',
|
|
] as const;
|
|
export type DocumentsHubTab = (typeof documentsHubTabs)[number];
|
|
|
|
export const listDocumentsSchema = baseListQuerySchema
|
|
.extend({
|
|
interestId: z.string().optional(),
|
|
clientId: z.string().optional(),
|
|
documentType: z.string().optional(),
|
|
folderId: z
|
|
.string()
|
|
.nullable()
|
|
.transform((v) => (v === '' ? null : v))
|
|
.optional(),
|
|
includeDescendants: z.coerce.boolean().optional(),
|
|
status: z.string().optional(),
|
|
/** Hub tab filter - applies tab-specific status / signer-membership constraints. */
|
|
tab: z.enum(documentsHubTabs).optional(),
|
|
/** Restrict to docs being watched by this user id. */
|
|
watcherUserId: z.string().optional(),
|
|
/** When true, only docs intended for signing (default true on hub). */
|
|
signatureOnly: z
|
|
.enum(['true', 'false'])
|
|
.transform((v) => v === 'true')
|
|
.optional(),
|
|
sentSince: z.string().datetime().optional(),
|
|
sentUntil: z.string().datetime().optional(),
|
|
/** Entity-aggregated projection params - mutually exclusive with folderId. */
|
|
entityType: z.enum(['client', 'company', 'yacht']).optional(),
|
|
entityId: z.string().uuid().optional(),
|
|
})
|
|
.refine(
|
|
(q) => !(q.folderId !== undefined && (q.entityType !== undefined || q.entityId !== undefined)),
|
|
{
|
|
message: 'folderId is mutually exclusive with entityType/entityId',
|
|
path: ['folderId'],
|
|
},
|
|
)
|
|
.refine((q) => Boolean(q.entityType) === Boolean(q.entityId), {
|
|
message: 'entityType and entityId must be provided together',
|
|
path: ['entityType'],
|
|
});
|
|
|
|
export const uploadSignedSchema = z.object({
|
|
documentId: z.string().min(1),
|
|
});
|
|
|
|
export type CreateDocumentInput = z.infer<typeof createDocumentSchema>;
|
|
export type UpdateDocumentInput = z.infer<typeof updateDocumentSchema>;
|
|
export type ListDocumentsInput = z.infer<typeof listDocumentsSchema>;
|