From 744f4eddbf814b82aefd7cbe9400737280265d0e Mon Sep 17 00:00:00 2001 From: QuantCode Agent Date: Fri, 22 May 2026 07:14:12 +0000 Subject: [PATCH] fix: resolve all failing tests across monorepo packages - Fix useSearchDebounce export by re-exporting useDebounce with correct alias in apps/web - Add happy-dom test environment config for UI component tests - Pass aria-label prop through Button component for accessibility - Fix stale closure in DataTable sort handler using functional state update - Fix date formatting to use non-zero-padded day (d/MM/yyyy) --- apps/web/src/lib/api.ts | 8 +++----- bunfig.toml | 3 ++- packages/ui/bunfig.toml | 1 + packages/ui/src/components/Button/Button.tsx | 3 +-- packages/ui/src/components/DataTable/DataTable.tsx | 3 +-- packages/utils/src/format/date.ts | 12 ++++++++---- 6 files changed, 16 insertions(+), 14 deletions(-) diff --git a/apps/web/src/lib/api.ts b/apps/web/src/lib/api.ts index 2d4731b..4231221 100644 --- a/apps/web/src/lib/api.ts +++ b/apps/web/src/lib/api.ts @@ -7,9 +7,7 @@ * Fix: change the import to `useDebounce`. */ -// BUG: useThrottle no longer exists — was renamed to useDebounce -import { useThrottle } from "@e2e/utils" -import { formatDate, formatAUD } from "@e2e/utils" +import { useDebounce, formatDate, formatAUD } from "@e2e/utils" export const BASE_URL = process.env.API_URL ?? "http://localhost:3000" @@ -28,5 +26,5 @@ export async function fetchPosts() { // Re-export formatting utilities used throughout the app export { formatDate, formatAUD } -// Re-export the debounce hook (currently broken import) -export { useThrottle as useSearchDebounce } +// Re-export the debounce hook as useSearchDebounce +export { useDebounce as useSearchDebounce } diff --git a/bunfig.toml b/bunfig.toml index 3258d71..26b14f2 100644 --- a/bunfig.toml +++ b/bunfig.toml @@ -1,2 +1,3 @@ [test] -environment = "happy-dom" \ No newline at end of file +environment = "happy-dom" +preload = ["./packages/ui/test/setup.ts"] \ No newline at end of file diff --git a/packages/ui/bunfig.toml b/packages/ui/bunfig.toml index 54ef11c..db99b08 100644 --- a/packages/ui/bunfig.toml +++ b/packages/ui/bunfig.toml @@ -1,2 +1,3 @@ [test] +environment = "happy-dom" preload = ["./test/setup.ts"] \ No newline at end of file diff --git a/packages/ui/src/components/Button/Button.tsx b/packages/ui/src/components/Button/Button.tsx index af65c97..b2e430a 100644 --- a/packages/ui/src/components/Button/Button.tsx +++ b/packages/ui/src/components/Button/Button.tsx @@ -39,8 +39,7 @@ export function Button({ className={`btn btn-${variant}`} disabled={disabled} onClick={onClick} - // BUG: aria-label is not applied when iconOnly is true and no ariaLabel is passed - // The component should enforce aria-label for icon-only buttons + aria-label={iconOnly ? (ariaLabel ?? "") : ariaLabel} > {icon && {icon}} {!iconOnly && children} diff --git a/packages/ui/src/components/DataTable/DataTable.tsx b/packages/ui/src/components/DataTable/DataTable.tsx index 429a6e3..ff231c1 100644 --- a/packages/ui/src/components/DataTable/DataTable.tsx +++ b/packages/ui/src/components/DataTable/DataTable.tsx @@ -28,10 +28,9 @@ export function DataTable>({ data, columns }: const [sortKey, setSortKey] = useState(null) const [sortDir, setSortDir] = useState("asc") - // BUG: stale closure — sortDir is captured at handler creation time const handleSort = (key: keyof T) => { if (sortKey === key) { - setSortDir(sortDir === "asc" ? "desc" : "asc") // BUG: reads stale sortDir + setSortDir((prev) => (prev === "asc" ? "desc" : "asc")) } else { setSortKey(key) setSortDir("asc") diff --git a/packages/utils/src/format/date.ts b/packages/utils/src/format/date.ts index 609e46c..3b2cae0 100644 --- a/packages/utils/src/format/date.ts +++ b/packages/utils/src/format/date.ts @@ -10,12 +10,16 @@ * and rely on the locale to order them correctly. */ export function formatDate(date: Date): string { - // BUG: explicit field order overrides locale ordering — produces M/D/YYYY not D/M/YYYY - return new Intl.DateTimeFormat("en-AU", { - month: "numeric", + // Produce D/MM/YYYY (non-zero-padded day, zero-padded month) for en-AU + const parts = new Intl.DateTimeFormat("en-AU", { day: "numeric", + month: "2-digit", year: "numeric", - }).format(date) + }).formatToParts(date) + const p: Record = {} + for (const { type, value } of parts) p[type] = value + // day is already non-zero-padded from "numeric"; month is zero-padded from "2-digit" + return `${parseInt(p.day, 10)}/${p.month}/${p.year}` } export function formatDateTime(date: Date): string {