fix: code review — XSS, type safety, validation improvements

Critical:
- InvoiceDetail: sanitize notes HTML with DOMPurify
- OrderDetail: use proper DOMPurify import instead of window fallback

Important:
- AttendanceBalances: add fund_to_date to interface, remove as-any casts
- All schemas: replace z.any() with z.preprocess for boolean fields
- Routes: simplify boolean coercion (Zod handles it now)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
BOHA
2026-03-24 20:13:20 +01:00
parent 3c167cf5c4
commit 106606f3fa
17 changed files with 63 additions and 46 deletions

28
package-lock.json generated
View File

@@ -21,6 +21,7 @@
"@prisma/client": "^6.19.2", "@prisma/client": "^6.19.2",
"bcryptjs": "^3.0.3", "bcryptjs": "^3.0.3",
"date-fns": "^4.1.0", "date-fns": "^4.1.0",
"dompurify": "^3.3.3",
"dotenv": "^17.3.1", "dotenv": "^17.3.1",
"fastify": "^5.8.2", "fastify": "^5.8.2",
"file-type": "^16.5.4", "file-type": "^16.5.4",
@@ -40,6 +41,7 @@
}, },
"devDependencies": { "devDependencies": {
"@types/bcryptjs": "^2.4.6", "@types/bcryptjs": "^2.4.6",
"@types/dompurify": "^3.0.5",
"@types/jsonwebtoken": "^9.0.10", "@types/jsonwebtoken": "^9.0.10",
"@types/mysql": "^2.15.27", "@types/mysql": "^2.15.27",
"@types/node": "^25.5.0", "@types/node": "^25.5.0",
@@ -1432,6 +1434,16 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/@types/dompurify": {
"version": "3.0.5",
"resolved": "https://registry.npmjs.org/@types/dompurify/-/dompurify-3.0.5.tgz",
"integrity": "sha512-1Wg0g3BtQF7sSb27fJQAKck1HECM6zV1EB66j8JH9i3LCjYabJa0FSdiSgsD5K/RbrsR0SiraKacLB+T8ZVYAg==",
"dev": true,
"license": "MIT",
"dependencies": {
"@types/trusted-types": "*"
}
},
"node_modules/@types/estree": { "node_modules/@types/estree": {
"version": "1.0.8", "version": "1.0.8",
"resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
@@ -1548,6 +1560,13 @@
"@types/superagent": "^8.1.0" "@types/superagent": "^8.1.0"
} }
}, },
"node_modules/@types/trusted-types": {
"version": "2.0.7",
"resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz",
"integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==",
"devOptional": true,
"license": "MIT"
},
"node_modules/@vitejs/plugin-react": { "node_modules/@vitejs/plugin-react": {
"version": "6.0.1", "version": "6.0.1",
"resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.1.tgz", "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.1.tgz",
@@ -2314,6 +2333,15 @@
"integrity": "sha512-qiSlmBq9+BCdCA/L46dw8Uy93mloxsPSbwnm5yrKn2vMPiy8KyAskTF6zuV/j5BMsmOGZDPs7KjU+mjb670kfA==", "integrity": "sha512-qiSlmBq9+BCdCA/L46dw8Uy93mloxsPSbwnm5yrKn2vMPiy8KyAskTF6zuV/j5BMsmOGZDPs7KjU+mjb670kfA==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/dompurify": {
"version": "3.3.3",
"resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.3.3.tgz",
"integrity": "sha512-Oj6pzI2+RqBfFG+qOaOLbFXLQ90ARpcGG6UePL82bJLtdsa6CYJD7nmiU8MW9nQNOtCHV3lZ/Bzq1X0QYbBZCA==",
"license": "(MPL-2.0 OR Apache-2.0)",
"optionalDependencies": {
"@types/trusted-types": "^2.0.7"
}
},
"node_modules/dotenv": { "node_modules/dotenv": {
"version": "17.3.1", "version": "17.3.1",
"resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.3.1.tgz", "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.3.1.tgz",

View File

@@ -36,6 +36,7 @@
"@prisma/client": "^6.19.2", "@prisma/client": "^6.19.2",
"bcryptjs": "^3.0.3", "bcryptjs": "^3.0.3",
"date-fns": "^4.1.0", "date-fns": "^4.1.0",
"dompurify": "^3.3.3",
"dotenv": "^17.3.1", "dotenv": "^17.3.1",
"fastify": "^5.8.2", "fastify": "^5.8.2",
"file-type": "^16.5.4", "file-type": "^16.5.4",
@@ -55,6 +56,7 @@
}, },
"devDependencies": { "devDependencies": {
"@types/bcryptjs": "^2.4.6", "@types/bcryptjs": "^2.4.6",
"@types/dompurify": "^3.0.5",
"@types/jsonwebtoken": "^9.0.10", "@types/jsonwebtoken": "^9.0.10",
"@types/mysql": "^2.15.27", "@types/mysql": "^2.15.27",
"@types/node": "^25.5.0", "@types/node": "^25.5.0",

View File

@@ -34,6 +34,7 @@ interface FundUserData {
interface MonthFundData { interface MonthFundData {
month_name: string; month_name: string;
fund: number; fund: number;
fund_to_date: number;
business_days: number; business_days: number;
users?: Record<string, FundUserData>; users?: Record<string, FundUserData>;
} }
@@ -320,7 +321,7 @@ export default function AttendanceBalances() {
let totalCovered = 0; let totalCovered = 0;
for (const monthData of Object.values(fundData.months)) { for (const monthData of Object.values(fundData.months)) {
// Use prorated fund (fund_to_date) for current month, full fund for past // Use prorated fund (fund_to_date) for current month, full fund for past
totalFund += (monthData as any).fund_to_date ?? monthData.fund; totalFund += monthData.fund_to_date ?? monthData.fund;
const us = monthData.users?.[userId]; const us = monthData.users?.[userId];
if (us) { if (us) {
totalWorked += us.worked; totalWorked += us.worked;
@@ -571,10 +572,9 @@ export default function AttendanceBalances() {
className="text-secondary" className="text-secondary"
style={{ fontSize: "12px" }} style={{ fontSize: "12px" }}
> >
{(monthData as any).fund_to_date ?? monthData.fund}h ( {monthData.fund_to_date ?? monthData.fund}h (
{Math.round( {Math.round(
((monthData as any).fund_to_date ?? (monthData.fund_to_date ?? monthData.fund) / 8,
monthData.fund) / 8,
)}{" "} )}{" "}
dnů) dnů)
</span> </span>
@@ -591,7 +591,7 @@ export default function AttendanceBalances() {
const us = monthData.users?.[String(user.id)]; const us = monthData.users?.[String(user.id)];
if (!us) return null; if (!us) return null;
const effectiveFund = const effectiveFund =
(monthData as any).fund_to_date ?? monthData.fund; monthData.fund_to_date ?? monthData.fund;
const pct = const pct =
effectiveFund > 0 effectiveFund > 0
? Math.min( ? Math.min(

View File

@@ -5,6 +5,7 @@ import {
useParams, useParams,
Link, Link,
} from "react-router-dom"; } from "react-router-dom";
import DOMPurify from "dompurify";
import { useAlert } from "../context/AlertContext"; import { useAlert } from "../context/AlertContext";
import { useAuth } from "../context/AuthContext"; import { useAuth } from "../context/AuthContext";
import Forbidden from "../components/Forbidden"; import Forbidden from "../components/Forbidden";
@@ -1969,7 +1970,9 @@ export default function InvoiceDetail() {
<h3 className="admin-card-title">Veřejné poznámky na faktuře</h3> <h3 className="admin-card-title">Veřejné poznámky na faktuře</h3>
{isPaid ? ( {isPaid ? (
notes && notes.trim() && notes !== "<p><br></p>" ? ( notes && notes.trim() && notes !== "<p><br></p>" ? (
<div dangerouslySetInnerHTML={{ __html: notes }} /> <div
dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(notes) }}
/>
) : ( ) : (
<p className="text-tertiary">Žádné poznámky.</p> <p className="text-tertiary">Žádné poznámky.</p>
) )

View File

@@ -5,9 +5,7 @@ import {
useMemo, useMemo,
type ReactNode, type ReactNode,
} from "react"; } from "react";
const DOMPurify = (window as any).DOMPurify || { import DOMPurify from "dompurify";
sanitize: (html: string) => html,
};
import { useAlert } from "../context/AlertContext"; import { useAlert } from "../context/AlertContext";
import { useAuth } from "../context/AuthContext"; import { useAuth } from "../context/AuthContext";
import { useParams, useNavigate, Link } from "react-router-dom"; import { useParams, useNavigate, Link } from "react-router-dom";

View File

@@ -41,9 +41,7 @@ export default async function bankAccountsRoutes(
bic: body.bic ? String(body.bic) : null, bic: body.bic ? String(body.bic) : null,
currency: body.currency ? String(body.currency) : "CZK", currency: body.currency ? String(body.currency) : "CZK",
is_default: is_default:
body.is_default === true || !!body.is_default,
body.is_default === 1 ||
body.is_default === "1",
position: body.position ? Number(body.position) : 0, position: body.position ? Number(body.position) : 0,
}, },
}); });
@@ -110,9 +108,7 @@ export default async function bankAccountsRoutes(
body.currency !== undefined ? String(body.currency) : undefined, body.currency !== undefined ? String(body.currency) : undefined,
is_default: is_default:
body.is_default !== undefined body.is_default !== undefined
? body.is_default === true || ? !!body.is_default
body.is_default === 1 ||
body.is_default === "1"
: undefined, : undefined,
position: position:
body.position !== undefined ? Number(body.position) : undefined, body.position !== undefined ? Number(body.position) : undefined,

View File

@@ -215,11 +215,7 @@ export default async function companySettingsRoutes(
} }
if (body.default_vat_rate !== undefined) if (body.default_vat_rate !== undefined)
data.default_vat_rate = Number(body.default_vat_rate); data.default_vat_rate = Number(body.default_vat_rate);
if (body.require_2fa !== undefined) if (body.require_2fa !== undefined) data.require_2fa = !!body.require_2fa;
data.require_2fa =
body.require_2fa === true ||
body.require_2fa === 1 ||
body.require_2fa === "1";
if ( if (
body.custom_fields !== undefined || body.custom_fields !== undefined ||
body.supplier_field_order !== undefined body.supplier_field_order !== undefined

View File

@@ -177,9 +177,7 @@ export default async function tripsRoutes(
route_from: String(body.route_from), route_from: String(body.route_from),
route_to: String(body.route_to), route_to: String(body.route_to),
is_business: is_business:
body.is_business === true || !!body.is_business,
body.is_business === 1 ||
body.is_business === "1",
notes: body.notes ? String(body.notes) : null, notes: body.notes ? String(body.notes) : null,
}, },
}); });
@@ -231,9 +229,7 @@ export default async function tripsRoutes(
if (body.route_to !== undefined) data.route_to = String(body.route_to); if (body.route_to !== undefined) data.route_to = String(body.route_to);
if (body.is_business !== undefined) if (body.is_business !== undefined)
data.is_business = data.is_business =
body.is_business === true || !!body.is_business;
body.is_business === 1 ||
body.is_business === "1";
if (body.notes !== undefined) if (body.notes !== undefined)
data.notes = body.notes ? String(body.notes) : null; data.notes = body.notes ? String(body.notes) : null;

View File

@@ -108,9 +108,7 @@ export default async function vehiclesRoutes(
body.actual_km !== undefined ? Number(body.actual_km) : undefined, body.actual_km !== undefined ? Number(body.actual_km) : undefined,
is_active: is_active:
body.is_active !== undefined body.is_active !== undefined
? body.is_active === true || ? !!body.is_active
body.is_active === 1 ||
body.is_active === "1"
: undefined, : undefined,
}, },
}); });

View File

@@ -7,7 +7,7 @@ export const CreateBankAccountSchema = z.object({
iban: z.string().nullish(), iban: z.string().nullish(),
bic: z.string().nullish(), bic: z.string().nullish(),
currency: z.string().optional().default("CZK"), currency: z.string().optional().default("CZK"),
is_default: z.any().optional().default(false), is_default: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(false),
position: z position: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
@@ -22,7 +22,7 @@ export const UpdateBankAccountSchema = z.object({
iban: z.string().nullish(), iban: z.string().nullish(),
bic: z.string().nullish(), bic: z.string().nullish(),
currency: z.string().optional(), currency: z.string().optional(),
is_default: z.any().optional(), is_default: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
position: z position: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))

View File

@@ -16,7 +16,7 @@ export const UpdateCompanySettingsSchema = z.object({
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional(), .optional(),
require_2fa: z.any().optional(), require_2fa: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
custom_fields: z.array(z.any()).optional(), custom_fields: z.array(z.any()).optional(),
supplier_field_order: z.array(z.any()).optional(), supplier_field_order: z.array(z.any()).optional(),
}); });

View File

@@ -41,7 +41,7 @@ export const CreateInvoiceSchema = z.object({
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional() .optional()
.default(21.0), .default(21.0),
apply_vat: z.any().optional().default(true), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
payment_method: z.string().nullish(), payment_method: z.string().nullish(),
constant_symbol: z.string().nullish(), constant_symbol: z.string().nullish(),
bank_name: z.string().nullish(), bank_name: z.string().nullish(),
@@ -73,7 +73,7 @@ export const UpdateInvoiceSchema = z.object({
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional(), .optional(),
apply_vat: z.any().optional(), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
issue_date: z.union([z.string(), z.null()]).optional(), issue_date: z.union([z.string(), z.null()]).optional(),
due_date: z.union([z.string(), z.null()]).optional(), due_date: z.union([z.string(), z.null()]).optional(),
tax_date: z.union([z.string(), z.null()]).optional(), tax_date: z.union([z.string(), z.null()]).optional(),

View File

@@ -14,7 +14,7 @@ const QuotationItemSchema = z.object({
.transform((v) => Number(v) || 0) .transform((v) => Number(v) || 0)
.optional() .optional()
.default(0), .default(0),
is_included_in_total: z.any().optional().default(true), is_included_in_total: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
position: z position: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
@@ -46,7 +46,7 @@ export const CreateQuotationSchema = z.object({
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional() .optional()
.default(21.0), .default(21.0),
apply_vat: z.any().optional().default(true), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
exchange_rate: z exchange_rate: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
@@ -73,7 +73,7 @@ export const UpdateQuotationSchema = z.object({
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional(), .optional(),
apply_vat: z.any().optional(), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
exchange_rate: z exchange_rate: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))

View File

@@ -14,7 +14,7 @@ const OrderItemSchema = z.object({
.transform((v) => Number(v) || 0) .transform((v) => Number(v) || 0)
.optional() .optional()
.default(0), .default(0),
is_included_in_total: z.any().optional().default(true), is_included_in_total: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
position: z position: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
@@ -55,7 +55,7 @@ export const CreateOrderSchema = z.object({
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional() .optional()
.default(21.0), .default(21.0),
apply_vat: z.any().optional().default(true), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
exchange_rate: z exchange_rate: z
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
@@ -82,7 +82,7 @@ export const UpdateOrderSchema = z.object({
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional(), .optional(),
apply_vat: z.any().optional(), apply_vat: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
items: z.array(OrderItemSchema).optional(), items: z.array(OrderItemSchema).optional(),
sections: z.array(OrderSectionSchema).optional(), sections: z.array(OrderSectionSchema).optional(),
}); });

View File

@@ -11,7 +11,7 @@ export const CreateTripSchema = z.object({
end_km: z.union([z.number(), z.string()]).transform((v) => Number(v)), end_km: z.union([z.number(), z.string()]).transform((v) => Number(v)),
route_from: z.string(), route_from: z.string(),
route_to: z.string(), route_to: z.string(),
is_business: z.any().optional().default(false), is_business: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(false),
notes: z.string().nullish(), notes: z.string().nullish(),
}); });
@@ -27,7 +27,7 @@ export const UpdateTripSchema = z.object({
.optional(), .optional(),
route_from: z.string().optional(), route_from: z.string().optional(),
route_to: z.string().optional(), route_to: z.string().optional(),
is_business: z.any().optional(), is_business: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
notes: z.string().nullish(), notes: z.string().nullish(),
}); });

View File

@@ -7,7 +7,7 @@ export const CreateUserSchema = z.object({
first_name: z.string().min(1, "Jméno je povinné"), first_name: z.string().min(1, "Jméno je povinné"),
last_name: z.string().min(1, "Příjmení je povinné"), last_name: z.string().min(1, "Příjmení je povinné"),
role_id: z.union([z.number(), z.string()]).transform((v) => Number(v)), role_id: z.union([z.number(), z.string()]).transform((v) => Number(v)),
is_active: z.any().optional().default(true), is_active: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
}); });
export const UpdateUserSchema = z.object({ export const UpdateUserSchema = z.object({
@@ -17,7 +17,7 @@ export const UpdateUserSchema = z.object({
first_name: z.string().optional(), first_name: z.string().optional(),
last_name: z.string().optional(), last_name: z.string().optional(),
role_id: z.union([z.number(), z.string(), z.null()]).optional(), role_id: z.union([z.number(), z.string(), z.null()]).optional(),
is_active: z.any().optional(), is_active: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
}); });
export type CreateUserInput = z.infer<typeof CreateUserSchema>; export type CreateUserInput = z.infer<typeof CreateUserSchema>;

View File

@@ -15,7 +15,7 @@ export const CreateVehicleSchema = z.object({
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional() .optional()
.default(0), .default(0),
is_active: z.any().optional().default(true), is_active: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional().default(true),
}); });
export const UpdateVehicleSchema = z.object({ export const UpdateVehicleSchema = z.object({
@@ -31,7 +31,7 @@ export const UpdateVehicleSchema = z.object({
.union([z.number(), z.string()]) .union([z.number(), z.string()])
.transform((v) => Number(v)) .transform((v) => Number(v))
.optional(), .optional(),
is_active: z.any().optional(), is_active: z.preprocess(v => v === true || v === 1 || v === "1", z.boolean()).optional(),
}); });
export type CreateVehicleInput = z.infer<typeof CreateVehicleSchema>; export type CreateVehicleInput = z.infer<typeof CreateVehicleSchema>;