2026-04-26 13:48:06 +02:00
|
|
|
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>',
|
|
|
|
|
};
|
|
|
|
|
|
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
|
|
|
describe('createTemplateSchema - mergeFields allow-list', () => {
|
2026-04-26 13:48:06 +02:00
|
|
|
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');
|
|
|
|
|
});
|
|
|
|
|
});
|