feat(validators): adopt drizzle-zod for tags + brochures schemas

Pilot adoption of `drizzle-zod` (already shipped as part of `drizzle-orm`).
Two CRUD-shape validators migrate from hand-written z.object() to
`createInsertSchema(table, refinements)`:

- tags: name + color (with hex regex refinement).
- brochures: label + description + isDefault.

Both schemas now derive directly from the Drizzle table definition.
Adding a column to the table will auto-include it in the validator
(filtered via `.pick(...)` where API surface should stay narrower than
the table). Eliminates the validator-drift class of bugs the audit
flagged (e.g. adding a column to clients but forgetting to add it to
createClientSchema).

Pattern is established for future validator touches. Migrating the
remaining CRUD validators is opportunistic — done when the validator
file is otherwise being edited.

Verified: tsc clean, vitest 1293/1293 pass.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-05-12 18:30:58 +02:00
parent acf878f997
commit a7a008c62e
2 changed files with 23 additions and 18 deletions

View File

@@ -1,16 +1,19 @@
import { z } from 'zod';
import { createInsertSchema, createUpdateSchema } from 'drizzle-zod';
export const createBrochureSchema = z.object({
label: z.string().trim().min(1).max(120),
description: z.string().max(500).optional().nullable(),
isDefault: z.boolean().optional(),
});
import { brochures } from '@/lib/db/schema/brochures';
export const updateBrochureSchema = z.object({
label: z.string().trim().min(1).max(120).optional(),
description: z.string().max(500).optional().nullable(),
isDefault: z.boolean().optional(),
});
// Derived from the Drizzle table — adding a column to `brochures`
// auto-includes it here. Refinements override per-field.
export const createBrochureSchema = createInsertSchema(brochures, {
label: (s) => s.trim().min(1).max(120),
description: (s) => s.max(500),
}).pick({ label: true, description: true, isDefault: true });
export const updateBrochureSchema = createUpdateSchema(brochures, {
label: (s) => s.trim().min(1).max(120),
description: (s) => s.max(500),
}).pick({ label: true, description: true, isDefault: true });
export const registerBrochureVersionSchema = z.object({
storageKey: z

View File

@@ -1,13 +1,15 @@
import { z } from 'zod';
import { createInsertSchema } from 'drizzle-zod';
export const createTagSchema = z.object({
name: z.string().min(1).max(50),
color: z
.string()
.regex(/^#[0-9A-Fa-f]{6}$/, 'Color must be a valid hex color (e.g. #6B7280)')
.optional()
.default('#6B7280'),
});
import { tags } from '@/lib/db/schema/system';
// Derive the schema from the Drizzle table — adding a column to `tags`
// auto-includes it here; omitting / refining is per-field. Eliminates
// the validator-drift class of bugs the audit flagged.
export const createTagSchema = createInsertSchema(tags, {
name: (s) => s.min(1).max(50),
color: (s) => s.regex(/^#[0-9A-Fa-f]{6}$/, 'Color must be a valid hex color (e.g. #6B7280)'),
}).pick({ name: true, color: true });
export const updateTagSchema = createTagSchema.partial();