Files
pn-new-crm/tests/unit/validators/document-templates.test.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

62 lines
2.0 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import { createTemplateSchema } from '@/lib/validators/document-templates';
const baseInput = {
name: 'Tmpl',
templateType: 'custom' as const,
bodyHtml: '<p>x</p>',
};
describe('createTemplateSchema - mergeFields allow-list', () => {
it('accepts valid tokens from the catalog', () => {
const parsed = createTemplateSchema.parse({
...baseInput,
mergeFields: ['{{client.fullName}}', '{{yacht.name}}', '{{berth.mooringNumber}}'],
});
expect(parsed.mergeFields).toEqual([
'{{client.fullName}}',
'{{yacht.name}}',
'{{berth.mooringNumber}}',
]);
});
it('rejects deprecated tokens that lived on `clients` before the refactor', () => {
const result = createTemplateSchema.safeParse({
...baseInput,
mergeFields: ['{{client.yachtName}}'],
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.issues[0]?.message).toMatch(/Unknown merge tokens.*client\.yachtName/);
}
});
it('rejects unknown tokens with a helpful message listing them', () => {
const result = createTemplateSchema.safeParse({
...baseInput,
mergeFields: ['{{client.fullName}}', '{{not.a.token}}', '{{also.bogus}}'],
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.issues[0]?.message).toContain('not.a.token');
expect(result.error.issues[0]?.message).toContain('also.bogus');
}
});
it('defaults mergeFields to an empty array when omitted', () => {
const parsed = createTemplateSchema.parse(baseInput);
expect(parsed.mergeFields).toEqual([]);
});
it('accepts an empty mergeFields array', () => {
const parsed = createTemplateSchema.parse({ ...baseInput, mergeFields: [] });
expect(parsed.mergeFields).toEqual([]);
});
it('allows the new `eoi` templateType', () => {
const parsed = createTemplateSchema.parse({ ...baseInput, templateType: 'eoi' });
expect(parsed.templateType).toBe('eoi');
});
});