feat: add Zod validation schemas for all domain routes

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
BOHA
2026-03-23 08:57:38 +01:00
parent a4303b0188
commit d2b22e9399
32 changed files with 819 additions and 140 deletions

View File

@@ -0,0 +1,55 @@
import { z } from 'zod';
const QuotationItemSchema = z.object({
description: z.string().nullish(),
item_description: z.string().nullish(),
quantity: z.number().optional().default(1),
unit: z.string().nullish(),
unit_price: z.number().optional().default(0),
is_included_in_total: z.any().optional().default(true),
position: z.number().optional(),
});
const ScopeSectionSchema = z.object({
title: z.string().nullish(),
title_cz: z.string().nullish(),
content: z.string().nullish(),
position: z.number().optional(),
});
export const CreateQuotationSchema = z.object({
quotation_number: z.string().nullish(),
project_code: z.string().nullish(),
customer_id: z.union([z.number(), z.string()]).transform(v => Number(v)).nullish(),
valid_until: z.string().nullish(),
currency: z.string().optional().default('CZK'),
language: z.string().optional().default('cs'),
vat_rate: z.union([z.number(), z.string()]).transform(v => Number(v)).optional().default(21.0),
apply_vat: z.any().optional().default(true),
exchange_rate: z.union([z.number(), z.string()]).transform(v => Number(v)).optional().default(1.0),
status: z.string().optional().default('active'),
scope_title: z.string().nullish(),
scope_description: z.string().nullish(),
items: z.array(QuotationItemSchema).optional(),
sections: z.array(ScopeSectionSchema).optional(),
});
export const UpdateQuotationSchema = z.object({
quotation_number: z.string().optional(),
project_code: z.string().nullish(),
customer_id: z.union([z.number(), z.string()]).transform(v => Number(v)).optional(),
valid_until: z.union([z.string(), z.null()]).optional(),
currency: z.string().optional(),
language: z.string().optional(),
vat_rate: z.union([z.number(), z.string()]).transform(v => Number(v)).optional(),
apply_vat: z.any().optional(),
exchange_rate: z.union([z.number(), z.string()]).transform(v => Number(v)).optional(),
status: z.string().optional(),
scope_title: z.string().nullish(),
scope_description: z.string().nullish(),
items: z.array(QuotationItemSchema).optional(),
sections: z.array(ScopeSectionSchema).optional(),
});
export type CreateQuotationInput = z.infer<typeof CreateQuotationSchema>;
export type UpdateQuotationInput = z.infer<typeof UpdateQuotationSchema>;