Files
pn-new-crm/src/lib/db/schema/interest-field-history.ts
Matt 221ae5784e chore(autonomous-session): consolidate uncommitted work from prior session
Bundles the prior autonomous-session output that was sitting unstaged:

- Em-dash sweep across src/ + tests/ (en-dash/em-dash to hyphen, ~2280 instances)
- country-flag-icons rollout (CountryFlag component, replaces emoji glyphs that
  never rendered on Windows; lazy-loads the 3x2 SVG index as a single chunk
  after the per-subpath dynamic-import approach silently failed in webpack)
- Admin IA Phase 1+2: 7-domain regroup, 41 to 38 pages, /admin/berths index,
  redirects (ocr to ai, reports to dashboard, invitations to users),
  docs/admin-ia-proposal.md
- Per-template email tester (registry + endpoint + UI on Email admin page)
- Cancel-document mode picker (delete-from-Documenso vs keep-for-audit)
- Dashboard PDF report: 25 widgets, SVG charts, date-range picker, 11 resolvers
- Customize-widgets per-region sortables at xl+ (charts/rails/feed); single
  flat sortable below xl when the layout stacks; per-viewport saved orders
- Audit doc updates capturing each shipped item
- Lint fixes: react-compiler immutability in DonutChart (reduce instead of
  let-reassign), set-state-in-effect disables in CountryFlag and
  UploadForSigning preview-bytes effect, unused 'confirm' destructures in
  interest contract + reservation tabs, unescaped apostrophe in test-template
  card copy
2026-05-23 00:52:59 +02:00

64 lines
2.7 KiB
TypeScript

/**
* Field-level override history for interests + their linked clients.
*
* Every time a field on an interest or its linked client is overridden
* via an explicit channel (today: supplemental-info form submission;
* future: form-templates, AI-assisted extraction acceptance), a row
* lands here. Distinct from `audit_logs` - that table tracks every
* CRUD event for compliance; this one tracks only deliberate overrides
* so the Interest + Client "Field history" panels can surface them
* compactly.
*/
import { pgTable, text, jsonb, timestamp, index } from 'drizzle-orm/pg-core';
import { sql } from 'drizzle-orm';
import { ports } from './ports';
import { interests } from './interests';
import { clients } from './clients';
export const interestFieldHistory = pgTable(
'interest_field_history',
{
id: text('id')
.primaryKey()
.$defaultFn(() => crypto.randomUUID()),
portId: text('port_id')
.notNull()
.references(() => ports.id),
interestId: text('interest_id').references(() => interests.id, { onDelete: 'cascade' }),
/** Denormalized for fast lookup on the Client detail "Field history"
* panel - overrides that come in via a supplemental-info form
* carry both interest + client refs. Direct-edit overrides may
* only carry one. */
clientId: text('client_id').references(() => clients.id, { onDelete: 'cascade' }),
/** Dotted path of the field that was overridden. Examples:
* 'client.fullName'
* 'interest.desiredLengthFt'
* 'client.address.streetAddress'
* The Field history panel formats this for display. */
fieldPath: text('field_path').notNull(),
oldValue: jsonb('old_value'),
newValue: jsonb('new_value').notNull(),
/** Provenance: 'supplemental_form' | 'rep_edit' | 'system_inferred' |
* 'ai_extraction' (future). Drives the "Submitted via X" copy. */
source: text('source').notNull(),
/** Optional FK to the form_submissions row that triggered the
* override. Lets the UI link back to the original submission. */
submissionId: text('submission_id'),
createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),
createdBy: text('created_by'),
},
(table) => [
index('idx_ifh_interest_created')
.on(table.portId, table.interestId, table.createdAt)
.where(sql`${table.interestId} IS NOT NULL`),
index('idx_ifh_client_created')
.on(table.portId, table.clientId, table.createdAt)
.where(sql`${table.clientId} IS NOT NULL`),
],
);
export type InterestFieldHistory = typeof interestFieldHistory.$inferSelect;
export type NewInterestFieldHistory = typeof interestFieldHistory.$inferInsert;