Files
pn-new-crm/src/lib/validators/yachts.ts
Matt eaab14943b feat(post-audit): Phase 3 EOI overrides + 3c spawn + 3d promote + Phase 4 worker
Phase 3b — EOI dialog field overrides:
- New EoiOverridesInput shape (clientEmail / clientPhone / yachtName)
  threaded through generate-and-sign validator + both pathways
  (in-app pdf-lib fill, Documenso template generate).
- src/lib/services/eoi-overrides.service.ts applies side-effects in one
  transaction: useOnlyForThisEoi writes documents.override_* and stops;
  setAsDefault demotes the prior primary + promotes (existing contactId)
  or inserts + promotes (fresh value); neither flag inserts a non-primary
  client_contacts row for future dropdown reuse.
- Document override columns persisted post-insert, with a 1-minute
  source_document_id backfill on freshly inserted contact rows.
- eoi-context route returns available.{emails, phones} so the dialog
  can render combobox options.
- <OverridableContactField> in eoi-generate-dialog.tsx renders the
  combobox + manual input + 2 checkboxes per field with mutually
  exclusive intent semantics.

Phase 3c — yacht spawn from EOI dialog:
- YachtForm gains createExtras + onCreated callbacks; the EOI dialog
  opens it as a nested Sheet pre-filled with the linked client as owner.
  On save the new yacht is stamped source='eoi-generated' and the
  interest is PATCHed with the new yachtId so the EOI context reflows.

Phase 3d — promote-to-primary + audit + [EOI] badge:
- POST /api/v1/clients/:id/contacts/:contactId/promote-to-primary
  (transactional demote+promote via promoteContactToPrimary).
- src/lib/audit.ts AuditAction type adds eoi_field_override,
  promote_to_primary, eoi_spawn_yacht (DB column is free-text).
- ContactsEditor surfaces an [EOI] badge on non-primary rows where
  source='eoi-custom-input'.

Phase 4 — worker + TOD picker:
- processOverdueReminders refactored to UPDATE...RETURNING with a
  fired_at IS NULL gate so parallel workers can't double-fire. Uses
  the idx_reminders_due_unfired partial index from migration 0072.
- /settings gets a "Default reminder time" time-of-day picker; the
  value lands in user_profiles.preferences.digestTimeOfDay (validated
  HH:MM at the route). <ReminderForm> seeds its dueAt from this
  preference via a React-Query me-prefs fetch.

Phase 6 hardening:
- IMAP bounce poller strips whitespace from IMAP_PASS so a copy-paste
  of Google Workspace's 16-char App Password formatted as
  "abcd efgh ijkl mnop" still authenticates. Workspace activation
  procedure documented in MASTER-PLAN §Phase 6 (was previously written
  to CLAUDE.md, which was bloat — moved to the plan).

Quality gates: 1374/1374 vitest, tsc clean, lint 0 errors.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-18 16:18:03 +02:00

67 lines
2.6 KiB
TypeScript

import { z } from 'zod';
import { baseListQuerySchema } from '@/lib/api/list-query';
export const ownerRefSchema = z.object({
type: z.enum(['client', 'company']),
id: z.string().min(1),
});
// Numeric columns on the yachts table accept a stringified decimal or
// null. The form posts empty strings for unfilled fields, which Postgres
// rejects with `invalid input syntax for type numeric: ""`. Strip empty
// strings here so the service can confidently `?? null` them.
const optionalNumericString = z
.string()
.optional()
.transform((v) => (v === '' || v === undefined ? undefined : v));
export const createYachtSchema = z.object({
name: z.string().min(1).max(200),
hullNumber: z.string().optional(),
registration: z.string().optional(),
flag: z.string().optional(),
yearBuilt: z.number().int().min(1800).max(2100).optional(),
builder: z.string().optional(),
model: z.string().optional(),
hullMaterial: z.string().optional(),
lengthFt: optionalNumericString,
widthFt: optionalNumericString,
draftFt: optionalNumericString,
lengthM: optionalNumericString,
widthM: optionalNumericString,
draftM: optionalNumericString,
owner: ownerRefSchema, // required; yacht must have an owner
status: z.enum(['active', 'retired', 'sold_away']).optional().default('active'),
notes: z.string().optional(),
tagIds: z.array(z.string()).optional().default([]),
// Phase 3c — origin tracking. Defaults to 'manual'; the EOI spawn flow
// sends 'eoi-generated' and the migration-0073 CHECK enforces the
// values at the DB level.
source: z.enum(['manual', 'imported', 'eoi-generated']).optional(),
sourceDocumentId: z.string().uuid().optional(),
});
export const updateYachtSchema = createYachtSchema.partial().omit({ owner: true });
// Owner changes go through /transfer, not PATCH.
export const transferOwnershipSchema = z.object({
newOwner: ownerRefSchema,
effectiveDate: z.coerce.date(),
transferReason: z
.enum(['sale', 'inheritance', 'gift', 'company_restructure', 'other'])
.optional(),
transferNotes: z.string().optional(),
});
export const listYachtsSchema = baseListQuerySchema.extend({
ownerType: z.enum(['client', 'company']).optional(),
ownerId: z.string().optional(),
status: z.enum(['active', 'retired', 'sold_away']).optional(),
search: z.string().optional(),
});
export type CreateYachtInput = z.infer<typeof createYachtSchema>;
export type UpdateYachtInput = z.infer<typeof updateYachtSchema>;
export type TransferOwnershipInput = z.infer<typeof transferOwnershipSchema>;
export type ListYachtsInput = z.infer<typeof listYachtsSchema>;