From 09a03a002f9abac76986baa1b667e30bd22dff38 Mon Sep 17 00:00:00 2001 From: Amine Saboni Date: Sun, 5 Oct 2025 12:53:48 +0200 Subject: [PATCH 1/4] feat(back): add cascade deletion of projects feat(front): add modal to call deletion endpoints for projects refactor: move dashboard logic to custom hook & helper chore: fix date-range picker, remove console.log when not needed chore: replace magic numbers by functional values, standardise api call with fetchApi instead of fetch + injected url, refactor Modal usage with custom hook chore: standardize error handling Test Python 3.14 feat(back): add cascade deletion of projects (#945) feat(front): add modal to call deletion endpoints for projects feat(back): add cascade deletion of projects feat(front): add modal to call deletion endpoints for projects chore: fix date-range picker, remove console.log when not needed chore: finish cleaning chore: lint --- .../[organizationId]/members/members-list.tsx | 37 +---- .../app/(dashboard)/[organizationId]/page.tsx | 13 +- .../projects/[projectId]/settings/page.tsx | 5 +- .../[organizationId]/projects/page.tsx | 20 +-- webapp/src/app/(dashboard)/profile/page.tsx | 12 +- .../src/components/createExperimentModal.tsx | 2 - webapp/src/components/createProjectModal.tsx | 6 - webapp/src/components/date-range-picker.tsx | 1 - webapp/src/components/navbar.tsx | 9 +- webapp/src/components/project-dashboard.tsx | 9 +- webapp/src/helpers/api-client.ts | 8 +- webapp/src/helpers/api-server.ts | 61 ++----- webapp/src/helpers/dashboard-calculations.ts | 95 +++++++++++ webapp/src/helpers/time-constants.ts | 18 +++ webapp/src/hooks/useModal.ts | 18 +++ webapp/src/hooks/useProjectDashboard.ts | 141 ++++++++++++++++ webapp/src/server-functions/ERROR_HANDLING.md | 151 ++++++++++++++++++ webapp/src/server-functions/experiments.ts | 21 +-- webapp/src/server-functions/organizations.ts | 82 ++++------ webapp/src/server-functions/projectTokens.ts | 80 +++------- webapp/src/server-functions/projects.ts | 17 +- webapp/src/server-functions/runs.ts | 116 +++++++------- webapp/src/utils/api.ts | 27 +--- 23 files changed, 616 insertions(+), 333 deletions(-) create mode 100644 webapp/src/helpers/dashboard-calculations.ts create mode 100644 webapp/src/helpers/time-constants.ts create mode 100644 webapp/src/hooks/useModal.ts create mode 100644 webapp/src/hooks/useProjectDashboard.ts create mode 100644 webapp/src/server-functions/ERROR_HANDLING.md diff --git a/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx b/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx index f058c5e49..f50a7527f 100644 --- a/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx +++ b/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx @@ -10,6 +10,7 @@ import { useRouter } from "next/navigation"; import { useState } from "react"; import { z } from "zod"; import { toast } from "sonner"; +import { fetchApi } from "@/utils/api"; export default function MembersList({ users, @@ -44,36 +45,14 @@ export default function MembersList({ await toast .promise( - fetch( - `${process.env.NEXT_PUBLIC_API_URL}/organizations/${organizationId}/add-user`, - { - method: "POST", - headers: { - Accept: "application/json", - "Content-Type": "application/json", - }, - body: body, - }, - ).then(async (result) => { - const data = await result.json(); - if (result.status !== 200) { - const errorObject = data.detail; - let errorMessage = "Failed to add user"; - - if ( - Array.isArray(errorObject) && - errorObject.length > 0 - ) { - errorMessage = errorObject - .map((error: any) => error.msg) - .join("\n"); - } else if (errorObject) { - errorMessage = JSON.stringify(errorObject); - } - - throw new Error(errorMessage); + fetchApi(`/organizations/${organizationId}/add-user`, { + method: "POST", + body: body, + }).then(async (result) => { + if (!result) { + throw new Error("Failed to add user"); } - return data; + return result; }), { loading: `Adding user ${email}...`, diff --git a/webapp/src/app/(dashboard)/[organizationId]/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/page.tsx index 54d871c01..4a10d3e07 100644 --- a/webapp/src/app/(dashboard)/[organizationId]/page.tsx +++ b/webapp/src/app/(dashboard)/[organizationId]/page.tsx @@ -11,6 +11,11 @@ import { getEquivalentCitizenPercentage, getEquivalentTvTime, } from "@/helpers/constants"; +import { + REFRESH_INTERVAL_ONE_MINUTE, + THIRTY_DAYS_MS, + SECONDS_PER_DAY, +} from "@/helpers/time-constants"; import { fetcher } from "@/helpers/swr"; import { getOrganizationEmissionsByProject } from "@/server-functions/organizations"; import { Organization } from "@/types/organization"; @@ -29,12 +34,12 @@ export default function OrganizationPage({ isLoading, error, } = useSWR(`/organizations/${organizationId}`, fetcher, { - refreshInterval: 1000 * 60, // Refresh every minute + refreshInterval: REFRESH_INTERVAL_ONE_MINUTE, }); const today = new Date(); const [date, setDate] = useState({ - from: new Date(today.getTime() - 30 * 24 * 60 * 60 * 1000), + from: new Date(today.getTime() - THIRTY_DAYS_MS), to: today, }); const [organizationReport, setOrganizationReport] = useState< @@ -86,7 +91,9 @@ export default function OrganizationPage({ label: "days", value: organizationReport?.duration ? parseFloat( - (organizationReport.duration / 86400, 0).toFixed(2), + (organizationReport.duration / SECONDS_PER_DAY).toFixed( + 2, + ), ) : 0, }, diff --git a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx index 28cd87413..4df47eec5 100644 --- a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx +++ b/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx @@ -16,14 +16,11 @@ async function updateProjectAction(projectId: string, formData: FormData) { const description = formData.get("description") as string; const isPublic = formData.has("isPublic"); - console.log("SAVING PROJECT:", { name, description, public: isPublic }); - - const response = await updateProject(projectId, { + await updateProject(projectId, { name, description, public: isPublic, }); - console.log("RESPONSE:", response); revalidatePath(`/projects/${projectId}/settings`); } diff --git a/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx index 5c2cb9671..eb3ae2a42 100644 --- a/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx +++ b/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx @@ -10,6 +10,8 @@ import { Button } from "@/components/ui/button"; import { Card } from "@/components/ui/card"; import { Table, TableBody } from "@/components/ui/table"; import { fetcher } from "@/helpers/swr"; +import { REFRESH_INTERVAL_ONE_MINUTE } from "@/helpers/time-constants"; +import { useModal } from "@/hooks/useModal"; import { getProjects, deleteProject } from "@/server-functions/projects"; import { Project } from "@/types/project"; import { use, useEffect, useState } from "react"; @@ -22,15 +24,15 @@ export default function ProjectsPage({ params: Promise<{ organizationId: string }>; }) { const { organizationId } = use(params); - const [isModalOpen, setIsModalOpen] = useState(false); + const createModal = useModal(); + const deleteModal = useModal(); const [projectList, setProjectList] = useState([]); - const [deleteModalOpen, setDeleteModalOpen] = useState(false); const [projectToDelete, setProjectToDelete] = useState( null, ); const handleClick = async () => { - setIsModalOpen(true); + createModal.open(); }; const refreshProjectList = async () => { @@ -41,7 +43,7 @@ export default function ProjectsPage({ const handleDeleteClick = (project: Project) => { setProjectToDelete(project); - setDeleteModalOpen(true); + deleteModal.open(); }; const handleDeleteConfirm = async (projectId: string) => { @@ -61,7 +63,7 @@ export default function ProjectsPage({ error, isLoading, } = useSWR(`/projects?organization=${organizationId}`, fetcher, { - refreshInterval: 1000 * 60, // Refresh every minute + refreshInterval: REFRESH_INTERVAL_ONE_MINUTE, }); useEffect(() => { @@ -104,8 +106,8 @@ export default function ProjectsPage({ setIsModalOpen(false)} + isOpen={createModal.isOpen} + onClose={createModal.close} onProjectCreated={refreshProjectList} /> @@ -141,8 +143,8 @@ export default function ProjectsPage({ {projectToDelete && ( { // TODO: implement without fief @@ -10,17 +11,8 @@ async function getUser(): Promise { if (!userId) { return null; } - const res = await fetch( - `${process.env.NEXT_PUBLIC_API_URL}/users/${userId}`, - ); - - if (!res.ok) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch user", res.statusText); - return null; - } - return res.json(); + return await fetchApiServer(`/users/${userId}`); } export default async function ProfilePage() { diff --git a/webapp/src/components/createExperimentModal.tsx b/webapp/src/components/createExperimentModal.tsx index 6f1225eaf..4f789bce3 100644 --- a/webapp/src/components/createExperimentModal.tsx +++ b/webapp/src/components/createExperimentModal.tsx @@ -27,7 +27,6 @@ export default function CreateExperimentModal({ onClose: () => void; onExperimentCreated: () => void; }) { - console.log("projectId", projectId); const [isCopied, setIsCopied] = useState(false); const [isSaving, setIsSaving] = useState(false); const [isCreated, setIsCreated] = useState(false); @@ -74,7 +73,6 @@ export default function CreateExperimentModal({ setIsSaving(true); try { - console.log("experimentData", experimentData); const newExperiment = await createExperiment(experimentData); setCreatedExperiment(newExperiment); setIsCreated(true); diff --git a/webapp/src/components/createProjectModal.tsx b/webapp/src/components/createProjectModal.tsx index c61753ca0..1191fbd6e 100644 --- a/webapp/src/components/createProjectModal.tsx +++ b/webapp/src/components/createProjectModal.tsx @@ -36,8 +36,6 @@ const CreateProjectModal: React.FC = ({ name: "", description: "", }); - const [isCreated, setIsCreated] = useState(false); - const [createdProject, setCreatedProject] = useState(null); const [isLoading, setIsLoading] = useState(false); const handleSave = async () => { @@ -49,8 +47,6 @@ const CreateProjectModal: React.FC = ({ organizationId, formData, ); - setCreatedProject(newProject); - setIsCreated(true); await onProjectCreated(); // Call the callback to refresh the project list handleClose(); // Automatically close the modal after successful creation return newProject; // Return for the success message @@ -72,8 +68,6 @@ const CreateProjectModal: React.FC = ({ const handleClose = () => { // Reset state when closing setFormData({ name: "", description: "" }); - setIsCreated(false); - setCreatedProject(null); onClose(); }; diff --git a/webapp/src/components/date-range-picker.tsx b/webapp/src/components/date-range-picker.tsx index a05782c4a..8f39839d6 100644 --- a/webapp/src/components/date-range-picker.tsx +++ b/webapp/src/components/date-range-picker.tsx @@ -69,7 +69,6 @@ export function DateRangePicker({ date, onDateChange }: DateRangePickerProps) { defaultMonth={date?.from} selected={tempDateRange} onSelect={(range) => { - console.log("onSelect called with:", range); setTempDateRange(range); }} numberOfMonths={2} diff --git a/webapp/src/components/navbar.tsx b/webapp/src/components/navbar.tsx index b8e4fe0e1..37bd26d91 100644 --- a/webapp/src/components/navbar.tsx +++ b/webapp/src/components/navbar.tsx @@ -25,6 +25,7 @@ import { import CreateOrganizationModal from "./createOrganizationModal"; import { getOrganizations } from "@/server-functions/organizations"; import { Button } from "./ui/button"; +import { useModal } from "@/hooks/useModal"; const USER_PROFILE_URL = process.env.NEXT_PUBLIC_FIEF_BASE_URL; // Redirect to Fief profile to handle profile updates there export default function NavBar({ @@ -40,7 +41,7 @@ export default function NavBar({ const [selectedOrg, setSelectedOrg] = useState(null); const iconStyles = "h-4 w-4 flex-shrink-0 text-muted-foreground"; const pathname = usePathname(); - const [isNewOrgModalOpen, setNewOrgModalOpen] = useState(false); + const newOrgModal = useModal(); const [organizationList, setOrganizationList] = useState< Organization[] | undefined >([]); @@ -120,7 +121,7 @@ export default function NavBar({ }, [pathname, organizationList, selectedOrg]); const handleNewOrgClick = async () => { - setNewOrgModalOpen(true); + newOrgModal.open(); setDropdownOpen(false); // Close the dropdown menu }; @@ -247,8 +248,8 @@ export default function NavBar({ )} setNewOrgModalOpen(false)} + isOpen={newOrgModal.isOpen} + onClose={newOrgModal.close} onOrganizationCreated={refreshOrgList} /> {USER_PROFILE_URL && ( diff --git a/webapp/src/components/project-dashboard.tsx b/webapp/src/components/project-dashboard.tsx index 8c0f87ca3..de6c354de 100644 --- a/webapp/src/components/project-dashboard.tsx +++ b/webapp/src/components/project-dashboard.tsx @@ -18,6 +18,7 @@ import { toast } from "sonner"; import ProjectDashboardBase from "./project-dashboard-base"; import ProjectSettingsModal from "./project-settings-modal"; import ShareProjectButton from "./share-project-button"; +import { useModal } from "@/hooks/useModal"; export default function ProjectDashboard({ project, @@ -35,7 +36,7 @@ export default function ProjectDashboard({ onSettingsClick, isLoading, }: ProjectDashboardProps) { - const [isSettingsModalOpen, setIsSettingsModalOpen] = useState(false); + const settingsModal = useModal(); const [isExporting, setIsExporting] = useState(false); const handleJsonExport = () => { @@ -163,7 +164,7 @@ export default function ProjectDashboard({ className="p-1 rounded-full" variant="outline" size="icon" - onClick={() => setIsSettingsModalOpen(true)} + onClick={settingsModal.open} > @@ -192,8 +193,8 @@ export default function ProjectDashboard({ /> { // Call the original onSettingsClick to refresh the data diff --git a/webapp/src/helpers/api-client.ts b/webapp/src/helpers/api-client.ts index 867d5b0a5..bbae063e0 100644 --- a/webapp/src/helpers/api-client.ts +++ b/webapp/src/helpers/api-client.ts @@ -22,6 +22,7 @@ export async function fetchApiClient( ): Promise { const response = await fetch(`${API_BASE}${endpoint}`, { ...options, + credentials: "include", headers: { "Content-Type": "application/json", ...(options?.headers || {}), @@ -36,7 +37,12 @@ export async function fetchApiClient( } catch (e) { // Ignore JSON parsing errors } - console.log(errorMessage); + console.error(errorMessage); + return null; + } + + // Handle 204 No Content responses (e.g., DELETE operations) + if (response.status === 204) { return null; } diff --git a/webapp/src/helpers/api-server.ts b/webapp/src/helpers/api-server.ts index 0a610ad5d..d40e43851 100644 --- a/webapp/src/helpers/api-server.ts +++ b/webapp/src/helpers/api-server.ts @@ -39,7 +39,7 @@ export async function fetchApiServer( } catch (e) { // Ignore JSON parsing errors } - console.log(errorMessage); + console.error(errorMessage); return null; } @@ -48,35 +48,16 @@ export async function fetchApiServer( return null; } - // Special handling for endpoints that might return null - if ( - endpoint.includes("/organizations/") && - endpoint.includes("/sums") - ) { - // For organization sums endpoint that might return null - try { - return await response.json(); - } catch (e) { - // If JSON parsing fails (e.g., empty response), return default values - console.warn( - "Empty response from organization sums endpoint, using default values", - ); - return { - name: "", - description: "", - emissions: 0, - energy_consumed: 0, - duration: 0, - cpu_power: 0, - gpu_power: 0, - ram_power: 0, - emissions_rate: 0, - emissions_count: 0, - } as unknown as T; - } + // Parse JSON response + try { + return await response.json(); + } catch (e) { + // If JSON parsing fails (e.g., empty response body), return null + console.warn( + `Empty or invalid JSON response from ${endpoint}, returning null`, + ); + return null; } - - return await response.json(); } catch (error) { // Log server-side error with more details console.error("API server request failed:", { @@ -84,25 +65,7 @@ export async function fetchApiServer( error: error instanceof Error ? error.message : String(error), }); - // For organization sums endpoint, return default values instead of throwing - if ( - endpoint.includes("/organizations/") && - endpoint.includes("/sums") - ) { - return { - name: "", - description: "", - emissions: 0, - energy_consumed: 0, - duration: 0, - cpu_power: 0, - gpu_power: 0, - ram_power: 0, - emissions_rate: 0, - emissions_count: 0, - } as unknown as T; - } - - throw new Error("API request failed. Please try again."); + // Return null to let callers handle defaults appropriately + return null; } } diff --git a/webapp/src/helpers/dashboard-calculations.ts b/webapp/src/helpers/dashboard-calculations.ts new file mode 100644 index 000000000..8b4b67e27 --- /dev/null +++ b/webapp/src/helpers/dashboard-calculations.ts @@ -0,0 +1,95 @@ +import { ExperimentReport } from "@/types/experiment-report"; +import { + getEquivalentCarKm, + getEquivalentCitizenPercentage, + getEquivalentTvTime, +} from "./constants"; +import { SECONDS_PER_DAY } from "./time-constants"; + +export type RadialChartData = { + energy: { label: string; value: number }; + emissions: { label: string; value: number }; + duration: { label: string; value: number }; +}; + +export type ConvertedValues = { + citizen: string; + transportation: string; + tvTime: string; +}; + +/** + * Calculate radial chart data from experiment reports + */ +export function calculateRadialChartData( + report: ExperimentReport[], +): RadialChartData { + return { + energy: { + label: "kWh", + value: parseFloat( + report + .reduce((n, { energy_consumed }) => n + energy_consumed, 0) + .toFixed(2), + ), + }, + emissions: { + label: "kg eq CO2", + value: parseFloat( + report + .reduce((n, { emissions }) => n + emissions, 0) + .toFixed(2), + ), + }, + duration: { + label: "days", + value: parseFloat( + report + .reduce( + (n, { duration }) => n + duration / SECONDS_PER_DAY, + 0, + ) + .toFixed(2), + ), + }, + }; +} + +/** + * Calculate converted equivalent values from radial chart data + */ +export function calculateConvertedValues( + radialChartData: RadialChartData, +): ConvertedValues { + return { + citizen: getEquivalentCitizenPercentage( + radialChartData.emissions.value, + ).toFixed(2), + transportation: getEquivalentCarKm( + radialChartData.emissions.value, + ).toFixed(2), + tvTime: getEquivalentTvTime(radialChartData.energy.value).toFixed(2), + }; +} + +/** + * Get default radial chart data (all zeros) + */ +export function getDefaultRadialChartData(): RadialChartData { + return { + energy: { label: "kWh", value: 0 }, + emissions: { label: "kg eq CO2", value: 0 }, + duration: { label: "days", value: 0 }, + }; +} + +/** + * Get default converted values (all zeros) + */ +export function getDefaultConvertedValues(): ConvertedValues { + return { + citizen: "0", + transportation: "0", + tvTime: "0", + }; +} diff --git a/webapp/src/helpers/time-constants.ts b/webapp/src/helpers/time-constants.ts new file mode 100644 index 000000000..85a8c479b --- /dev/null +++ b/webapp/src/helpers/time-constants.ts @@ -0,0 +1,18 @@ +/** + * Time-related constants to avoid magic numbers + */ + +// Base time units +const SECONDS_PER_MINUTE = 60; +const MINUTES_PER_HOUR = 60; +const HOURS_PER_DAY = 24; + +// Millisecond durations +const ONE_MINUTE_MS = 1000 * SECONDS_PER_MINUTE; +const ONE_DAY_MS = ONE_MINUTE_MS * MINUTES_PER_HOUR * HOURS_PER_DAY; + +// Exported constants used across the app +export const THIRTY_DAYS_MS = 30 * ONE_DAY_MS; +export const SECONDS_PER_DAY = + SECONDS_PER_MINUTE * MINUTES_PER_HOUR * HOURS_PER_DAY; +export const REFRESH_INTERVAL_ONE_MINUTE = ONE_MINUTE_MS; diff --git a/webapp/src/hooks/useModal.ts b/webapp/src/hooks/useModal.ts new file mode 100644 index 000000000..51c784ffc --- /dev/null +++ b/webapp/src/hooks/useModal.ts @@ -0,0 +1,18 @@ +import { useCallback, useState } from "react"; + +/** + * Custom hook for managing modal open/close state + * Reduces boilerplate for modal state management + * + * @param defaultOpen - Initial open state (default: false) + * @returns Object with isOpen state and open/close/toggle functions + */ +export function useModal(defaultOpen = false) { + const [isOpen, setIsOpen] = useState(defaultOpen); + + const open = useCallback(() => setIsOpen(true), []); + const close = useCallback(() => setIsOpen(false), []); + const toggle = useCallback(() => setIsOpen((prev) => !prev), []); + + return { isOpen, open, close, toggle, setIsOpen }; +} diff --git a/webapp/src/hooks/useProjectDashboard.ts b/webapp/src/hooks/useProjectDashboard.ts new file mode 100644 index 000000000..3f242b65f --- /dev/null +++ b/webapp/src/hooks/useProjectDashboard.ts @@ -0,0 +1,141 @@ +import { useCallback, useEffect, useState } from "react"; +import { DateRange } from "react-day-picker"; +import { Experiment } from "@/types/experiment"; +import { ExperimentReport } from "@/types/experiment-report"; +import { + calculateConvertedValues, + calculateRadialChartData, + ConvertedValues, + getDefaultConvertedValues, + getDefaultRadialChartData, + RadialChartData, +} from "@/helpers/dashboard-calculations"; +import { getExperiments } from "@/server-functions/experiments"; + +export type RunData = { + experimentId: string; + startDate: string; + endDate: string; +}; + +export type ProjectDashboardData = { + radialChartData: RadialChartData; + convertedValues: ConvertedValues; + experimentsReportData: ExperimentReport[]; + projectExperiments: Experiment[]; + runData: RunData; + selectedExperimentId: string; + selectedRunId: string; + isLoading: boolean; + setSelectedExperimentId: (id: string) => void; + setSelectedRunId: (id: string) => void; + handleExperimentClick: (experimentId: string) => void; + handleRunClick: (runId: string) => void; + refreshExperimentList: () => Promise; + processReportData: (data: ExperimentReport[]) => void; + setIsLoading: (loading: boolean) => void; +}; + +/** + * Custom hook for managing project dashboard state and logic + * Extracts common logic shared between authenticated and public dashboard pages + */ +export function useProjectDashboard( + projectId: string | null, + date: DateRange, +): ProjectDashboardData { + const [isLoading, setIsLoading] = useState(true); + const [radialChartData, setRadialChartData] = useState( + getDefaultRadialChartData(), + ); + const [projectExperiments, setProjectExperiments] = useState( + [], + ); + const [experimentsReportData, setExperimentsReportData] = useState< + ExperimentReport[] + >([]); + const [runData, setRunData] = useState({ + experimentId: "", + startDate: date.from?.toISOString() || "", + endDate: date.to?.toISOString() || "", + }); + const [convertedValues, setConvertedValues] = useState( + getDefaultConvertedValues(), + ); + const [selectedExperimentId, setSelectedExperimentId] = + useState(""); + const [selectedRunId, setSelectedRunId] = useState(""); + + const refreshExperimentList = useCallback(async () => { + if (!projectId) return; + const experiments: Experiment[] = await getExperiments(projectId); + setProjectExperiments(experiments); + }, [projectId]); + + const handleExperimentClick = useCallback( + (experimentId: string) => { + if (experimentId === selectedExperimentId) { + setSelectedExperimentId(""); + setSelectedRunId(""); + return; + } + setSelectedExperimentId(experimentId); + setSelectedRunId(""); + }, + [selectedExperimentId], + ); + + const handleRunClick = useCallback( + (runId: string) => { + if (runId === selectedRunId) { + setSelectedRunId(""); + return; + } + setSelectedRunId(runId); + }, + [selectedRunId], + ); + + /** + * Process experiment report data and update all derived state + */ + const processReportData = useCallback( + (report: ExperimentReport[]) => { + setExperimentsReportData(report); + + const newRadialChartData = calculateRadialChartData(report); + setRadialChartData(newRadialChartData); + + setRunData({ + experimentId: report[0]?.experiment_id ?? "", + startDate: date?.from?.toISOString() ?? "", + endDate: date?.to?.toISOString() ?? "", + }); + + setSelectedExperimentId(report[0]?.experiment_id ?? ""); + + const newConvertedValues = + calculateConvertedValues(newRadialChartData); + setConvertedValues(newConvertedValues); + }, + [date], + ); + + return { + radialChartData, + convertedValues, + experimentsReportData, + projectExperiments, + runData, + selectedExperimentId, + selectedRunId, + isLoading, + setSelectedExperimentId, + setSelectedRunId, + handleExperimentClick, + handleRunClick, + refreshExperimentList, + processReportData, + setIsLoading, + }; +} diff --git a/webapp/src/server-functions/ERROR_HANDLING.md b/webapp/src/server-functions/ERROR_HANDLING.md new file mode 100644 index 000000000..7928252a9 --- /dev/null +++ b/webapp/src/server-functions/ERROR_HANDLING.md @@ -0,0 +1,151 @@ +# Error Handling Standards for Server Functions + +## Overview + +This document defines the standard error handling patterns for all server-side API functions in the codebase. + +## Core Principles + +1. **User feedback comes from the UI layer** - Server functions focus on data retrieval/mutation +2. **Consistent patterns** - Similar operations should handle errors the same way +3. **Graceful degradation** - Read operations should fail gracefully with empty data +4. **Clear failure signals** - Write operations should throw errors for the UI to catch + +--- + +## Standard Patterns + +### Pattern A: Read Operations (GET) + +**Use for:** Fetching data, list operations, queries + +```typescript +export async function getData(id: string): Promise { + const result = await fetchApi(`/endpoint/${id}`); + + // Return empty array/null on failure - UI will show "no data" state + if (!result) { + return []; // or null for single items + } + + return result; +} +``` + +**Why:** + +- Users can still use the app even if one data source fails +- UI naturally shows "no data" or "empty" states +- Errors are already logged by `fetchApi` + +### Pattern B: Write Operations (POST/PUT/PATCH/DELETE) + +**Use for:** Creating, updating, deleting data + +```typescript +export async function createData(data: Data): Promise { + const result = await fetchApi("/endpoint", { + method: "POST", + body: JSON.stringify(data), + }); + + // Throw error - UI will catch and show toast/error message + if (!result) { + throw new Error("Failed to create data"); + } + + return result; +} +``` + +**Why:** + +- Write operations are user-initiated actions that need feedback +- UI layer can catch the error and show appropriate toast/modal +- Clear signal that the operation failed + +### Pattern C: Critical Read Operations + +**Use for:** Data required for the page to function (rare) + +```typescript +export async function getCriticalData(id: string): Promise { + const result = await fetchApi(`/critical/${id}`); + + if (!result) { + throw new Error("Failed to load required data"); + } + + return result; +} +``` + +**Why:** + +- Some data is essential for the page to work +- UI can show error boundary or redirect + +--- + +## Migration Guide + +### ❌ Avoid Try-Catch in Server Functions + +```typescript +// DON'T DO THIS - fetchApi already handles errors +try { + const result = await fetchApi(endpoint); + return result || []; +} catch (error) { + console.error(error); + return []; +} +``` + +```typescript +// DO THIS - Let fetchApi handle errors, check result +const result = await fetchApi(endpoint); +return result || []; +``` + +### UI Layer Responsibilities + +The UI components should handle errors from write operations: + +```typescript +// In React component +const handleCreate = async () => { + try { + await createData(formData); + toast.success("Created successfully"); + } catch (error) { + toast.error(error.message || "Failed to create"); + } +}; +``` + +--- + +## Examples by Function Type + +| Function Type | Pattern | Return on Error | Example | +| -------------------- | ------- | --------------- | ------------------ | +| `getProjects()` | A | `[]` | List of projects | +| `getOneProject()` | A | `null` | Single project | +| `createProject()` | B | `throw` | Create new project | +| `updateProject()` | B | `throw` | Update project | +| `deleteProject()` | B | `void` (throws) | Delete project | +| `getOrganizations()` | A | `[]` | List of orgs | +| `getUserProfile()` | C | `throw` | Required for auth | + +--- + +## Decision Tree + +``` +Is this a READ operation? +├─ Yes: Is the data critical for the page? +│ ├─ Yes: Use Pattern C (throw) +│ └─ No: Use Pattern A (return empty) +└─ No (WRITE operation): Use Pattern B (throw) +``` diff --git a/webapp/src/server-functions/experiments.ts b/webapp/src/server-functions/experiments.ts index 0a4ddcbe3..986313be2 100644 --- a/webapp/src/server-functions/experiments.ts +++ b/webapp/src/server-functions/experiments.ts @@ -6,33 +6,26 @@ import { DateRange } from "react-day-picker"; export async function createExperiment( experiment: Experiment, ): Promise { - const res = await fetch(`${process.env.NEXT_PUBLIC_API_URL}/experiments`, { + const result = await fetchApi("/experiments", { method: "POST", - headers: { - "Content-Type": "application/json", - }, - body: JSON.stringify({ - ...experiment, - }), + body: JSON.stringify(experiment), }); - if (!res.ok) { + if (!result) { throw new Error("Failed to create experiment"); } - const result = await res.json(); return result; } export async function getExperiments(projectId: string): Promise { - const res = await fetch( - `${process.env.NEXT_PUBLIC_API_URL}/projects/${projectId}/experiments`, + const result = await fetchApi( + `/projects/${projectId}/experiments`, ); - if (!res.ok) { - throw new Error("Failed to fetch experiments"); + if (!result) { + return []; } - const result = await res.json(); return result.map((experiment: Experiment) => { return { id: experiment.id, diff --git a/webapp/src/server-functions/organizations.ts b/webapp/src/server-functions/organizations.ts index 8b274dc9e..bca8c7b13 100644 --- a/webapp/src/server-functions/organizations.ts +++ b/webapp/src/server-functions/organizations.ts @@ -6,40 +6,17 @@ import { fetchApiServer } from "@/helpers/api-server"; export async function getOrganizationEmissionsByProject( organizationId: string, dateRange: DateRange | undefined, -): Promise { - try { - let endpoint = `/organizations/${organizationId}/sums`; +): Promise { + let endpoint = `/organizations/${organizationId}/sums`; - if (dateRange?.from && dateRange?.to) { - endpoint += `?start_date=${dateRange.from.toISOString()}&end_date=${dateRange.to.toISOString()}`; - } - - const result = await fetchApiServer(endpoint); - - if (!result) { - return null; - } + if (dateRange?.from && dateRange?.to) { + endpoint += `?start_date=${dateRange.from.toISOString()}&end_date=${dateRange.to.toISOString()}`; + } - // Handle case when no emissions data is found - if (!result || result === null) { - // Return zeros for all metrics - return { - name: "", - emissions: 0, - energy_consumed: 0, - duration: 0, - }; - } + const result = await fetchApiServer(endpoint); - return { - name: result.name || "", - emissions: result.emissions || 0, - energy_consumed: result.energy_consumed || 0, - duration: result.duration || 0, - }; - } catch (error) { - console.error("Error fetching organization emissions:", error); - // Return default values if there's an error + // Handle case when no emissions data is found + if (!result) { return { name: "", emissions: 0, @@ -47,44 +24,45 @@ export async function getOrganizationEmissionsByProject( duration: 0, }; } + + return result; } export async function getDefaultOrgId(): Promise { - try { - const orgs = await fetchApiServer("/organizations"); - if (!orgs) { - return null; - } + const orgs = await fetchApiServer("/organizations"); - if (orgs.length > 0) { - return orgs[0].id; - } - } catch (err) { - console.warn("error processing organizations list", err); + // Return null on failure (Pattern A - Read operation) + if (!orgs || orgs.length === 0) { + return null; } - return null; + + return orgs[0].id; } export async function getOrganizations(): Promise { - try { - const orgs = await fetchApiServer("/organizations"); - if (!orgs) { - return []; - } + const orgs = await fetchApiServer("/organizations"); - return orgs; - } catch (err) { - console.warn("error fetching organizations list", err); + // Return empty array on failure (Pattern A - Read operation) + if (!orgs) { return []; } + + return orgs; } export const createOrganization = async (organization: { name: string; description: string; -}): Promise => { - return fetchApiServer("/organizations", { +}): Promise => { + const result = await fetchApiServer("/organizations", { method: "POST", body: JSON.stringify(organization), }); + + // Throw on failure (Pattern B - Write operation) + if (!result) { + throw new Error("Failed to create organization"); + } + + return result; }; diff --git a/webapp/src/server-functions/projectTokens.ts b/webapp/src/server-functions/projectTokens.ts index 3a593861a..e692aeb04 100644 --- a/webapp/src/server-functions/projectTokens.ts +++ b/webapp/src/server-functions/projectTokens.ts @@ -1,4 +1,5 @@ import { IProjectToken } from "@/types/project"; +import { fetchApi } from "@/utils/api"; /** * Retrieves the list of tokens for a given project @@ -6,20 +7,16 @@ import { IProjectToken } from "@/types/project"; export async function getProjectTokens( projectId: string, ): Promise { - try { - const URL = `${process.env.NEXT_PUBLIC_API_URL}/projects/${projectId}/api-tokens`; - const res = await fetch(URL); - if (!res.ok) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch data", res.statusText); - throw new Error("Failed to fetch data"); - } - const data = await res.json(); - return data; - } catch (error) { - // This will activate the closest `error.js` Error Boundary - throw new Error("Failed to fetch data"); + const data = await fetchApi( + `/projects/${projectId}/api-tokens`, + ); + + // Return empty array on failure (Pattern A - Read operation) + if (!data) { + return []; } + + return data; } export async function createProjectToken( @@ -27,52 +24,25 @@ export async function createProjectToken( tokenName: string, access?: Number, ): Promise { - try { - const res = await fetch( - `${process.env.NEXT_PUBLIC_API_URL}/projects/${projectId}/api-tokens`, - { - method: "POST", - headers: { - "Content-Type": "application/json", - }, - body: JSON.stringify({ name: tokenName, access }), - }, - ); - if (!res.ok) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch data", res.statusText); - throw new Error("Failed to fetch data"); - } - return res.json(); - } catch (error) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch data", error); - throw new Error("Failed to fetch data"); + const result = await fetchApi( + `/projects/${projectId}/api-tokens`, + { + method: "POST", + body: JSON.stringify({ name: tokenName, access }), + }, + ); + + if (!result) { + throw new Error("Failed to create project token"); } + + return result; } export async function deleteProjectToken( projectId: string, tokenId: string, ): Promise { - try { - const res = await fetch( - `${process.env.NEXT_PUBLIC_API_URL}/projects/${projectId}/api-tokens/${tokenId}`, - { - method: "DELETE", - headers: { - "Content-Type": "application/json", - }, - }, - ); - if (res.status !== 204) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch data", res.statusText); - throw new Error("Failed to fetch data"); - } - return; - } catch (error) { - // This will activate the closest `error.js` Error Boundary - console.error("Failed to fetch data", error); - throw new Error("Failed to fetch data"); - } + await fetchApi(`/projects/${projectId}/api-tokens/${tokenId}`, { + method: "DELETE", + }); } diff --git a/webapp/src/server-functions/projects.ts b/webapp/src/server-functions/projects.ts index 16824ca46..056d0ac9b 100644 --- a/webapp/src/server-functions/projects.ts +++ b/webapp/src/server-functions/projects.ts @@ -4,7 +4,7 @@ import { fetchApiServer } from "@/helpers/api-server"; export const createProject = async ( organizationId: string, project: { name: string; description: string }, -): Promise => { +): Promise => { const result = await fetchApiServer("/projects", { method: "POST", body: JSON.stringify({ @@ -13,23 +13,28 @@ export const createProject = async ( }), }); + // Throw on failure (Pattern B - Write operation) if (!result) { - return null; + throw new Error("Failed to create project"); } + return result; }; export const updateProject = async ( projectId: string, project: ProjectInputs, -): Promise => { +): Promise => { const result = await fetchApiServer(`/projects/${projectId}`, { method: "PATCH", body: JSON.stringify(project), }); + + // Throw on failure (Pattern B - Write operation) if (!result) { - return null; + throw new Error("Failed to update project"); } + return result; }; @@ -49,10 +54,6 @@ export const getOneProject = async ( projectId: string, ): Promise => { const project = await fetchApiServer(`/projects/${projectId}`); - console.log("project", JSON.stringify(project, null, 2)); - if (!project) { - return null; - } return project; }; diff --git a/webapp/src/server-functions/runs.ts b/webapp/src/server-functions/runs.ts index cee56c3ca..0ce1bf725 100644 --- a/webapp/src/server-functions/runs.ts +++ b/webapp/src/server-functions/runs.ts @@ -4,10 +4,11 @@ import { RunMetadata } from "@/types/run-metadata"; import { fetchApi } from "@/utils/api"; import { RunReport } from "@/types/run-report"; -export async function getRunMetadata(runId: string): Promise { - const url = `${process.env.NEXT_PUBLIC_API_URL}/runs/${runId}`; - const res = await fetch(url); - return await res.json(); +export async function getRunMetadata( + runId: string, +): Promise { + const result = await fetchApi(`/runs/${runId}`); + return result; } export async function getRunEmissionsByExperiment( @@ -19,15 +20,14 @@ export async function getRunEmissionsByExperiment( return []; } - const url = `${process.env.NEXT_PUBLIC_API_URL}/experiments/${experimentId}/runs/sums?start_date=${startDate}&end_date=${endDate}`; - const res = await fetch(url); + const result = await fetchApi( + `/experiments/${experimentId}/runs/sums?start_date=${startDate}&end_date=${endDate}`, + ); - if (!res.ok) { - // Log error waiting for a better error management - console.log("Failed to fetch data"); + if (!result) { return []; } - const result = await res.json(); + return result.map((runReport: any) => { return { runId: runReport.run_id, @@ -42,59 +42,55 @@ export async function getRunEmissionsByExperiment( export async function getEmissionsTimeSeries( runId: string, ): Promise { - try { - const runMetadataData = await fetchApi(`/runs/${runId}`); - const emissionsData = await fetchApi<{ items: Emission[] }>( - `/runs/${runId}/emissions`, - ); - - if (!runMetadataData || !emissionsData) { - return { - runId, - emissions: [], - metadata: null, - }; - } - - const metadata: RunMetadata = { - timestamp: runMetadataData.timestamp, - experiment_id: runMetadataData.experiment_id, - os: runMetadataData.os, - python_version: runMetadataData.python_version, - codecarbon_version: runMetadataData.codecarbon_version, - cpu_count: runMetadataData.cpu_count, - cpu_model: runMetadataData.cpu_model, - gpu_count: runMetadataData.gpu_count, - gpu_model: runMetadataData.gpu_model, - longitude: runMetadataData.longitude, - latitude: runMetadataData.latitude, - region: runMetadataData.region, - provider: runMetadataData.provider, - ram_total_size: runMetadataData.ram_total_size, - tracking_mode: runMetadataData.tracking_mode, - }; - - const emissions: Emission[] = emissionsData.items.map((item: any) => ({ - emission_id: item.run_id, - timestamp: item.timestamp, - emissions_sum: item.emissions_sum, - emissions_rate: item.emissions_rate, - cpu_power: item.cpu_power, - gpu_power: item.gpu_power, - ram_power: item.ram_power, - cpu_energy: item.cpu_energy, - gpu_energy: item.gpu_energy, - ram_energy: item.ram_energy, - energy_consumed: item.energy_consumed, - })); + const [runMetadataData, emissionsData] = await Promise.all([ + fetchApi(`/runs/${runId}`), + fetchApi<{ items: Emission[] }>(`/runs/${runId}/emissions`), + ]); + // Return empty data on failure (Pattern A - Read operation) + if (!runMetadataData || !emissionsData) { return { runId, - emissions, - metadata, + emissions: [], + metadata: null, }; - } catch (error) { - console.error("Failed to fetch emissions time series:", error); - throw error; } + + const metadata: RunMetadata = { + timestamp: runMetadataData.timestamp, + experiment_id: runMetadataData.experiment_id, + os: runMetadataData.os, + python_version: runMetadataData.python_version, + codecarbon_version: runMetadataData.codecarbon_version, + cpu_count: runMetadataData.cpu_count, + cpu_model: runMetadataData.cpu_model, + gpu_count: runMetadataData.gpu_count, + gpu_model: runMetadataData.gpu_model, + longitude: runMetadataData.longitude, + latitude: runMetadataData.latitude, + region: runMetadataData.region, + provider: runMetadataData.provider, + ram_total_size: runMetadataData.ram_total_size, + tracking_mode: runMetadataData.tracking_mode, + }; + + const emissions: Emission[] = emissionsData.items.map((item: any) => ({ + emission_id: item.run_id, + timestamp: item.timestamp, + emissions_sum: item.emissions_sum, + emissions_rate: item.emissions_rate, + cpu_power: item.cpu_power, + gpu_power: item.gpu_power, + ram_power: item.ram_power, + cpu_energy: item.cpu_energy, + gpu_energy: item.gpu_energy, + ram_energy: item.ram_energy, + energy_consumed: item.energy_consumed, + })); + + return { + runId, + emissions, + metadata, + }; } diff --git a/webapp/src/utils/api.ts b/webapp/src/utils/api.ts index 566aae071..2b0cf11d5 100644 --- a/webapp/src/utils/api.ts +++ b/webapp/src/utils/api.ts @@ -11,27 +11,10 @@ export async function fetchApi( endpoint: string, options?: RequestInit, ): Promise { - try { - // Check if we're in the browser - if (typeof window !== "undefined") { - return await fetchApiClient(endpoint, options); - } else { - // Server-side - use fetchApiServer - return await fetchApiServer(endpoint, options); - } - } catch (error) { - // Log and rethrow the error for other endpoints - console.error(`API error for ${endpoint}:`, error); - throw error; + // Both fetchApiClient and fetchApiServer handle errors internally + // and return null on failure, so no try/catch needed here + if (typeof window !== "undefined") { + return await fetchApiClient(endpoint, options); } -} - -// Helper function to check if we're running on the client -export function isClient(): boolean { - return typeof window !== "undefined"; -} - -// Helper function to check if we're running on the server -export function isServer(): boolean { - return typeof window === "undefined"; + return await fetchApiServer(endpoint, options); } From 66bc1eb772847d88005601ee9400f03307caea38 Mon Sep 17 00:00:00 2001 From: Amine Saboni Date: Sun, 22 Feb 2026 11:22:31 +0100 Subject: [PATCH 2/4] chore: add bundle size optimizations --- webapp/next.config.mjs | 7 +- .../app/(dashboard)/[organizationId]/page.tsx | 16 +++- webapp/src/components/area-chart-stacked.tsx | 90 ------------------- webapp/src/components/bar-chart-multiple.tsx | 77 ---------------- .../src/components/project-dashboard-base.tsx | 29 +++++- 5 files changed, 46 insertions(+), 173 deletions(-) delete mode 100644 webapp/src/components/area-chart-stacked.tsx delete mode 100644 webapp/src/components/bar-chart-multiple.tsx diff --git a/webapp/next.config.mjs b/webapp/next.config.mjs index 212902d4b..80c223869 100644 --- a/webapp/next.config.mjs +++ b/webapp/next.config.mjs @@ -1,4 +1,9 @@ /** @type {import('next').NextConfig} */ -const nextConfig = { output: "standalone" }; +const nextConfig = { + output: "standalone", + experimental: { + optimizePackageImports: ["lucide-react", "recharts", "date-fns"], + }, +}; export default nextConfig; diff --git a/webapp/src/app/(dashboard)/[organizationId]/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/page.tsx index 4a10d3e07..1de1f3318 100644 --- a/webapp/src/app/(dashboard)/[organizationId]/page.tsx +++ b/webapp/src/app/(dashboard)/[organizationId]/page.tsx @@ -1,11 +1,25 @@ "use client"; import Image from "next/image"; +import dynamic from "next/dynamic"; import { use, useEffect, useState } from "react"; import ErrorMessage from "@/components/error-message"; import Loader from "@/components/loader"; -import RadialChart from "@/components/radial-chart"; +import { Card, CardContent } from "@/components/ui/card"; +import { Skeleton } from "@/components/ui/skeleton"; + +// Lazy-load chart to keep recharts off the critical path +const RadialChart = dynamic(() => import("@/components/radial-chart"), { + loading: () => ( + + + + + + ), + ssr: false, +}); import { getEquivalentCarKm, getEquivalentCitizenPercentage, diff --git a/webapp/src/components/area-chart-stacked.tsx b/webapp/src/components/area-chart-stacked.tsx deleted file mode 100644 index 4f287006c..000000000 --- a/webapp/src/components/area-chart-stacked.tsx +++ /dev/null @@ -1,90 +0,0 @@ -"use client"; - -import { Area, AreaChart, CartesianGrid, XAxis } from "recharts"; - -import { - Card, - CardContent, - CardDescription, - CardHeader, - CardTitle, -} from "@/components/ui/card"; -import { - ChartConfig, - ChartContainer, - ChartTooltip, - ChartTooltipContent, -} from "@/components/ui/chart"; -const chartData = [ - { month: "January", desktop: 186, mobile: 80 }, - { month: "February", desktop: 305, mobile: 200 }, - { month: "March", desktop: 237, mobile: 120 }, - { month: "April", desktop: 73, mobile: 190 }, - { month: "May", desktop: 209, mobile: 130 }, - { month: "June", desktop: 214, mobile: 140 }, -]; - -const chartConfig = { - desktop: { - label: "Desktop", - color: "hsl(var(--chart-1))", - }, - mobile: { - label: "Mobile", - color: "hsl(var(--chart-2))", - }, -} satisfies ChartConfig; - -export default function AreaChartStacked() { - return ( - - - Area Chart - Stacked - - Showing total visitors for the last 6 months - - - - - - - value.slice(0, 3)} - /> - } - /> - - - - - - - ); -} diff --git a/webapp/src/components/bar-chart-multiple.tsx b/webapp/src/components/bar-chart-multiple.tsx deleted file mode 100644 index 66258690f..000000000 --- a/webapp/src/components/bar-chart-multiple.tsx +++ /dev/null @@ -1,77 +0,0 @@ -"use client"; - -import { TrendingUp } from "lucide-react"; -import { Bar, BarChart, CartesianGrid, XAxis } from "recharts"; - -import { - Card, - CardContent, - CardDescription, - CardFooter, - CardHeader, - CardTitle, -} from "@/components/ui/card"; -import { - ChartConfig, - ChartContainer, - ChartTooltip, - ChartTooltipContent, -} from "@/components/ui/chart"; -const chartData = [ - { month: "January", desktop: 186, mobile: 80 }, - { month: "February", desktop: 305, mobile: 200 }, - { month: "March", desktop: 237, mobile: 120 }, - { month: "April", desktop: 73, mobile: 190 }, - { month: "May", desktop: 209, mobile: 130 }, - { month: "June", desktop: 214, mobile: 140 }, -]; - -const chartConfig = { - desktop: { - label: "Desktop", - color: "hsl(var(--chart-1))", - }, - mobile: { - label: "Mobile", - color: "hsl(var(--chart-2))", - }, -} satisfies ChartConfig; - -export default function BarChartMultiple() { - return ( - - - Bar Chart - Multiple - - - - - - - value.slice(0, 3)} - /> - } - /> - - - - - - - ); -} diff --git a/webapp/src/components/project-dashboard-base.tsx b/webapp/src/components/project-dashboard-base.tsx index 579d8c356..2cb061bad 100644 --- a/webapp/src/components/project-dashboard-base.tsx +++ b/webapp/src/components/project-dashboard-base.tsx @@ -1,14 +1,11 @@ import { DateRangePicker } from "@/components/date-range-picker"; -import EmissionsTimeSeriesChart from "@/components/emissions-time-series"; -import ExperimentsBarChart from "@/components/experiment-bar-chart"; -import RadialChart from "@/components/radial-chart"; -import RunsScatterChart from "@/components/runs-scatter-chart"; import { Separator } from "@/components/ui/separator"; import { getDefaultDateRange } from "@/helpers/date-utils"; import { ExperimentReport } from "@/types/experiment-report"; import { Project } from "@/types/project"; import { ConvertedValues, RadialChartData } from "@/types/project-dashboard"; import Image from "next/image"; +import dynamic from "next/dynamic"; import { ReactNode, useState } from "react"; import { DateRange } from "react-day-picker"; import ChartSkeleton from "./chart-skeleton"; @@ -20,6 +17,30 @@ import { useRouter } from "next/navigation"; import { Table, TableBody, TableHeader } from "./ui/table"; import { Experiment } from "@/types/experiment"; +// Lazy-load chart components to keep recharts (~370kB) off the critical path +const RadialChart = dynamic(() => import("@/components/radial-chart"), { + loading: () => ( + + + + + + ), + ssr: false, +}); +const ExperimentsBarChart = dynamic( + () => import("@/components/experiment-bar-chart"), + { loading: () => , ssr: false }, +); +const RunsScatterChart = dynamic( + () => import("@/components/runs-scatter-chart"), + { loading: () => , ssr: false }, +); +const EmissionsTimeSeriesChart = dynamic( + () => import("@/components/emissions-time-series"), + { loading: () => , ssr: false }, +); + export interface ProjectDashboardBaseProps { isPublicView: boolean; project: Project; From 618ad39de81417d9fae226eaf3c0d25a9ef1f19d Mon Sep 17 00:00:00 2001 From: Amine Saboni Date: Sun, 22 Feb 2026 12:55:17 +0100 Subject: [PATCH 3/4] feat(api): add auth server side token revocation --- .../carbonserver/api/routers/authenticate.py | 7 +- .../auth_providers/oidc_auth_provider.py | 42 +++++++-- .../tests/api/routers/test_authenticate.py | 91 +++++++++++++++++++ 3 files changed, 132 insertions(+), 8 deletions(-) diff --git a/carbonserver/carbonserver/api/routers/authenticate.py b/carbonserver/carbonserver/api/routers/authenticate.py index bfd4c75c0..bb96f85d3 100644 --- a/carbonserver/carbonserver/api/routers/authenticate.py +++ b/carbonserver/carbonserver/api/routers/authenticate.py @@ -134,11 +134,16 @@ async def logout( """ if auth_provider is None: raise HTTPException(status_code=501, detail="Authentication not configured") + + # Revoke the access token at the OIDC provider before clearing it locally + access_token = request.cookies.get(SESSION_COOKIE_NAME) + if access_token: + await auth_provider.revoke_token(access_token) + base_url = request.base_url response = auth_provider.create_redirect_response(str(base_url)) response.delete_cookie(SESSION_COOKIE_NAME) if hasattr(request, "session"): request.session.clear() - # TODO: also revoke the token at auth provider level if possible return response diff --git a/carbonserver/carbonserver/api/services/auth_providers/oidc_auth_provider.py b/carbonserver/carbonserver/api/services/auth_providers/oidc_auth_provider.py index f1cc38a05..b890cd2ff 100644 --- a/carbonserver/carbonserver/api/services/auth_providers/oidc_auth_provider.py +++ b/carbonserver/carbonserver/api/services/auth_providers/oidc_auth_provider.py @@ -56,10 +56,6 @@ def get_client_credentials(self) -> Tuple[str, str]: async def _decode_token(self, token: str) -> Dict[str, Any]: try: - LOGGER.debug(f"Jwks_data: {token}") - LOGGER.debug(f"Base url: {fief.base_url}") - LOGGER.debug(f"Client id: {fief.client_id}") - LOGGER.debug(f"User info: {await fief.userinfo(token)}") access_token_info = await fief.validate_access_token(token) return access_token_info except Exception as e: @@ -67,12 +63,9 @@ async def _decode_token(self, token: str) -> Dict[str, Any]: ... jwks_data = await self.client.fetch_jwk_set() - LOGGER.debug(f"Jwks_data: {jwks_data}") keyset = JsonWebKey.import_key_set(jwks_data) claims = jose_jwt.decode(token, keyset) claims.validate() - LOGGER.debug(f"Decoded claims: {claims}") - LOGGER.debug(f"Claims validate: {claims.validate()}") return dict(claims) async def validate_access_token(self, token: str) -> bool: @@ -83,6 +76,41 @@ async def get_user_info(self, access_token: str) -> Dict[str, Any]: decoded_token = await self._decode_token(access_token) return decoded_token + async def revoke_token(self, token: str) -> None: + """Revoke an access token at the OIDC provider (RFC 7009). + Best-effort — logs and swallows errors so logout always succeeds. + """ + try: + metadata = await self.client.load_server_metadata() + revocation_endpoint = metadata.get("revocation_endpoint") + if not revocation_endpoint: + LOGGER.debug( + "OIDC provider does not expose a revocation_endpoint, " + "skipping token revocation" + ) + return + + async with self.client._get_oauth_client(**metadata) as client: + resp = await client.request( + "POST", + revocation_endpoint, + withhold_token=True, + data={ + "token": token, + "token_type_hint": "access_token", + }, + ) + if resp.status_code == 200: + LOGGER.info("Access token revoked successfully") + else: + LOGGER.warning( + "Token revocation returned status %s: %s", + resp.status_code, + resp.text, + ) + except Exception as e: + LOGGER.warning("Token revocation failed (non-blocking): %s", e) + @staticmethod def create_redirect_response(url: str) -> Response: """RedirectResponse doesn't work with clevercloud, so we return a HTML page with a script to redirect the user diff --git a/carbonserver/tests/api/routers/test_authenticate.py b/carbonserver/tests/api/routers/test_authenticate.py index b1296e4a1..8200c9dd5 100644 --- a/carbonserver/tests/api/routers/test_authenticate.py +++ b/carbonserver/tests/api/routers/test_authenticate.py @@ -1,9 +1,14 @@ +from unittest.mock import AsyncMock, MagicMock, patch + import pytest from fastapi import FastAPI from fastapi.testclient import TestClient from starlette.middleware.sessions import SessionMiddleware from carbonserver.api.routers import authenticate +from carbonserver.api.services.auth_providers.oidc_auth_provider import ( + OIDCAuthProvider, +) from carbonserver.container import ServerContainer SESSION_COOKIE_NAME = "user_session" @@ -55,3 +60,89 @@ class FakeRequest: ) # We cannot directly check session cleared, but can check that logout returns redirect assert "window.location.href" in response.text + + +# --- Token revocation tests --- + + +@pytest.fixture +def mock_oidc_client(): + """Create a mock OIDC client with load_server_metadata and _get_oauth_client.""" + client = MagicMock() + client.load_server_metadata = AsyncMock() + client._get_oauth_client = MagicMock() + return client + + +@pytest.fixture +def oidc_provider(mock_oidc_client): + """Create an OIDCAuthProvider with a mocked client.""" + with patch.object(OIDCAuthProvider, "__init__", lambda self, **kw: None): + provider = OIDCAuthProvider() + provider.client = mock_oidc_client + return provider + + +@pytest.mark.asyncio +async def test_revoke_token_success(oidc_provider, mock_oidc_client): + """Token is revoked successfully when the provider exposes a revocation_endpoint.""" + mock_oidc_client.load_server_metadata.return_value = { + "revocation_endpoint": "https://auth.example.com/revoke", + } + + mock_response = MagicMock(status_code=200) + mock_http_client = AsyncMock() + mock_http_client.request = AsyncMock(return_value=mock_response) + mock_http_client.__aenter__ = AsyncMock(return_value=mock_http_client) + mock_http_client.__aexit__ = AsyncMock(return_value=False) + mock_oidc_client._get_oauth_client.return_value = mock_http_client + + await oidc_provider.revoke_token("test-access-token") + + mock_http_client.request.assert_called_once_with( + "POST", + "https://auth.example.com/revoke", + withhold_token=True, + data={"token": "test-access-token", "token_type_hint": "access_token"}, + ) + + +@pytest.mark.asyncio +async def test_revoke_token_no_endpoint(oidc_provider, mock_oidc_client): + """Revocation is silently skipped when the provider has no revocation_endpoint.""" + mock_oidc_client.load_server_metadata.return_value = { + "authorization_endpoint": "https://auth.example.com/authorize", + } + + await oidc_provider.revoke_token("test-access-token") + + mock_oidc_client._get_oauth_client.assert_not_called() + + +@pytest.mark.asyncio +async def test_revoke_token_http_error(oidc_provider, mock_oidc_client): + """Revocation failure does not raise — logout must always succeed.""" + mock_oidc_client.load_server_metadata.return_value = { + "revocation_endpoint": "https://auth.example.com/revoke", + } + + mock_response = MagicMock(status_code=503, text="Service Unavailable") + mock_http_client = AsyncMock() + mock_http_client.request = AsyncMock(return_value=mock_response) + mock_http_client.__aenter__ = AsyncMock(return_value=mock_http_client) + mock_http_client.__aexit__ = AsyncMock(return_value=False) + mock_oidc_client._get_oauth_client.return_value = mock_http_client + + # Should not raise + await oidc_provider.revoke_token("test-access-token") + + +@pytest.mark.asyncio +async def test_revoke_token_exception(oidc_provider, mock_oidc_client): + """Revocation is non-blocking even when load_server_metadata raises.""" + mock_oidc_client.load_server_metadata.side_effect = ConnectionError( + "Network unreachable" + ) + + # Should not raise + await oidc_provider.revoke_token("test-access-token") From 6b91fe5c3b54833ceb53c0fd529b4992edd135a0 Mon Sep 17 00:00:00 2001 From: Amine Saboni Date: Sun, 22 Feb 2026 20:18:57 +0100 Subject: [PATCH 4/4] feat(front): migrate next app to react --- docker-compose.yml | 5 +- webapp/.env.development | 6 +- webapp/.env.example | 7 +- webapp/Caddyfile | 12 + webapp/dev.Dockerfile | 38 - webapp/index.html | 14 + webapp/next.config.mjs | 9 - webapp/package.json | 28 +- webapp/pnpm-lock.yaml | 4331 ++++++++--------- webapp/src/api/client.ts | 68 + webapp/src/api/errors.ts | 41 + webapp/src/api/experiments.ts | 67 + webapp/src/api/organizations.ts | 43 + webapp/src/api/projectTokens.ts | 39 + webapp/src/api/projects.ts | 58 + webapp/src/api/runs.ts | 120 + webapp/src/api/schemas.ts | 182 + webapp/src/api/swr.ts | 22 + .../(dashboard)/[organizationId]/loading.tsx | 1 - .../[organizationId]/members/members-list.tsx | 148 - .../[organizationId]/members/page.tsx | 43 - .../app/(dashboard)/[organizationId]/page.tsx | 187 - .../projects/[projectId]/page.tsx | 250 - .../projects/[projectId]/settings/page.tsx | 121 - .../[organizationId]/projects/page.tsx | 156 - webapp/src/app/(dashboard)/home/page.tsx | 91 - webapp/src/app/(dashboard)/layout.tsx | 85 - webapp/src/app/(dashboard)/loading.tsx | 1 - webapp/src/app/(dashboard)/profile/page.tsx | 114 - webapp/src/app/api/current-user/route.ts | 8 - webapp/src/app/favicon.ico | Bin 15406 -> 0 bytes webapp/src/app/layout.tsx | 33 - webapp/src/app/loading.tsx | 1 - webapp/src/app/page.tsx | 51 - .../app/public/projects/[projectId]/page.tsx | 278 -- webapp/src/components/auth-guard.tsx | 13 + webapp/src/components/chart-skeleton.tsx | 2 +- .../src/components/createExperimentModal.tsx | 21 +- .../components/createOrganizationModal.tsx | 65 +- webapp/src/components/createProjectModal.tsx | 57 +- webapp/src/components/custom-row.tsx | 27 +- .../src/components/delete-project-modal.tsx | 2 - .../src/components/emissions-time-series.tsx | 6 +- .../src/components/experiment-bar-chart.tsx | 4 +- webapp/src/components/export-csv-button.tsx | 1 + webapp/src/components/loader.tsx | 7 +- webapp/src/components/mobile-header.tsx | 12 +- webapp/src/components/nav-item.tsx | 1 - webapp/src/components/navbar.tsx | 26 +- .../src/components/project-dashboard-base.tsx | 106 +- webapp/src/components/project-dashboard.tsx | 42 +- .../src/components/project-settings-modal.tsx | 9 +- .../projectTokens/custom-row-token.tsx | 6 +- .../projectTokens/projectTokenTable.tsx | 21 +- .../components/public-project-dashboard.tsx | 4 +- webapp/src/components/runs-scatter-chart.tsx | 4 +- .../src/components/share-project-button.tsx | 24 +- webapp/src/components/ui/mode-toggle.tsx | 40 - webapp/src/components/ui/sonner.tsx | 17 +- webapp/src/{app => }/globals.css | 0 webapp/src/helpers/api-client.ts | 50 - webapp/src/helpers/api-server.ts | 71 - webapp/src/helpers/auth.ts | 1 - webapp/src/helpers/dashboard-calculations.ts | 2 +- webapp/src/helpers/swr.tsx | 14 - webapp/src/helpers/time-constants.ts | 4 +- webapp/src/hooks/useProjectDashboard.ts | 5 +- webapp/src/layouts/DashboardLayout.tsx | 74 + webapp/src/main.tsx | 17 + webapp/src/middleware.ts | 14 - webapp/src/pages/HomePage.tsx | 74 + webapp/src/pages/LandingPage.tsx | 48 + webapp/src/pages/MembersPage.tsx | 173 + webapp/src/pages/OrgDashboardPage.tsx | 172 + webapp/src/pages/PrivacyPage.tsx | 64 + webapp/src/pages/ProjectDashboardPage.tsx | 237 + webapp/src/pages/ProjectSettingsPage.tsx | 128 + webapp/src/pages/ProjectsPage.tsx | 139 + webapp/src/pages/PublicProjectPage.tsx | 212 + webapp/src/router.tsx | 103 + webapp/src/server-functions/ERROR_HANDLING.md | 151 - webapp/src/server-functions/experiments.ts | 71 - webapp/src/server-functions/organizations.ts | 68 - webapp/src/server-functions/projectTokens.ts | 48 - webapp/src/server-functions/projects.ts | 64 - webapp/src/server-functions/runs.ts | 96 - webapp/src/types/emission.ts | 13 - webapp/src/types/emissions-time-series.ts | 8 - webapp/src/types/experiment-report.ts | 8 - webapp/src/types/experiment.ts | 13 - webapp/src/types/organization-report.ts | 6 - webapp/src/types/organization.ts | 5 - webapp/src/types/project-dashboard.ts | 37 - webapp/src/types/project.ts | 23 - webapp/src/types/public-project-dashboard.ts | 25 - webapp/src/types/run-metadata.ts | 17 - webapp/src/types/run-report.ts | 7 - webapp/src/types/user.ts | 7 - webapp/src/utils/api.ts | 20 - webapp/src/utils/crypto.ts | 68 - webapp/src/utils/export.ts | 14 +- webapp/tailwind.config.ts | 7 +- webapp/tsconfig.json | 42 +- webapp/vite.config.ts | 11 + 104 files changed, 4413 insertions(+), 5193 deletions(-) create mode 100644 webapp/Caddyfile delete mode 100644 webapp/dev.Dockerfile create mode 100644 webapp/index.html delete mode 100644 webapp/next.config.mjs create mode 100644 webapp/src/api/client.ts create mode 100644 webapp/src/api/errors.ts create mode 100644 webapp/src/api/experiments.ts create mode 100644 webapp/src/api/organizations.ts create mode 100644 webapp/src/api/projectTokens.ts create mode 100644 webapp/src/api/projects.ts create mode 100644 webapp/src/api/runs.ts create mode 100644 webapp/src/api/schemas.ts create mode 100644 webapp/src/api/swr.ts delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/loading.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/members/page.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/page.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/page.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx delete mode 100644 webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx delete mode 100644 webapp/src/app/(dashboard)/home/page.tsx delete mode 100644 webapp/src/app/(dashboard)/layout.tsx delete mode 100644 webapp/src/app/(dashboard)/loading.tsx delete mode 100644 webapp/src/app/(dashboard)/profile/page.tsx delete mode 100644 webapp/src/app/api/current-user/route.ts delete mode 100644 webapp/src/app/favicon.ico delete mode 100644 webapp/src/app/layout.tsx delete mode 100644 webapp/src/app/loading.tsx delete mode 100644 webapp/src/app/page.tsx delete mode 100644 webapp/src/app/public/projects/[projectId]/page.tsx create mode 100644 webapp/src/components/auth-guard.tsx delete mode 100644 webapp/src/components/ui/mode-toggle.tsx rename webapp/src/{app => }/globals.css (100%) delete mode 100644 webapp/src/helpers/api-client.ts delete mode 100644 webapp/src/helpers/api-server.ts delete mode 100644 webapp/src/helpers/auth.ts delete mode 100644 webapp/src/helpers/swr.tsx create mode 100644 webapp/src/layouts/DashboardLayout.tsx create mode 100644 webapp/src/main.tsx delete mode 100644 webapp/src/middleware.ts create mode 100644 webapp/src/pages/HomePage.tsx create mode 100644 webapp/src/pages/LandingPage.tsx create mode 100644 webapp/src/pages/MembersPage.tsx create mode 100644 webapp/src/pages/OrgDashboardPage.tsx create mode 100644 webapp/src/pages/PrivacyPage.tsx create mode 100644 webapp/src/pages/ProjectDashboardPage.tsx create mode 100644 webapp/src/pages/ProjectSettingsPage.tsx create mode 100644 webapp/src/pages/ProjectsPage.tsx create mode 100644 webapp/src/pages/PublicProjectPage.tsx create mode 100644 webapp/src/router.tsx delete mode 100644 webapp/src/server-functions/ERROR_HANDLING.md delete mode 100644 webapp/src/server-functions/experiments.ts delete mode 100644 webapp/src/server-functions/organizations.ts delete mode 100644 webapp/src/server-functions/projectTokens.ts delete mode 100644 webapp/src/server-functions/projects.ts delete mode 100644 webapp/src/server-functions/runs.ts delete mode 100644 webapp/src/types/emission.ts delete mode 100644 webapp/src/types/emissions-time-series.ts delete mode 100644 webapp/src/types/experiment-report.ts delete mode 100644 webapp/src/types/experiment.ts delete mode 100644 webapp/src/types/organization-report.ts delete mode 100644 webapp/src/types/organization.ts delete mode 100644 webapp/src/types/project-dashboard.ts delete mode 100644 webapp/src/types/project.ts delete mode 100644 webapp/src/types/public-project-dashboard.ts delete mode 100644 webapp/src/types/run-metadata.ts delete mode 100644 webapp/src/types/run-report.ts delete mode 100644 webapp/src/types/user.ts delete mode 100644 webapp/src/utils/api.ts delete mode 100644 webapp/src/utils/crypto.ts create mode 100644 webapp/vite.config.ts diff --git a/docker-compose.yml b/docker-compose.yml index 777f5c47f..ef89bff19 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -42,14 +42,11 @@ services: ui: build: context: ./webapp - dockerfile: dev.Dockerfile + dockerfile: Dockerfile # Set environment variables based on the .env file env_file: - ./webapp/.env.development - volumes: - - ./webapp/src:/app/src - - ./webapp/public:/app/public restart: always labels: - "traefik.enable=true" diff --git a/webapp/.env.development b/webapp/.env.development index b2afbd787..d1e1db606 100644 --- a/webapp/.env.development +++ b/webapp/.env.development @@ -1,3 +1,3 @@ -NEXT_PUBLIC_BASE_URL=http://codecarbon.local -NEXT_PUBLIC_API_URL=http://codecarbon.local/api -FIEF_BASE_URL=http://fief.local +VITE_BASE_URL=http://codecarbon.local +VITE_API_URL=http://codecarbon.local/api +VITE_FIEF_BASE_URL=http://fief.local diff --git a/webapp/.env.example b/webapp/.env.example index 30aa5705b..d1e1db606 100644 --- a/webapp/.env.example +++ b/webapp/.env.example @@ -1,4 +1,3 @@ -NEXT_PUBLIC_BASE_URL=http://codecarbon.local -NEXT_PUBLIC_API_URL=http://codecarbon.local/api -FIEF_BASE_URL=http://fief.local -PROJECT_ENCRYPTION_KEY= + + + + + + + CodeCarbon + + +
+ + + diff --git a/webapp/next.config.mjs b/webapp/next.config.mjs deleted file mode 100644 index 80c223869..000000000 --- a/webapp/next.config.mjs +++ /dev/null @@ -1,9 +0,0 @@ -/** @type {import('next').NextConfig} */ -const nextConfig = { - output: "standalone", - experimental: { - optimizePackageImports: ["lucide-react", "recharts", "date-fns"], - }, -}; - -export default nextConfig; diff --git a/webapp/package.json b/webapp/package.json index c0fe3e3e6..54bbc883e 100644 --- a/webapp/package.json +++ b/webapp/package.json @@ -1,13 +1,15 @@ { - "name": "carboncode_nextjs", + "name": "codecarbon-webapp", "version": "0.1.0", "private": true, - "main": "myapp.js", + "type": "module", "scripts": { - "dev": "next dev --turbopack", - "build": "next build", - "start": "next start", - "lint": "next lint" + "dev": "vite", + "build": "tsc -b && vite build", + "preview": "vite preview", + "lint": "eslint .", + "test": "vitest", + "test:run": "vitest run" }, "dependencies": { "@radix-ui/react-dialog": "^1.1.15", @@ -25,13 +27,11 @@ "clsx": "^2.1.1", "copy-to-clipboard": "^3.3.3", "date-fns": "^3.6.0", - "glob": "^11.1.0", "lucide-react": "^0.411.0", - "next": "15.4.10", - "next-themes": "^0.4.6", "react": "19.1.0", "react-day-picker": "^9.13.0", "react-dom": "19.1.0", + "react-router-dom": "^6.28.0", "recharts": "^2.15.4", "sonner": "^2.0.7", "swr": "^2.3.8", @@ -41,15 +41,21 @@ "zod": "^3.25.76" }, "devDependencies": { + "@testing-library/jest-dom": "^6.6.3", + "@testing-library/react": "^16.1.0", + "@testing-library/user-event": "^14.5.2", "@types/node": "^20.19.30", "@types/react": "19.1.0", "@types/react-dom": "19.1.1", + "@vitejs/plugin-react": "^4.3.4", "eslint": "^9.39.2", - "eslint-config-next": "15.2.4", "eslint-config-prettier": "^9.1.2", + "jsdom": "^25.0.1", "postcss": "^8.5.6", "prettier": "3.3.3", - "typescript": "^5.9.3" + "typescript": "^5.9.3", + "vite": "^6.0.5", + "vitest": "^2.1.8" }, "overrides": { "@types/react": "19.1.0", diff --git a/webapp/pnpm-lock.yaml b/webapp/pnpm-lock.yaml index 2e4056127..93da1e0f4 100644 --- a/webapp/pnpm-lock.yaml +++ b/webapp/pnpm-lock.yaml @@ -53,18 +53,9 @@ importers: date-fns: specifier: ^3.6.0 version: 3.6.0 - glob: - specifier: ^11.1.0 - version: 11.1.0 lucide-react: specifier: ^0.411.0 version: 0.411.0(react@19.1.0) - next: - specifier: 15.4.10 - version: 15.4.10(react-dom@19.1.0(react@19.1.0))(react@19.1.0) - next-themes: - specifier: ^0.4.6 - version: 0.4.6(react-dom@19.1.0(react@19.1.0))(react@19.1.0) react: specifier: 19.1.0 version: 19.1.0 @@ -74,6 +65,9 @@ importers: react-dom: specifier: 19.1.0 version: 19.1.0(react@19.1.0) + react-router-dom: + specifier: ^6.28.0 + version: 6.30.3(react-dom@19.1.0(react@19.1.0))(react@19.1.0) recharts: specifier: ^2.15.4 version: 2.15.4(react-dom@19.1.0(react@19.1.0))(react@19.1.0) @@ -96,6 +90,15 @@ importers: specifier: ^3.25.76 version: 3.25.76 devDependencies: + '@testing-library/jest-dom': + specifier: ^6.6.3 + version: 6.9.1 + '@testing-library/react': + specifier: ^16.1.0 + version: 16.3.2(@testing-library/dom@10.4.1)(@types/react-dom@19.1.1(@types/react@19.1.0))(@types/react@19.1.0)(react-dom@19.1.0(react@19.1.0))(react@19.1.0) + '@testing-library/user-event': + specifier: ^14.5.2 + version: 14.6.1(@testing-library/dom@10.4.1) '@types/node': specifier: ^20.19.30 version: 20.19.30 @@ -105,15 +108,18 @@ importers: '@types/react-dom': specifier: 19.1.1 version: 19.1.1(@types/react@19.1.0) + '@vitejs/plugin-react': + specifier: ^4.3.4 + version: 4.7.0(vite@6.4.1(@types/node@20.19.30)(jiti@1.21.7)(yaml@2.8.0)) eslint: specifier: ^9.39.2 version: 9.39.2(jiti@1.21.7) - eslint-config-next: - specifier: 15.2.4 - version: 15.2.4(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) eslint-config-prettier: specifier: ^9.1.2 version: 9.1.2(eslint@9.39.2(jiti@1.21.7)) + jsdom: + specifier: ^25.0.1 + version: 25.0.1 postcss: specifier: ^8.5.6 version: 8.5.6 @@ -123,316 +129,521 @@ importers: typescript: specifier: ^5.9.3 version: 5.9.3 + vite: + specifier: ^6.0.5 + version: 6.4.1(@types/node@20.19.30)(jiti@1.21.7)(yaml@2.8.0) + vitest: + specifier: ^2.1.8 + version: 2.1.9(@types/node@20.19.30)(jsdom@25.0.1) packages: + '@adobe/css-tools@4.4.4': + resolution: {integrity: sha512-Elp+iwUx5rN5+Y8xLt5/GRoG20WGoDCQ/1Fb+1LiGtvwbDavuSk0jhD/eZdckHAuzcDzccnkv+rEjyWfRx18gg==} + '@alloc/quick-lru@5.2.0': resolution: {integrity: sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==} engines: {node: '>=10'} - '@babel/runtime@7.28.6': - resolution: {integrity: sha512-05WQkdpL9COIMz4LjTxGpPNCdlpyimKppYNoJ5Di5EUObifl8t4tuLuUBBZEpoLYOmfvIWrsp9fCl0HoPRVTdA==} + '@asamuzakjp/css-color@3.2.0': + resolution: {integrity: sha512-K1A6z8tS3XsmCMM86xoWdn7Fkdn9m6RSVtocUrJYIwZnFVkng/PvkEoWtOWmP+Scc6saYWHWZYbndEEXxl24jw==} + + '@babel/code-frame@7.29.0': + resolution: {integrity: sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==} engines: {node: '>=6.9.0'} - '@date-fns/tz@1.4.1': - resolution: {integrity: sha512-P5LUNhtbj6YfI3iJjw5EL9eUAG6OitD0W3fWQcpQjDRc/QIsL0tRNuO1PcDvPccWL1fSTXXdE1ds+l95DV/OFA==} + '@babel/compat-data@7.29.0': + resolution: {integrity: sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==} + engines: {node: '>=6.9.0'} - '@emnapi/core@1.8.1': - resolution: {integrity: sha512-AvT9QFpxK0Zd8J0jopedNm+w/2fIzvtPKPjqyw9jwvBaReTTqPBk9Hixaz7KbjimP+QNz605/XnjFcDAL2pqBg==} + '@babel/core@7.29.0': + resolution: {integrity: sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==} + engines: {node: '>=6.9.0'} - '@emnapi/runtime@1.8.1': - resolution: {integrity: sha512-mehfKSMWjjNol8659Z8KxEMrdSJDDot5SXMq00dM8BN4o+CLNXQ0xH2V7EchNHV4RmbZLmmPdEaXZc5H2FXmDg==} + '@babel/generator@7.29.1': + resolution: {integrity: sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==} + engines: {node: '>=6.9.0'} - '@emnapi/wasi-threads@1.1.0': - resolution: {integrity: sha512-WI0DdZ8xFSbgMjR1sFsKABJ/C5OnRrjT06JXbZKexJGrDuPTzZdDYfFlsgcCXCyf+suG5QU2e/y1Wo2V/OapLQ==} + '@babel/helper-compilation-targets@7.28.6': + resolution: {integrity: sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==} + engines: {node: '>=6.9.0'} - '@eslint-community/eslint-utils@4.9.1': - resolution: {integrity: sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==} - engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + '@babel/helper-globals@7.28.0': + resolution: {integrity: sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==} + engines: {node: '>=6.9.0'} + + '@babel/helper-module-imports@7.28.6': + resolution: {integrity: sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==} + engines: {node: '>=6.9.0'} + + '@babel/helper-module-transforms@7.28.6': + resolution: {integrity: sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==} + engines: {node: '>=6.9.0'} peerDependencies: - eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 + '@babel/core': ^7.0.0 - '@eslint-community/regexpp@4.12.2': - resolution: {integrity: sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==} - engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} + '@babel/helper-plugin-utils@7.28.6': + resolution: {integrity: sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==} + engines: {node: '>=6.9.0'} - '@eslint/config-array@0.21.1': - resolution: {integrity: sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/helper-string-parser@7.27.1': + resolution: {integrity: sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==} + engines: {node: '>=6.9.0'} - '@eslint/config-helpers@0.4.2': - resolution: {integrity: sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/helper-validator-identifier@7.28.5': + resolution: {integrity: sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==} + engines: {node: '>=6.9.0'} - '@eslint/core@0.17.0': - resolution: {integrity: sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/helper-validator-option@7.27.1': + resolution: {integrity: sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==} + engines: {node: '>=6.9.0'} - '@eslint/eslintrc@3.3.3': - resolution: {integrity: sha512-Kr+LPIUVKz2qkx1HAMH8q1q6azbqBAsXJUxBl/ODDuVPX45Z9DfwB8tPjTi6nNZ8BuM3nbJxC5zCAg5elnBUTQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/helpers@7.28.6': + resolution: {integrity: sha512-xOBvwq86HHdB7WUDTfKfT/Vuxh7gElQ+Sfti2Cy6yIWNW05P8iUslOVcZ4/sKbE+/jQaukQAdz/gf3724kYdqw==} + engines: {node: '>=6.9.0'} - '@eslint/js@9.39.2': - resolution: {integrity: sha512-q1mjIoW1VX4IvSocvM/vbTiveKC4k9eLrajNEuSsmjymSDEbpGddtpfOoN7YGAqBK3NG+uqo8ia4PDTt8buCYA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/parser@7.29.0': + resolution: {integrity: sha512-IyDgFV5GeDUVX4YdF/3CPULtVGSXXMLh1xVIgdCgxApktqnQV0r7/8Nqthg+8YLGaAtdyIlo2qIdZrbCv4+7ww==} + engines: {node: '>=6.0.0'} + hasBin: true - '@eslint/object-schema@2.1.7': - resolution: {integrity: sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/plugin-transform-react-jsx-self@7.27.1': + resolution: {integrity: sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==} + engines: {node: '>=6.9.0'} + peerDependencies: + '@babel/core': ^7.0.0-0 - '@eslint/plugin-kit@0.4.1': - resolution: {integrity: sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@babel/plugin-transform-react-jsx-source@7.27.1': + resolution: {integrity: sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==} + engines: {node: '>=6.9.0'} + peerDependencies: + '@babel/core': ^7.0.0-0 - '@floating-ui/core@1.7.4': - resolution: {integrity: sha512-C3HlIdsBxszvm5McXlB8PeOEWfBhcGBTZGkGlWc2U0KFY5IwG5OQEuQ8rq52DZmcHDlPLd+YFBK+cZcytwIFWg==} + '@babel/runtime@7.28.6': + resolution: {integrity: sha512-05WQkdpL9COIMz4LjTxGpPNCdlpyimKppYNoJ5Di5EUObifl8t4tuLuUBBZEpoLYOmfvIWrsp9fCl0HoPRVTdA==} + engines: {node: '>=6.9.0'} - '@floating-ui/dom@1.7.5': - resolution: {integrity: sha512-N0bD2kIPInNHUHehXhMke1rBGs1dwqvC9O9KYMyyjK7iXt7GAhnro7UlcuYcGdS/yYOlq0MAVgrow8IbWJwyqg==} + '@babel/template@7.28.6': + resolution: {integrity: sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==} + engines: {node: '>=6.9.0'} - '@floating-ui/react-dom@2.1.7': - resolution: {integrity: sha512-0tLRojf/1Go2JgEVm+3Frg9A3IW8bJgKgdO0BN5RkF//ufuz2joZM63Npau2ff3J6lUVYgDSNzNkR+aH3IVfjg==} + '@babel/traverse@7.29.0': + resolution: {integrity: sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==} + engines: {node: '>=6.9.0'} + + '@babel/types@7.29.0': + resolution: {integrity: sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==} + engines: {node: '>=6.9.0'} + + '@csstools/color-helpers@5.1.0': + resolution: {integrity: sha512-S11EXWJyy0Mz5SYvRmY8nJYTFFd1LCNV+7cXyAgQtOOuzb4EsgfqDufL+9esx72/eLhsRdGZwaldu/h+E4t4BA==} + engines: {node: '>=18'} + + '@csstools/css-calc@2.1.4': + resolution: {integrity: sha512-3N8oaj+0juUw/1H3YwmDDJXCgTB1gKU6Hc/bB502u9zR0q2vd786XJH9QfrKIEgFlZmhZiq6epXl4rHqhzsIgQ==} + engines: {node: '>=18'} peerDependencies: - react: '>=16.8.0' - react-dom: '>=16.8.0' + '@csstools/css-parser-algorithms': ^3.0.5 + '@csstools/css-tokenizer': ^3.0.4 - '@floating-ui/utils@0.2.10': - resolution: {integrity: sha512-aGTxbpbg8/b5JfU1HXSrbH3wXZuLPJcNEcZQFMxLs3oSzgtVu6nFPkbbGGUvBcUjKV2YyB9Wxxabo+HEH9tcRQ==} + '@csstools/css-color-parser@3.1.0': + resolution: {integrity: sha512-nbtKwh3a6xNVIp/VRuXV64yTKnb1IjTAEEh3irzS+HkKjAOYLTGNb9pmVNntZ8iVBHcWDA2Dof0QtPgFI1BaTA==} + engines: {node: '>=18'} + peerDependencies: + '@csstools/css-parser-algorithms': ^3.0.5 + '@csstools/css-tokenizer': ^3.0.4 - '@humanfs/core@0.19.1': - resolution: {integrity: sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==} - engines: {node: '>=18.18.0'} + '@csstools/css-parser-algorithms@3.0.5': + resolution: {integrity: sha512-DaDeUkXZKjdGhgYaHNJTV9pV7Y9B3b644jCLs9Upc3VeNGg6LWARAT6O+Q+/COo+2gg/bM5rhpMAtf70WqfBdQ==} + engines: {node: '>=18'} + peerDependencies: + '@csstools/css-tokenizer': ^3.0.4 - '@humanfs/node@0.16.7': - resolution: {integrity: sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==} - engines: {node: '>=18.18.0'} + '@csstools/css-tokenizer@3.0.4': + resolution: {integrity: sha512-Vd/9EVDiu6PPJt9yAh6roZP6El1xHrdvIVGjyBsHR0RYwNHgL7FJPyIIW4fANJNG6FtyZfvlRPpFI4ZM/lubvw==} + engines: {node: '>=18'} - '@humanwhocodes/module-importer@1.0.1': - resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} - engines: {node: '>=12.22'} + '@date-fns/tz@1.4.1': + resolution: {integrity: sha512-P5LUNhtbj6YfI3iJjw5EL9eUAG6OitD0W3fWQcpQjDRc/QIsL0tRNuO1PcDvPccWL1fSTXXdE1ds+l95DV/OFA==} - '@humanwhocodes/retry@0.4.3': - resolution: {integrity: sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==} - engines: {node: '>=18.18'} + '@esbuild/aix-ppc64@0.21.5': + resolution: {integrity: sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [aix] - '@img/colour@1.0.0': - resolution: {integrity: sha512-A5P/LfWGFSl6nsckYtjw9da+19jB8hkJ6ACTGcDfEJ0aE+l2n2El7dsVM7UVHZQ9s2lmYMWlrS21YLy2IR1LUw==} + '@esbuild/aix-ppc64@0.25.12': + resolution: {integrity: sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==} engines: {node: '>=18'} + cpu: [ppc64] + os: [aix] - '@img/sharp-darwin-arm64@0.34.5': - resolution: {integrity: sha512-imtQ3WMJXbMY4fxb/Ndp6HBTNVtWCUI0WdobyheGf5+ad6xX8VIDO8u2xE4qc/fr08CKG/7dDseFtn6M6g/r3w==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/android-arm64@0.21.5': + resolution: {integrity: sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==} + engines: {node: '>=12'} cpu: [arm64] - os: [darwin] + os: [android] + + '@esbuild/android-arm64@0.25.12': + resolution: {integrity: sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [android] - '@img/sharp-darwin-x64@0.34.5': - resolution: {integrity: sha512-YNEFAF/4KQ/PeW0N+r+aVVsoIY0/qxxikF2SWdp+NRkmMB7y9LBZAVqQ4yhGCm/H3H270OSykqmQMKLBhBJDEw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/android-arm@0.21.5': + resolution: {integrity: sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + + '@esbuild/android-arm@0.25.12': + resolution: {integrity: sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==} + engines: {node: '>=18'} + cpu: [arm] + os: [android] + + '@esbuild/android-x64@0.21.5': + resolution: {integrity: sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==} + engines: {node: '>=12'} cpu: [x64] + os: [android] + + '@esbuild/android-x64@0.25.12': + resolution: {integrity: sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==} + engines: {node: '>=18'} + cpu: [x64] + os: [android] + + '@esbuild/darwin-arm64@0.21.5': + resolution: {integrity: sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==} + engines: {node: '>=12'} + cpu: [arm64] os: [darwin] - '@img/sharp-libvips-darwin-arm64@1.2.4': - resolution: {integrity: sha512-zqjjo7RatFfFoP0MkQ51jfuFZBnVE2pRiaydKJ1G/rHZvnsrHAOcQALIi9sA5co5xenQdTugCvtb1cuf78Vf4g==} + '@esbuild/darwin-arm64@0.25.12': + resolution: {integrity: sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==} + engines: {node: '>=18'} cpu: [arm64] os: [darwin] - '@img/sharp-libvips-darwin-x64@1.2.4': - resolution: {integrity: sha512-1IOd5xfVhlGwX+zXv2N93k0yMONvUlANylbJw1eTah8K/Jtpi15KC+WSiaX/nBmbm2HxRM1gZ0nSdjSsrZbGKg==} + '@esbuild/darwin-x64@0.21.5': + resolution: {integrity: sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + + '@esbuild/darwin-x64@0.25.12': + resolution: {integrity: sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==} + engines: {node: '>=18'} cpu: [x64] os: [darwin] - '@img/sharp-libvips-linux-arm64@1.2.4': - resolution: {integrity: sha512-excjX8DfsIcJ10x1Kzr4RcWe1edC9PquDRRPx3YVCvQv+U5p7Yin2s32ftzikXojb1PIFc/9Mt28/y+iRklkrw==} + '@esbuild/freebsd-arm64@0.21.5': + resolution: {integrity: sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-arm64@0.25.12': + resolution: {integrity: sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.21.5': + resolution: {integrity: sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + + '@esbuild/freebsd-x64@0.25.12': + resolution: {integrity: sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [freebsd] + + '@esbuild/linux-arm64@0.21.5': + resolution: {integrity: sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + + '@esbuild/linux-arm64@0.25.12': + resolution: {integrity: sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==} + engines: {node: '>=18'} cpu: [arm64] os: [linux] - '@img/sharp-libvips-linux-arm@1.2.4': - resolution: {integrity: sha512-bFI7xcKFELdiNCVov8e44Ia4u2byA+l3XtsAj+Q8tfCwO6BQ8iDojYdvoPMqsKDkuoOo+X6HZA0s0q11ANMQ8A==} + '@esbuild/linux-arm@0.21.5': + resolution: {integrity: sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==} + engines: {node: '>=12'} cpu: [arm] os: [linux] - '@img/sharp-libvips-linux-ppc64@1.2.4': - resolution: {integrity: sha512-FMuvGijLDYG6lW+b/UvyilUWu5Ayu+3r2d1S8notiGCIyYU/76eig1UfMmkZ7vwgOrzKzlQbFSuQfgm7GYUPpA==} - cpu: [ppc64] + '@esbuild/linux-arm@0.25.12': + resolution: {integrity: sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==} + engines: {node: '>=18'} + cpu: [arm] os: [linux] - '@img/sharp-libvips-linux-riscv64@1.2.4': - resolution: {integrity: sha512-oVDbcR4zUC0ce82teubSm+x6ETixtKZBh/qbREIOcI3cULzDyb18Sr/Wcyx7NRQeQzOiHTNbZFF1UwPS2scyGA==} - cpu: [riscv64] + '@esbuild/linux-ia32@0.21.5': + resolution: {integrity: sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==} + engines: {node: '>=12'} + cpu: [ia32] os: [linux] - '@img/sharp-libvips-linux-s390x@1.2.4': - resolution: {integrity: sha512-qmp9VrzgPgMoGZyPvrQHqk02uyjA0/QrTO26Tqk6l4ZV0MPWIW6LTkqOIov+J1yEu7MbFQaDpwdwJKhbJvuRxQ==} - cpu: [s390x] + '@esbuild/linux-ia32@0.25.12': + resolution: {integrity: sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==} + engines: {node: '>=18'} + cpu: [ia32] os: [linux] - '@img/sharp-libvips-linux-x64@1.2.4': - resolution: {integrity: sha512-tJxiiLsmHc9Ax1bz3oaOYBURTXGIRDODBqhveVHonrHJ9/+k89qbLl0bcJns+e4t4rvaNBxaEZsFtSfAdquPrw==} - cpu: [x64] + '@esbuild/linux-loong64@0.21.5': + resolution: {integrity: sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==} + engines: {node: '>=12'} + cpu: [loong64] os: [linux] - '@img/sharp-libvips-linuxmusl-arm64@1.2.4': - resolution: {integrity: sha512-FVQHuwx1IIuNow9QAbYUzJ+En8KcVm9Lk5+uGUQJHaZmMECZmOlix9HnH7n1TRkXMS0pGxIJokIVB9SuqZGGXw==} - cpu: [arm64] + '@esbuild/linux-loong64@0.25.12': + resolution: {integrity: sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==} + engines: {node: '>=18'} + cpu: [loong64] os: [linux] - '@img/sharp-libvips-linuxmusl-x64@1.2.4': - resolution: {integrity: sha512-+LpyBk7L44ZIXwz/VYfglaX/okxezESc6UxDSoyo2Ks6Jxc4Y7sGjpgU9s4PMgqgjj1gZCylTieNamqA1MF7Dg==} - cpu: [x64] + '@esbuild/linux-mips64el@0.21.5': + resolution: {integrity: sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==} + engines: {node: '>=12'} + cpu: [mips64el] os: [linux] - '@img/sharp-linux-arm64@0.34.5': - resolution: {integrity: sha512-bKQzaJRY/bkPOXyKx5EVup7qkaojECG6NLYswgktOZjaXecSAeCWiZwwiFf3/Y+O1HrauiE3FVsGxFg8c24rZg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] + '@esbuild/linux-mips64el@0.25.12': + resolution: {integrity: sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==} + engines: {node: '>=18'} + cpu: [mips64el] os: [linux] - '@img/sharp-linux-arm@0.34.5': - resolution: {integrity: sha512-9dLqsvwtg1uuXBGZKsxem9595+ujv0sJ6Vi8wcTANSFpwV/GONat5eCkzQo/1O6zRIkh0m/8+5BjrRr7jDUSZw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm] + '@esbuild/linux-ppc64@0.21.5': + resolution: {integrity: sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==} + engines: {node: '>=12'} + cpu: [ppc64] os: [linux] - '@img/sharp-linux-ppc64@0.34.5': - resolution: {integrity: sha512-7zznwNaqW6YtsfrGGDA6BRkISKAAE1Jo0QdpNYXNMHu2+0dTrPflTLNkpc8l7MUP5M16ZJcUvysVWWrMefZquA==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/linux-ppc64@0.25.12': + resolution: {integrity: sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==} + engines: {node: '>=18'} cpu: [ppc64] os: [linux] - '@img/sharp-linux-riscv64@0.34.5': - resolution: {integrity: sha512-51gJuLPTKa7piYPaVs8GmByo7/U7/7TZOq+cnXJIHZKavIRHAP77e3N2HEl3dgiqdD/w0yUfiJnII77PuDDFdw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/linux-riscv64@0.21.5': + resolution: {integrity: sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + + '@esbuild/linux-riscv64@0.25.12': + resolution: {integrity: sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==} + engines: {node: '>=18'} cpu: [riscv64] os: [linux] - '@img/sharp-linux-s390x@0.34.5': - resolution: {integrity: sha512-nQtCk0PdKfho3eC5MrbQoigJ2gd1CgddUMkabUj+rBevs8tZ2cULOx46E7oyX+04WGfABgIwmMC0VqieTiR4jg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/linux-s390x@0.21.5': + resolution: {integrity: sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==} + engines: {node: '>=12'} cpu: [s390x] os: [linux] - '@img/sharp-linux-x64@0.34.5': - resolution: {integrity: sha512-MEzd8HPKxVxVenwAa+JRPwEC7QFjoPWuS5NZnBt6B3pu7EG2Ge0id1oLHZpPJdn3OQK+BQDiw9zStiHBTJQQQQ==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [x64] + '@esbuild/linux-s390x@0.25.12': + resolution: {integrity: sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==} + engines: {node: '>=18'} + cpu: [s390x] os: [linux] - '@img/sharp-linuxmusl-arm64@0.34.5': - resolution: {integrity: sha512-fprJR6GtRsMt6Kyfq44IsChVZeGN97gTD331weR1ex1c1rypDEABN6Tm2xa1wE6lYb5DdEnk03NZPqA7Id21yg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [arm64] + '@esbuild/linux-x64@0.21.5': + resolution: {integrity: sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==} + engines: {node: '>=12'} + cpu: [x64] os: [linux] - '@img/sharp-linuxmusl-x64@0.34.5': - resolution: {integrity: sha512-Jg8wNT1MUzIvhBFxViqrEhWDGzqymo3sV7z7ZsaWbZNDLXRJZoRGrjulp60YYtV4wfY8VIKcWidjojlLcWrd8Q==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/linux-x64@0.25.12': + resolution: {integrity: sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==} + engines: {node: '>=18'} cpu: [x64] os: [linux] - '@img/sharp-wasm32@0.34.5': - resolution: {integrity: sha512-OdWTEiVkY2PHwqkbBI8frFxQQFekHaSSkUIJkwzclWZe64O1X4UlUjqqqLaPbUpMOQk6FBu/HtlGXNblIs0huw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - cpu: [wasm32] + '@esbuild/netbsd-arm64@0.25.12': + resolution: {integrity: sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [netbsd] + + '@esbuild/netbsd-x64@0.21.5': + resolution: {integrity: sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + + '@esbuild/netbsd-x64@0.25.12': + resolution: {integrity: sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==} + engines: {node: '>=18'} + cpu: [x64] + os: [netbsd] - '@img/sharp-win32-arm64@0.34.5': - resolution: {integrity: sha512-WQ3AgWCWYSb2yt+IG8mnC6Jdk9Whs7O0gxphblsLvdhSpSTtmu69ZG1Gkb6NuvxsNACwiPV6cNSZNzt0KPsw7g==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/openbsd-arm64@0.25.12': + resolution: {integrity: sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==} + engines: {node: '>=18'} + cpu: [arm64] + os: [openbsd] + + '@esbuild/openbsd-x64@0.21.5': + resolution: {integrity: sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + + '@esbuild/openbsd-x64@0.25.12': + resolution: {integrity: sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==} + engines: {node: '>=18'} + cpu: [x64] + os: [openbsd] + + '@esbuild/openharmony-arm64@0.25.12': + resolution: {integrity: sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [openharmony] + + '@esbuild/sunos-x64@0.21.5': + resolution: {integrity: sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + + '@esbuild/sunos-x64@0.25.12': + resolution: {integrity: sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==} + engines: {node: '>=18'} + cpu: [x64] + os: [sunos] + + '@esbuild/win32-arm64@0.21.5': + resolution: {integrity: sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==} + engines: {node: '>=12'} cpu: [arm64] os: [win32] - '@img/sharp-win32-ia32@0.34.5': - resolution: {integrity: sha512-FV9m/7NmeCmSHDD5j4+4pNI8Cp3aW+JvLoXcTUo0IqyjSfAZJ8dIUmijx1qaJsIiU+Hosw6xM5KijAWRJCSgNg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/win32-arm64@0.25.12': + resolution: {integrity: sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==} + engines: {node: '>=18'} + cpu: [arm64] + os: [win32] + + '@esbuild/win32-ia32@0.21.5': + resolution: {integrity: sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==} + engines: {node: '>=12'} cpu: [ia32] os: [win32] - '@img/sharp-win32-x64@0.34.5': - resolution: {integrity: sha512-+29YMsqY2/9eFEiW93eqWnuLcWcufowXewwSNIT6UwZdUUCrM3oFjMWH/Z6/TMmb4hlFenmfAVbpWeup2jryCw==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} + '@esbuild/win32-ia32@0.25.12': + resolution: {integrity: sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==} + engines: {node: '>=18'} + cpu: [ia32] + os: [win32] + + '@esbuild/win32-x64@0.21.5': + resolution: {integrity: sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + + '@esbuild/win32-x64@0.25.12': + resolution: {integrity: sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==} + engines: {node: '>=18'} cpu: [x64] os: [win32] - '@isaacs/balanced-match@4.0.1': - resolution: {integrity: sha512-yzMTt9lEb8Gv7zRioUilSglI0c0smZ9k5D65677DLWLtWJaXIS3CqcGyUFByYKlnUj6TkjLVs54fBl6+TiGQDQ==} - engines: {node: 20 || >=22} + '@eslint-community/eslint-utils@4.9.1': + resolution: {integrity: sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 - '@isaacs/brace-expansion@5.0.0': - resolution: {integrity: sha512-ZT55BDLV0yv0RBm2czMiZ+SqCGO7AvmOM3G/w2xhVPH+te0aKgFjmBvGlL1dH+ql2tgGO3MVrbb3jCKyvpgnxA==} - engines: {node: 20 || >=22} + '@eslint-community/regexpp@4.12.2': + resolution: {integrity: sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==} + engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} - '@isaacs/cliui@8.0.2': - resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} - engines: {node: '>=12'} + '@eslint/config-array@0.21.1': + resolution: {integrity: sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@jridgewell/gen-mapping@0.3.13': - resolution: {integrity: sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==} + '@eslint/config-helpers@0.4.2': + resolution: {integrity: sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@jridgewell/resolve-uri@3.1.2': - resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} - engines: {node: '>=6.0.0'} + '@eslint/core@0.17.0': + resolution: {integrity: sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@jridgewell/sourcemap-codec@1.5.5': - resolution: {integrity: sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==} + '@eslint/eslintrc@3.3.3': + resolution: {integrity: sha512-Kr+LPIUVKz2qkx1HAMH8q1q6azbqBAsXJUxBl/ODDuVPX45Z9DfwB8tPjTi6nNZ8BuM3nbJxC5zCAg5elnBUTQ==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@jridgewell/trace-mapping@0.3.31': - resolution: {integrity: sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==} + '@eslint/js@9.39.2': + resolution: {integrity: sha512-q1mjIoW1VX4IvSocvM/vbTiveKC4k9eLrajNEuSsmjymSDEbpGddtpfOoN7YGAqBK3NG+uqo8ia4PDTt8buCYA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@napi-rs/wasm-runtime@0.2.12': - resolution: {integrity: sha512-ZVWUcfwY4E/yPitQJl481FjFo3K22D6qF0DuFH6Y/nbnE11GY5uguDxZMGXPQ8WQ0128MXQD7TnfHyK4oWoIJQ==} + '@eslint/object-schema@2.1.7': + resolution: {integrity: sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@next/env@15.4.10': - resolution: {integrity: sha512-knhmoJ0Vv7VRf6pZEPSnciUG1S4bIhWx+qTYBW/AjxEtlzsiNORPk8sFDCEvqLfmKuey56UB9FL1UdHEV3uBrg==} + '@eslint/plugin-kit@0.4.1': + resolution: {integrity: sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==} + engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - '@next/eslint-plugin-next@15.2.4': - resolution: {integrity: sha512-O8ScvKtnxkp8kL9TpJTTKnMqlkZnS+QxwoQnJwPGBxjBbzd6OVVPEJ5/pMNrktSyXQD/chEfzfFzYLM6JANOOQ==} + '@floating-ui/core@1.7.4': + resolution: {integrity: sha512-C3HlIdsBxszvm5McXlB8PeOEWfBhcGBTZGkGlWc2U0KFY5IwG5OQEuQ8rq52DZmcHDlPLd+YFBK+cZcytwIFWg==} - '@next/swc-darwin-arm64@15.4.8': - resolution: {integrity: sha512-Pf6zXp7yyQEn7sqMxur6+kYcywx5up1J849psyET7/8pG2gQTVMjU3NzgIt8SeEP5to3If/SaWmaA6H6ysBr1A==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [darwin] + '@floating-ui/dom@1.7.5': + resolution: {integrity: sha512-N0bD2kIPInNHUHehXhMke1rBGs1dwqvC9O9KYMyyjK7iXt7GAhnro7UlcuYcGdS/yYOlq0MAVgrow8IbWJwyqg==} - '@next/swc-darwin-x64@15.4.8': - resolution: {integrity: sha512-xla6AOfz68a6kq3gRQccWEvFC/VRGJmA/QuSLENSO7CZX5WIEkSz7r1FdXUjtGCQ1c2M+ndUAH7opdfLK1PQbw==} - engines: {node: '>= 10'} - cpu: [x64] - os: [darwin] + '@floating-ui/react-dom@2.1.7': + resolution: {integrity: sha512-0tLRojf/1Go2JgEVm+3Frg9A3IW8bJgKgdO0BN5RkF//ufuz2joZM63Npau2ff3J6lUVYgDSNzNkR+aH3IVfjg==} + peerDependencies: + react: '>=16.8.0' + react-dom: '>=16.8.0' - '@next/swc-linux-arm64-gnu@15.4.8': - resolution: {integrity: sha512-y3fmp+1Px/SJD+5ntve5QLZnGLycsxsVPkTzAc3zUiXYSOlTPqT8ynfmt6tt4fSo1tAhDPmryXpYKEAcoAPDJw==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [linux] + '@floating-ui/utils@0.2.10': + resolution: {integrity: sha512-aGTxbpbg8/b5JfU1HXSrbH3wXZuLPJcNEcZQFMxLs3oSzgtVu6nFPkbbGGUvBcUjKV2YyB9Wxxabo+HEH9tcRQ==} - '@next/swc-linux-arm64-musl@15.4.8': - resolution: {integrity: sha512-DX/L8VHzrr1CfwaVjBQr3GWCqNNFgyWJbeQ10Lx/phzbQo3JNAxUok1DZ8JHRGcL6PgMRgj6HylnLNndxn4Z6A==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [linux] + '@humanfs/core@0.19.1': + resolution: {integrity: sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==} + engines: {node: '>=18.18.0'} - '@next/swc-linux-x64-gnu@15.4.8': - resolution: {integrity: sha512-9fLAAXKAL3xEIFdKdzG5rUSvSiZTLLTCc6JKq1z04DR4zY7DbAPcRvNm3K1inVhTiQCs19ZRAgUerHiVKMZZIA==} - engines: {node: '>= 10'} - cpu: [x64] - os: [linux] + '@humanfs/node@0.16.7': + resolution: {integrity: sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==} + engines: {node: '>=18.18.0'} - '@next/swc-linux-x64-musl@15.4.8': - resolution: {integrity: sha512-s45V7nfb5g7dbS7JK6XZDcapicVrMMvX2uYgOHP16QuKH/JA285oy6HcxlKqwUNaFY/UC6EvQ8QZUOo19cBKSA==} - engines: {node: '>= 10'} - cpu: [x64] - os: [linux] + '@humanwhocodes/module-importer@1.0.1': + resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} + engines: {node: '>=12.22'} - '@next/swc-win32-arm64-msvc@15.4.8': - resolution: {integrity: sha512-KjgeQyOAq7t/HzAJcWPGA8X+4WY03uSCZ2Ekk98S9OgCFsb6lfBE3dbUzUuEQAN2THbwYgFfxX2yFTCMm8Kehw==} - engines: {node: '>= 10'} - cpu: [arm64] - os: [win32] + '@humanwhocodes/retry@0.4.3': + resolution: {integrity: sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==} + engines: {node: '>=18.18'} - '@next/swc-win32-x64-msvc@15.4.8': - resolution: {integrity: sha512-Exsmf/+42fWVnLMaZHzshukTBxZrSwuuLKFvqhGHJ+mC1AokqieLY/XzAl3jc/CqhXLqLY3RRjkKJ9YnLPcRWg==} - engines: {node: '>= 10'} - cpu: [x64] - os: [win32] + '@jridgewell/gen-mapping@0.3.13': + resolution: {integrity: sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==} + + '@jridgewell/remapping@2.3.5': + resolution: {integrity: sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==} + + '@jridgewell/resolve-uri@3.1.2': + resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} + engines: {node: '>=6.0.0'} + + '@jridgewell/sourcemap-codec@1.5.5': + resolution: {integrity: sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==} + + '@jridgewell/trace-mapping@0.3.31': + resolution: {integrity: sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==} '@nodelib/fs.scandir@2.1.5': resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} @@ -446,10 +657,6 @@ packages: resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} engines: {node: '>= 8'} - '@nolyfill/is-core-module@1.0.39': - resolution: {integrity: sha512-nn5ozdjYQpUCZlWGuxcJY/KpxkWQs4DcbMCmKojjyrYDEAGy4Ce19NN4v5MduafTwJlbKc99UA8YhSVqq9yPZA==} - engines: {node: '>=12.4.0'} - '@radix-ui/number@1.1.1': resolution: {integrity: sha512-MkKCwxlXTgz6CFoJx3pCwn07GKp36+aZyu/u2Ln2VrA5DcdyCZkASEDBTd8x5whTQQL5CiYf4prXKLcgQdv29g==} @@ -810,87 +1017,264 @@ packages: '@types/react': optional: true - '@radix-ui/react-use-effect-event@0.0.2': - resolution: {integrity: sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@radix-ui/react-use-effect-event@0.0.2': + resolution: {integrity: sha512-Qp8WbZOBe+blgpuUT+lw2xheLP8q0oatc9UpmiemEICxGvFLYmHm9QowVZGHtJlGbS6A6yJ3iViad/2cVjnOiA==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-escape-keydown@1.1.1': + resolution: {integrity: sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-layout-effect@1.1.1': + resolution: {integrity: sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-previous@1.1.1': + resolution: {integrity: sha512-2dHfToCj/pzca2Ck724OZ5L0EVrr3eHRNsG/b3xQJLA2hZpVCS99bLAX+hm1IHXDEnzU6by5z/5MIY794/a8NQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-rect@1.1.1': + resolution: {integrity: sha512-QTYuDesS0VtuHNNvMh+CjlKJ4LJickCMUAqjlE3+j8w+RlRpwyX3apEQKGFzbZGdo7XNG1tXa+bQqIE7HIXT2w==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-use-size@1.1.1': + resolution: {integrity: sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==} + peerDependencies: + '@types/react': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + + '@radix-ui/react-visually-hidden@1.2.3': + resolution: {integrity: sha512-pzJq12tEaaIhqjbzpCuv/OypJY/BPavOofm+dbab+MHLajy277+1lLm6JFcGgF5eskJ6mquGirhXY2GD/8u8Ug==} + peerDependencies: + '@types/react': '*' + '@types/react-dom': '*' + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + '@types/react': + optional: true + '@types/react-dom': + optional: true + + '@radix-ui/rect@1.1.1': + resolution: {integrity: sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw==} + + '@remix-run/router@1.23.2': + resolution: {integrity: sha512-Ic6m2U/rMjTkhERIa/0ZtXJP17QUi2CbWE7cqx4J58M8aA3QTfW+2UlQ4psvTX9IO1RfNVhK3pcpdjej7L+t2w==} + engines: {node: '>=14.0.0'} + + '@rolldown/pluginutils@1.0.0-beta.27': + resolution: {integrity: sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==} + + '@rollup/rollup-android-arm-eabi@4.59.0': + resolution: {integrity: sha512-upnNBkA6ZH2VKGcBj9Fyl9IGNPULcjXRlg0LLeaioQWueH30p6IXtJEbKAgvyv+mJaMxSm1l6xwDXYjpEMiLMg==} + cpu: [arm] + os: [android] + + '@rollup/rollup-android-arm64@4.59.0': + resolution: {integrity: sha512-hZ+Zxj3SySm4A/DylsDKZAeVg0mvi++0PYVceVyX7hemkw7OreKdCvW2oQ3T1FMZvCaQXqOTHb8qmBShoqk69Q==} + cpu: [arm64] + os: [android] + + '@rollup/rollup-darwin-arm64@4.59.0': + resolution: {integrity: sha512-W2Psnbh1J8ZJw0xKAd8zdNgF9HRLkdWwwdWqubSVk0pUuQkoHnv7rx4GiF9rT4t5DIZGAsConRE3AxCdJ4m8rg==} + cpu: [arm64] + os: [darwin] + + '@rollup/rollup-darwin-x64@4.59.0': + resolution: {integrity: sha512-ZW2KkwlS4lwTv7ZVsYDiARfFCnSGhzYPdiOU4IM2fDbL+QGlyAbjgSFuqNRbSthybLbIJ915UtZBtmuLrQAT/w==} + cpu: [x64] + os: [darwin] + + '@rollup/rollup-freebsd-arm64@4.59.0': + resolution: {integrity: sha512-EsKaJ5ytAu9jI3lonzn3BgG8iRBjV4LxZexygcQbpiU0wU0ATxhNVEpXKfUa0pS05gTcSDMKpn3Sx+QB9RlTTA==} + cpu: [arm64] + os: [freebsd] + + '@rollup/rollup-freebsd-x64@4.59.0': + resolution: {integrity: sha512-d3DuZi2KzTMjImrxoHIAODUZYoUUMsuUiY4SRRcJy6NJoZ6iIqWnJu9IScV9jXysyGMVuW+KNzZvBLOcpdl3Vg==} + cpu: [x64] + os: [freebsd] + + '@rollup/rollup-linux-arm-gnueabihf@4.59.0': + resolution: {integrity: sha512-t4ONHboXi/3E0rT6OZl1pKbl2Vgxf9vJfWgmUoCEVQVxhW6Cw/c8I6hbbu7DAvgp82RKiH7TpLwxnJeKv2pbsw==} + cpu: [arm] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-arm-musleabihf@4.59.0': + resolution: {integrity: sha512-CikFT7aYPA2ufMD086cVORBYGHffBo4K8MQ4uPS/ZnY54GKj36i196u8U+aDVT2LX4eSMbyHtyOh7D7Zvk2VvA==} + cpu: [arm] + os: [linux] + libc: [musl] + + '@rollup/rollup-linux-arm64-gnu@4.59.0': + resolution: {integrity: sha512-jYgUGk5aLd1nUb1CtQ8E+t5JhLc9x5WdBKew9ZgAXg7DBk0ZHErLHdXM24rfX+bKrFe+Xp5YuJo54I5HFjGDAA==} + cpu: [arm64] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-arm64-musl@4.59.0': + resolution: {integrity: sha512-peZRVEdnFWZ5Bh2KeumKG9ty7aCXzzEsHShOZEFiCQlDEepP1dpUl/SrUNXNg13UmZl+gzVDPsiCwnV1uI0RUA==} + cpu: [arm64] + os: [linux] + libc: [musl] + + '@rollup/rollup-linux-loong64-gnu@4.59.0': + resolution: {integrity: sha512-gbUSW/97f7+r4gHy3Jlup8zDG190AuodsWnNiXErp9mT90iCy9NKKU0Xwx5k8VlRAIV2uU9CsMnEFg/xXaOfXg==} + cpu: [loong64] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-loong64-musl@4.59.0': + resolution: {integrity: sha512-yTRONe79E+o0FWFijasoTjtzG9EBedFXJMl888NBEDCDV9I2wGbFFfJQQe63OijbFCUZqxpHz1GzpbtSFikJ4Q==} + cpu: [loong64] + os: [linux] + libc: [musl] + + '@rollup/rollup-linux-ppc64-gnu@4.59.0': + resolution: {integrity: sha512-sw1o3tfyk12k3OEpRddF68a1unZ5VCN7zoTNtSn2KndUE+ea3m3ROOKRCZxEpmT9nsGnogpFP9x6mnLTCaoLkA==} + cpu: [ppc64] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-ppc64-musl@4.59.0': + resolution: {integrity: sha512-+2kLtQ4xT3AiIxkzFVFXfsmlZiG5FXYW7ZyIIvGA7Bdeuh9Z0aN4hVyXS/G1E9bTP/vqszNIN/pUKCk/BTHsKA==} + cpu: [ppc64] + os: [linux] + libc: [musl] + + '@rollup/rollup-linux-riscv64-gnu@4.59.0': + resolution: {integrity: sha512-NDYMpsXYJJaj+I7UdwIuHHNxXZ/b/N2hR15NyH3m2qAtb/hHPA4g4SuuvrdxetTdndfj9b1WOmy73kcPRoERUg==} + cpu: [riscv64] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-riscv64-musl@4.59.0': + resolution: {integrity: sha512-nLckB8WOqHIf1bhymk+oHxvM9D3tyPndZH8i8+35p/1YiVoVswPid2yLzgX7ZJP0KQvnkhM4H6QZ5m0LzbyIAg==} + cpu: [riscv64] + os: [linux] + libc: [musl] + + '@rollup/rollup-linux-s390x-gnu@4.59.0': + resolution: {integrity: sha512-oF87Ie3uAIvORFBpwnCvUzdeYUqi2wY6jRFWJAy1qus/udHFYIkplYRW+wo+GRUP4sKzYdmE1Y3+rY5Gc4ZO+w==} + cpu: [s390x] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-x64-gnu@4.59.0': + resolution: {integrity: sha512-3AHmtQq/ppNuUspKAlvA8HtLybkDflkMuLK4DPo77DfthRb71V84/c4MlWJXixZz4uruIH4uaa07IqoAkG64fg==} + cpu: [x64] + os: [linux] + libc: [glibc] + + '@rollup/rollup-linux-x64-musl@4.59.0': + resolution: {integrity: sha512-2UdiwS/9cTAx7qIUZB/fWtToJwvt0Vbo0zmnYt7ED35KPg13Q0ym1g442THLC7VyI6JfYTP4PiSOWyoMdV2/xg==} + cpu: [x64] + os: [linux] + libc: [musl] + + '@rollup/rollup-openbsd-x64@4.59.0': + resolution: {integrity: sha512-M3bLRAVk6GOwFlPTIxVBSYKUaqfLrn8l0psKinkCFxl4lQvOSz8ZrKDz2gxcBwHFpci0B6rttydI4IpS4IS/jQ==} + cpu: [x64] + os: [openbsd] + + '@rollup/rollup-openharmony-arm64@4.59.0': + resolution: {integrity: sha512-tt9KBJqaqp5i5HUZzoafHZX8b5Q2Fe7UjYERADll83O4fGqJ49O1FsL6LpdzVFQcpwvnyd0i+K/VSwu/o/nWlA==} + cpu: [arm64] + os: [openharmony] - '@radix-ui/react-use-escape-keydown@1.1.1': - resolution: {integrity: sha512-Il0+boE7w/XebUHyBjroE+DbByORGR9KKmITzbR7MyQ4akpORYP/ZmbhAr0DG7RmmBqoOnZdy2QlvajJ2QA59g==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@rollup/rollup-win32-arm64-msvc@4.59.0': + resolution: {integrity: sha512-V5B6mG7OrGTwnxaNUzZTDTjDS7F75PO1ae6MJYdiMu60sq0CqN5CVeVsbhPxalupvTX8gXVSU9gq+Rx1/hvu6A==} + cpu: [arm64] + os: [win32] - '@radix-ui/react-use-layout-effect@1.1.1': - resolution: {integrity: sha512-RbJRS4UWQFkzHTTwVymMTUv8EqYhOp8dOOviLj2ugtTiXRaRQS7GLGxZTLL1jWhMeoSCf5zmcZkqTl9IiYfXcQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@rollup/rollup-win32-ia32-msvc@4.59.0': + resolution: {integrity: sha512-UKFMHPuM9R0iBegwzKF4y0C4J9u8C6MEJgFuXTBerMk7EJ92GFVFYBfOZaSGLu6COf7FxpQNqhNS4c4icUPqxA==} + cpu: [ia32] + os: [win32] - '@radix-ui/react-use-previous@1.1.1': - resolution: {integrity: sha512-2dHfToCj/pzca2Ck724OZ5L0EVrr3eHRNsG/b3xQJLA2hZpVCS99bLAX+hm1IHXDEnzU6by5z/5MIY794/a8NQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@rollup/rollup-win32-x64-gnu@4.59.0': + resolution: {integrity: sha512-laBkYlSS1n2L8fSo1thDNGrCTQMmxjYY5G0WFWjFFYZkKPjsMBsgJfGf4TLxXrF6RyhI60L8TMOjBMvXiTcxeA==} + cpu: [x64] + os: [win32] - '@radix-ui/react-use-rect@1.1.1': - resolution: {integrity: sha512-QTYuDesS0VtuHNNvMh+CjlKJ4LJickCMUAqjlE3+j8w+RlRpwyX3apEQKGFzbZGdo7XNG1tXa+bQqIE7HIXT2w==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@rollup/rollup-win32-x64-msvc@4.59.0': + resolution: {integrity: sha512-2HRCml6OztYXyJXAvdDXPKcawukWY2GpR5/nxKp4iBgiO3wcoEGkAaqctIbZcNB6KlUQBIqt8VYkNSj2397EfA==} + cpu: [x64] + os: [win32] - '@radix-ui/react-use-size@1.1.1': - resolution: {integrity: sha512-ewrXRDTAqAXlkl6t/fkXWNAhFX9I+CkKlw6zjEwk86RSPKwZr3xpBRso655aqYafwtnbpHLj6toFzmd6xdVptQ==} - peerDependencies: - '@types/react': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - peerDependenciesMeta: - '@types/react': - optional: true + '@testing-library/dom@10.4.1': + resolution: {integrity: sha512-o4PXJQidqJl82ckFaXUeoAW+XysPLauYI43Abki5hABd853iMhitooc6znOnczgbTYmEP6U6/y1ZyKAIsvMKGg==} + engines: {node: '>=18'} - '@radix-ui/react-visually-hidden@1.2.3': - resolution: {integrity: sha512-pzJq12tEaaIhqjbzpCuv/OypJY/BPavOofm+dbab+MHLajy277+1lLm6JFcGgF5eskJ6mquGirhXY2GD/8u8Ug==} + '@testing-library/jest-dom@6.9.1': + resolution: {integrity: sha512-zIcONa+hVtVSSep9UT3jZ5rizo2BsxgyDYU7WFD5eICBE7no3881HGeb/QkGfsJs6JTkY1aQhT7rIPC7e+0nnA==} + engines: {node: '>=14', npm: '>=6', yarn: '>=1'} + + '@testing-library/react@16.3.2': + resolution: {integrity: sha512-XU5/SytQM+ykqMnAnvB2umaJNIOsLF3PVv//1Ew4CTcpz0/BRyy/af40qqrt7SjKpDdT1saBMc42CUok5gaw+g==} + engines: {node: '>=18'} peerDependencies: - '@types/react': '*' - '@types/react-dom': '*' - react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc - react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + '@testing-library/dom': ^10.0.0 + '@types/react': ^18.0.0 || ^19.0.0 + '@types/react-dom': ^18.0.0 || ^19.0.0 + react: ^18.0.0 || ^19.0.0 + react-dom: ^18.0.0 || ^19.0.0 peerDependenciesMeta: '@types/react': optional: true '@types/react-dom': optional: true - '@radix-ui/rect@1.1.1': - resolution: {integrity: sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw==} + '@testing-library/user-event@14.6.1': + resolution: {integrity: sha512-vq7fv0rnt+QTXgPxr5Hjc210p6YKq2kmdziLgnsZGgLJ9e6VAShx1pACLuRjd/AS/sr7phAR58OIIpf0LlmQNw==} + engines: {node: '>=12', npm: '>=6'} + peerDependencies: + '@testing-library/dom': '>=7.21.4' - '@rtsao/scc@1.1.0': - resolution: {integrity: sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==} + '@types/aria-query@5.0.4': + resolution: {integrity: sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==} - '@rushstack/eslint-patch@1.15.0': - resolution: {integrity: sha512-ojSshQPKwVvSMR8yT2L/QtUkV5SXi/IfDiJ4/8d6UbTPjiHVmxZzUAzGD8Tzks1b9+qQkZa0isUOvYObedITaw==} + '@types/babel__core@7.20.5': + resolution: {integrity: sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==} - '@swc/helpers@0.5.15': - resolution: {integrity: sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==} + '@types/babel__generator@7.27.0': + resolution: {integrity: sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==} - '@tybys/wasm-util@0.10.1': - resolution: {integrity: sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg==} + '@types/babel__template@7.4.4': + resolution: {integrity: sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==} + + '@types/babel__traverse@7.28.0': + resolution: {integrity: sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==} '@types/d3-array@3.2.2': resolution: {integrity: sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw==} @@ -925,9 +1309,6 @@ packages: '@types/json-schema@7.0.15': resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} - '@types/json5@0.0.29': - resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==} - '@types/node@20.19.30': resolution: {integrity: sha512-WJtwWJu7UdlvzEAUm484QNg5eAoq5QR08KDNx7g45Usrs2NtOPiX8ugDqmKdXkyL03rBqU5dYNYVQetEpBHq2g==} @@ -939,159 +1320,40 @@ packages: '@types/react@19.1.0': resolution: {integrity: sha512-UaicktuQI+9UKyA4njtDOGBD/67t8YEBt2xdfqu8+gP9hqPUPsiXlNPcpS2gVdjmis5GKPG3fCxbQLVgxsQZ8w==} - '@typescript-eslint/eslint-plugin@8.54.0': - resolution: {integrity: sha512-hAAP5io/7csFStuOmR782YmTthKBJ9ND3WVL60hcOjvtGFb+HJxH4O5huAcmcZ9v9G8P+JETiZ/G1B8MALnWZQ==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - peerDependencies: - '@typescript-eslint/parser': ^8.54.0 - eslint: ^8.57.0 || ^9.0.0 - typescript: '>=4.8.4 <6.0.0' - - '@typescript-eslint/parser@8.54.0': - resolution: {integrity: sha512-BtE0k6cjwjLZoZixN0t5AKP0kSzlGu7FctRXYuPAm//aaiZhmfq1JwdYpYr1brzEspYyFeF+8XF5j2VK6oalrA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - peerDependencies: - eslint: ^8.57.0 || ^9.0.0 - typescript: '>=4.8.4 <6.0.0' - - '@typescript-eslint/project-service@8.54.0': - resolution: {integrity: sha512-YPf+rvJ1s7MyiWM4uTRhE4DvBXrEV+d8oC3P9Y2eT7S+HBS0clybdMIPnhiATi9vZOYDc7OQ1L/i6ga6NFYK/g==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - peerDependencies: - typescript: '>=4.8.4 <6.0.0' - - '@typescript-eslint/scope-manager@8.54.0': - resolution: {integrity: sha512-27rYVQku26j/PbHYcVfRPonmOlVI6gihHtXFbTdB5sb6qA0wdAQAbyXFVarQ5t4HRojIz64IV90YtsjQSSGlQg==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@typescript-eslint/tsconfig-utils@8.54.0': - resolution: {integrity: sha512-dRgOyT2hPk/JwxNMZDsIXDgyl9axdJI3ogZ2XWhBPsnZUv+hPesa5iuhdYt2gzwA9t8RE5ytOJ6xB0moV0Ujvw==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - peerDependencies: - typescript: '>=4.8.4 <6.0.0' - - '@typescript-eslint/type-utils@8.54.0': - resolution: {integrity: sha512-hiLguxJWHjjwL6xMBwD903ciAwd7DmK30Y9Axs/etOkftC3ZNN9K44IuRD/EB08amu+Zw6W37x9RecLkOo3pMA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@vitejs/plugin-react@4.7.0': + resolution: {integrity: sha512-gUu9hwfWvvEDBBmgtAowQCojwZmJ5mcLn3aufeCsitijs3+f2NsrPtlAWIR6OPiqljl96GVCUbLe0HyqIpVaoA==} + engines: {node: ^14.18.0 || >=16.0.0} peerDependencies: - eslint: ^8.57.0 || ^9.0.0 - typescript: '>=4.8.4 <6.0.0' + vite: ^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 - '@typescript-eslint/types@8.54.0': - resolution: {integrity: sha512-PDUI9R1BVjqu7AUDsRBbKMtwmjWcn4J3le+5LpcFgWULN3LvHC5rkc9gCVxbrsrGmO1jfPybN5s6h4Jy+OnkAA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@typescript-eslint/typescript-estree@8.54.0': - resolution: {integrity: sha512-BUwcskRaPvTk6fzVWgDPdUndLjB87KYDrN5EYGetnktoeAvPtO4ONHlAZDnj5VFnUANg0Sjm7j4usBlnoVMHwA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - peerDependencies: - typescript: '>=4.8.4 <6.0.0' + '@vitest/expect@2.1.9': + resolution: {integrity: sha512-UJCIkTBenHeKT1TTlKMJWy1laZewsRIzYighyYiJKZreqtdxSos/S1t+ktRMQWu2CKqaarrkeszJx1cgC5tGZw==} - '@typescript-eslint/utils@8.54.0': - resolution: {integrity: sha512-9Cnda8GS57AQakvRyG0PTejJNlA2xhvyNtEVIMlDWOOeEyBkYWhGPnfrIAnqxLMTSTo6q8g12XVjjev5l1NvMA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@vitest/mocker@2.1.9': + resolution: {integrity: sha512-tVL6uJgoUdi6icpxmdrn5YNo3g3Dxv+IHJBr0GXHaEdTcw3F+cPKnsXFhli6nO+f/6SDKPHEK1UN+k+TQv0Ehg==} peerDependencies: - eslint: ^8.57.0 || ^9.0.0 - typescript: '>=4.8.4 <6.0.0' - - '@typescript-eslint/visitor-keys@8.54.0': - resolution: {integrity: sha512-VFlhGSl4opC0bprJiItPQ1RfUhGDIBokcPwaFH4yiBCaNPeld/9VeXbiPO1cLyorQi1G1vL+ecBk1x8o1axORA==} - engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} - - '@unrs/resolver-binding-android-arm-eabi@1.11.1': - resolution: {integrity: sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw==} - cpu: [arm] - os: [android] - - '@unrs/resolver-binding-android-arm64@1.11.1': - resolution: {integrity: sha512-lCxkVtb4wp1v+EoN+HjIG9cIIzPkX5OtM03pQYkG+U5O/wL53LC4QbIeazgiKqluGeVEeBlZahHalCaBvU1a2g==} - cpu: [arm64] - os: [android] - - '@unrs/resolver-binding-darwin-arm64@1.11.1': - resolution: {integrity: sha512-gPVA1UjRu1Y/IsB/dQEsp2V1pm44Of6+LWvbLc9SDk1c2KhhDRDBUkQCYVWe6f26uJb3fOK8saWMgtX8IrMk3g==} - cpu: [arm64] - os: [darwin] - - '@unrs/resolver-binding-darwin-x64@1.11.1': - resolution: {integrity: sha512-cFzP7rWKd3lZaCsDze07QX1SC24lO8mPty9vdP+YVa3MGdVgPmFc59317b2ioXtgCMKGiCLxJ4HQs62oz6GfRQ==} - cpu: [x64] - os: [darwin] - - '@unrs/resolver-binding-freebsd-x64@1.11.1': - resolution: {integrity: sha512-fqtGgak3zX4DCB6PFpsH5+Kmt/8CIi4Bry4rb1ho6Av2QHTREM+47y282Uqiu3ZRF5IQioJQ5qWRV6jduA+iGw==} - cpu: [x64] - os: [freebsd] - - '@unrs/resolver-binding-linux-arm-gnueabihf@1.11.1': - resolution: {integrity: sha512-u92mvlcYtp9MRKmP+ZvMmtPN34+/3lMHlyMj7wXJDeXxuM0Vgzz0+PPJNsro1m3IZPYChIkn944wW8TYgGKFHw==} - cpu: [arm] - os: [linux] - - '@unrs/resolver-binding-linux-arm-musleabihf@1.11.1': - resolution: {integrity: sha512-cINaoY2z7LVCrfHkIcmvj7osTOtm6VVT16b5oQdS4beibX2SYBwgYLmqhBjA1t51CarSaBuX5YNsWLjsqfW5Cw==} - cpu: [arm] - os: [linux] - - '@unrs/resolver-binding-linux-arm64-gnu@1.11.1': - resolution: {integrity: sha512-34gw7PjDGB9JgePJEmhEqBhWvCiiWCuXsL9hYphDF7crW7UgI05gyBAi6MF58uGcMOiOqSJ2ybEeCvHcq0BCmQ==} - cpu: [arm64] - os: [linux] - - '@unrs/resolver-binding-linux-arm64-musl@1.11.1': - resolution: {integrity: sha512-RyMIx6Uf53hhOtJDIamSbTskA99sPHS96wxVE/bJtePJJtpdKGXO1wY90oRdXuYOGOTuqjT8ACccMc4K6QmT3w==} - cpu: [arm64] - os: [linux] - - '@unrs/resolver-binding-linux-ppc64-gnu@1.11.1': - resolution: {integrity: sha512-D8Vae74A4/a+mZH0FbOkFJL9DSK2R6TFPC9M+jCWYia/q2einCubX10pecpDiTmkJVUH+y8K3BZClycD8nCShA==} - cpu: [ppc64] - os: [linux] - - '@unrs/resolver-binding-linux-riscv64-gnu@1.11.1': - resolution: {integrity: sha512-frxL4OrzOWVVsOc96+V3aqTIQl1O2TjgExV4EKgRY09AJ9leZpEg8Ak9phadbuX0BA4k8U5qtvMSQQGGmaJqcQ==} - cpu: [riscv64] - os: [linux] - - '@unrs/resolver-binding-linux-riscv64-musl@1.11.1': - resolution: {integrity: sha512-mJ5vuDaIZ+l/acv01sHoXfpnyrNKOk/3aDoEdLO/Xtn9HuZlDD6jKxHlkN8ZhWyLJsRBxfv9GYM2utQ1SChKew==} - cpu: [riscv64] - os: [linux] - - '@unrs/resolver-binding-linux-s390x-gnu@1.11.1': - resolution: {integrity: sha512-kELo8ebBVtb9sA7rMe1Cph4QHreByhaZ2QEADd9NzIQsYNQpt9UkM9iqr2lhGr5afh885d/cB5QeTXSbZHTYPg==} - cpu: [s390x] - os: [linux] - - '@unrs/resolver-binding-linux-x64-gnu@1.11.1': - resolution: {integrity: sha512-C3ZAHugKgovV5YvAMsxhq0gtXuwESUKc5MhEtjBpLoHPLYM+iuwSj3lflFwK3DPm68660rZ7G8BMcwSro7hD5w==} - cpu: [x64] - os: [linux] + msw: ^2.4.9 + vite: ^5.0.0 + peerDependenciesMeta: + msw: + optional: true + vite: + optional: true - '@unrs/resolver-binding-linux-x64-musl@1.11.1': - resolution: {integrity: sha512-rV0YSoyhK2nZ4vEswT/QwqzqQXw5I6CjoaYMOX0TqBlWhojUf8P94mvI7nuJTeaCkkds3QE4+zS8Ko+GdXuZtA==} - cpu: [x64] - os: [linux] + '@vitest/pretty-format@2.1.9': + resolution: {integrity: sha512-KhRIdGV2U9HOUzxfiHmY8IFHTdqtOhIzCpd8WRdJiE7D/HUcZVD0EgQCVjm+Q9gkUXWgBvMmTtZgIG48wq7sOQ==} - '@unrs/resolver-binding-wasm32-wasi@1.11.1': - resolution: {integrity: sha512-5u4RkfxJm+Ng7IWgkzi3qrFOvLvQYnPBmjmZQ8+szTK/b31fQCnleNl1GgEt7nIsZRIf5PLhPwT0WM+q45x/UQ==} - engines: {node: '>=14.0.0'} - cpu: [wasm32] + '@vitest/runner@2.1.9': + resolution: {integrity: sha512-ZXSSqTFIrzduD63btIfEyOmNcBmQvgOVsPNPe0jYtESiXkhd8u2erDLnMxmGrDCwHCCHE7hxwRDCT3pt0esT4g==} - '@unrs/resolver-binding-win32-arm64-msvc@1.11.1': - resolution: {integrity: sha512-nRcz5Il4ln0kMhfL8S3hLkxI85BXs3o8EYoattsJNdsX4YUU89iOkVn7g0VHSRxFuVMdM4Q1jEpIId1Ihim/Uw==} - cpu: [arm64] - os: [win32] + '@vitest/snapshot@2.1.9': + resolution: {integrity: sha512-oBO82rEjsxLNJincVhLhaxxZdEtV0EFHMK5Kmx5sJ6H9L183dHECjiefOAdnqpIgT5eZwT04PoggUnW88vOBNQ==} - '@unrs/resolver-binding-win32-ia32-msvc@1.11.1': - resolution: {integrity: sha512-DCEI6t5i1NmAZp6pFonpD5m7i6aFrpofcp4LA2i8IIq60Jyo28hamKBxNrZcyOwVOZkgsRp9O2sXWBWP8MnvIQ==} - cpu: [ia32] - os: [win32] + '@vitest/spy@2.1.9': + resolution: {integrity: sha512-E1B35FwzXXTs9FHNK6bDszs7mtydNi5MIfUWpceJ8Xbfb1gBMscAnwLbEu+B44ed6W3XjL9/ehLPHR1fkf1KLQ==} - '@unrs/resolver-binding-win32-x64-msvc@1.11.1': - resolution: {integrity: sha512-lrW200hZdbfRtztbygyaq/6jP6AKE8qQN2KvPcJ+x7wiD038YtnYtZ82IMNJ69GJibV7bwL3y9FgK+5w/pYt6g==} - cpu: [x64] - os: [win32] + '@vitest/utils@2.1.9': + resolution: {integrity: sha512-v0psaMSkNJ3A2NMrUEHFRzJtDPFn+/VWZ5WxImB21T9fjucJRmS7xCS3ppEnARb9y11OAzaD+P2Ps+b+BGX5iQ==} acorn-jsx@5.3.2: resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} @@ -1103,6 +1365,10 @@ packages: engines: {node: '>=0.4.0'} hasBin: true + agent-base@7.1.4: + resolution: {integrity: sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==} + engines: {node: '>= 14'} + ajv@6.12.6: resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} @@ -1110,17 +1376,13 @@ packages: resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} engines: {node: '>=8'} - ansi-regex@6.2.2: - resolution: {integrity: sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==} - engines: {node: '>=12'} - ansi-styles@4.3.0: resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} engines: {node: '>=8'} - ansi-styles@6.2.3: - resolution: {integrity: sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==} - engines: {node: '>=12'} + ansi-styles@5.2.0: + resolution: {integrity: sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==} + engines: {node: '>=10'} any-promise@1.3.0: resolution: {integrity: sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==} @@ -1139,64 +1401,28 @@ packages: resolution: {integrity: sha512-ik3ZgC9dY/lYVVM++OISsaYDeg1tb0VtP5uL3ouh1koGOaUMDPpbFIei4JkFimWUFPn90sbMNMXQAIVOlnYKJA==} engines: {node: '>=10'} + aria-query@5.3.0: + resolution: {integrity: sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==} + aria-query@5.3.2: resolution: {integrity: sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==} engines: {node: '>= 0.4'} - array-buffer-byte-length@1.0.2: - resolution: {integrity: sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==} - engines: {node: '>= 0.4'} - - array-includes@3.1.9: - resolution: {integrity: sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==} - engines: {node: '>= 0.4'} - - array.prototype.findlast@1.2.5: - resolution: {integrity: sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==} - engines: {node: '>= 0.4'} - - array.prototype.findlastindex@1.2.6: - resolution: {integrity: sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ==} - engines: {node: '>= 0.4'} - - array.prototype.flat@1.3.3: - resolution: {integrity: sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==} - engines: {node: '>= 0.4'} - - array.prototype.flatmap@1.3.3: - resolution: {integrity: sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==} - engines: {node: '>= 0.4'} - - array.prototype.tosorted@1.1.4: - resolution: {integrity: sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==} - engines: {node: '>= 0.4'} - - arraybuffer.prototype.slice@1.0.4: - resolution: {integrity: sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==} - engines: {node: '>= 0.4'} - - ast-types-flow@0.0.8: - resolution: {integrity: sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==} - - async-function@1.0.0: - resolution: {integrity: sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==} - engines: {node: '>= 0.4'} - - available-typed-arrays@1.0.7: - resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} - engines: {node: '>= 0.4'} - - axe-core@4.11.1: - resolution: {integrity: sha512-BASOg+YwO2C+346x3LZOeoovTIoTrRqEsqMa6fmfAV0P+U9mFr9NsyOEpiYvFjbc64NMrSswhV50WdXzdb/Z5A==} - engines: {node: '>=4'} + assertion-error@2.0.1: + resolution: {integrity: sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==} + engines: {node: '>=12'} - axobject-query@4.1.0: - resolution: {integrity: sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==} - engines: {node: '>= 0.4'} + asynckit@0.4.0: + resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} balanced-match@1.0.2: resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + baseline-browser-mapping@2.10.0: + resolution: {integrity: sha512-lIyg0szRfYbiy67j9KN8IyeD7q7hcmqnJ1ddWmNt19ItGpNN64mnllmxUNFIOdOm6by97jlL6wfpTTJrmnjWAA==} + engines: {node: '>=6.0.0'} + hasBin: true + binary-extensions@2.3.0: resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} engines: {node: '>=8'} @@ -1204,23 +1430,21 @@ packages: brace-expansion@1.1.12: resolution: {integrity: sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==} - brace-expansion@2.0.2: - resolution: {integrity: sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==} - braces@3.0.3: resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} engines: {node: '>=8'} - call-bind-apply-helpers@1.0.2: - resolution: {integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==} - engines: {node: '>= 0.4'} + browserslist@4.28.1: + resolution: {integrity: sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==} + engines: {node: ^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7} + hasBin: true - call-bind@1.0.8: - resolution: {integrity: sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==} - engines: {node: '>= 0.4'} + cac@6.7.14: + resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} + engines: {node: '>=8'} - call-bound@1.0.4: - resolution: {integrity: sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==} + call-bind-apply-helpers@1.0.2: + resolution: {integrity: sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==} engines: {node: '>= 0.4'} callsites@3.1.0: @@ -1234,10 +1458,18 @@ packages: caniuse-lite@1.0.30001766: resolution: {integrity: sha512-4C0lfJ0/YPjJQHagaE9x2Elb69CIqEPZeG0anQt9SIvIoOH4a4uaRl73IavyO+0qZh6MDLH//DrXThEYKHkmYA==} + chai@5.3.3: + resolution: {integrity: sha512-4zNhdJD/iOjSH0A05ea+Ke6MU5mmpQcbQsSOkgdaUMJ9zTlDTD/GYlwohmIE2u0gaxHYiVHEn1Fw9mZ/ktJWgw==} + engines: {node: '>=18'} + chalk@4.1.2: resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} engines: {node: '>=10'} + check-error@2.1.3: + resolution: {integrity: sha512-PAJdDJusoxnwm1VwW07VWwUN1sl7smmC3OKggvndJFadxxDRyFJBX/ggnu/KE4kQAB7a3Dp8f/YXC1FlUprWmA==} + engines: {node: '>= 16'} + chokidar@3.6.0: resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} engines: {node: '>= 8.10.0'} @@ -1245,9 +1477,6 @@ packages: class-variance-authority@0.7.1: resolution: {integrity: sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg==} - client-only@0.0.1: - resolution: {integrity: sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==} - clsx@2.1.1: resolution: {integrity: sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==} engines: {node: '>=6'} @@ -1259,6 +1488,10 @@ packages: color-name@1.1.4: resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + combined-stream@1.0.8: + resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} + engines: {node: '>= 0.8'} + commander@4.1.1: resolution: {integrity: sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==} engines: {node: '>= 6'} @@ -1266,6 +1499,9 @@ packages: concat-map@0.0.1: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + convert-source-map@2.0.0: + resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==} + copy-to-clipboard@3.3.3: resolution: {integrity: sha512-2KV8NhB5JqC3ky0r9PMCAZKbUHSwtEo4CwCs0KXgruG43gX5PMqDEBbVU4OUzw2MuAWUfsuFmWvEKG5QRfSnJA==} @@ -1273,11 +1509,18 @@ packages: resolution: {integrity: sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==} engines: {node: '>= 8'} + css.escape@1.5.1: + resolution: {integrity: sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg==} + cssesc@3.0.0: resolution: {integrity: sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==} engines: {node: '>=4'} hasBin: true + cssstyle@4.6.0: + resolution: {integrity: sha512-2z+rWdzbbSZv6/rhtvzvqeZQHrBaqgogqt85sqFNbabZOuFbCVFb8kPeEtZjiKkbrm395irpNKiYeFeLiQnFPg==} + engines: {node: '>=18'} + csstype@3.2.3: resolution: {integrity: sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==} @@ -1325,20 +1568,9 @@ packages: resolution: {integrity: sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==} engines: {node: '>=12'} - damerau-levenshtein@1.0.8: - resolution: {integrity: sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==} - - data-view-buffer@1.0.2: - resolution: {integrity: sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==} - engines: {node: '>= 0.4'} - - data-view-byte-length@1.0.2: - resolution: {integrity: sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==} - engines: {node: '>= 0.4'} - - data-view-byte-offset@1.0.1: - resolution: {integrity: sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==} - engines: {node: '>= 0.4'} + data-urls@5.0.0: + resolution: {integrity: sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg==} + engines: {node: '>=18'} date-fns-jalali@4.1.0-0: resolution: {integrity: sha512-hTIP/z+t+qKwBDcmmsnmjWTduxCg+5KfdqWQvb2X/8C9+knYY6epN/pfxdDuyVlSVeFz0sM5eEfwIUQ70U4ckg==} @@ -1349,14 +1581,6 @@ packages: date-fns@4.1.0: resolution: {integrity: sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==} - debug@3.2.7: - resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==} - peerDependencies: - supports-color: '*' - peerDependenciesMeta: - supports-color: - optional: true - debug@4.4.3: resolution: {integrity: sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==} engines: {node: '>=6.0'} @@ -1369,25 +1593,24 @@ packages: decimal.js-light@2.5.1: resolution: {integrity: sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg==} + decimal.js@10.6.0: + resolution: {integrity: sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==} + + deep-eql@5.0.2: + resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} + engines: {node: '>=6'} + deep-is@0.1.4: resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} - define-data-property@1.1.4: - resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} - engines: {node: '>= 0.4'} - - define-properties@1.2.1: - resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} - engines: {node: '>= 0.4'} + delayed-stream@1.0.0: + resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} + engines: {node: '>=0.4.0'} dequal@2.0.3: resolution: {integrity: sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==} engines: {node: '>=6'} - detect-libc@2.1.2: - resolution: {integrity: sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==} - engines: {node: '>=8'} - detect-node-es@1.1.0: resolution: {integrity: sha512-ypdmJU/TbBby2Dxibuv7ZLW3Bs1QEmM7nHjEANfohJLvE0XVujisn1qPJcZxg+qDucsr+bP6fLD1rPS3AhJ7EQ==} @@ -1397,9 +1620,11 @@ packages: dlv@1.1.3: resolution: {integrity: sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==} - doctrine@2.1.0: - resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==} - engines: {node: '>=0.10.0'} + dom-accessibility-api@0.5.16: + resolution: {integrity: sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==} + + dom-accessibility-api@0.6.3: + resolution: {integrity: sha512-7ZgogeTnjuHbo+ct10G9Ffp0mif17idi0IyWNVA/wcwcm7NPOD/WEHVP3n7n3MhXqxoIYm8d6MuZohYWIZ4T3w==} dom-helpers@5.2.1: resolution: {integrity: sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==} @@ -1408,18 +1633,12 @@ packages: resolution: {integrity: sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==} engines: {node: '>= 0.4'} - eastasianwidth@0.2.0: - resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} + electron-to-chromium@1.5.302: + resolution: {integrity: sha512-sM6HAN2LyK82IyPBpznDRqlTQAtuSaO+ShzFiWTvoMJLHyZ+Y39r8VMfHzwbU8MVBzQ4Wdn85+wlZl2TLGIlwg==} - emoji-regex@8.0.0: - resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} - - emoji-regex@9.2.2: - resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} - - es-abstract@1.24.1: - resolution: {integrity: sha512-zHXBLhP+QehSSbsS9Pt23Gg964240DPd6QCf8WpkqEXxQ7fhdZzYsocOr5u7apWonsS5EjZDmTF+/slGMyasvw==} - engines: {node: '>= 0.4'} + entities@6.0.1: + resolution: {integrity: sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g==} + engines: {node: '>=0.12'} es-define-property@1.0.1: resolution: {integrity: sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==} @@ -1429,109 +1648,40 @@ packages: resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} engines: {node: '>= 0.4'} - es-iterator-helpers@1.2.2: - resolution: {integrity: sha512-BrUQ0cPTB/IwXj23HtwHjS9n7O4h9FX94b4xc5zlTHxeLgTAdzYUDyy6KdExAl9lbN5rtfe44xpjpmj9grxs5w==} - engines: {node: '>= 0.4'} + es-module-lexer@1.7.0: + resolution: {integrity: sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==} es-object-atoms@1.1.1: resolution: {integrity: sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==} engines: {node: '>= 0.4'} es-set-tostringtag@2.1.0: - resolution: {integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==} - engines: {node: '>= 0.4'} - - es-shim-unscopables@1.1.0: - resolution: {integrity: sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==} - engines: {node: '>= 0.4'} - - es-to-primitive@1.3.0: - resolution: {integrity: sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==} - engines: {node: '>= 0.4'} - - escape-string-regexp@4.0.0: - resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} - engines: {node: '>=10'} - - eslint-config-next@15.2.4: - resolution: {integrity: sha512-v4gYjd4eYIme8qzaJItpR5MMBXJ0/YV07u7eb50kEnlEmX7yhOjdUdzz70v4fiINYRjLf8X8TbogF0k7wlz6sA==} - peerDependencies: - eslint: ^7.23.0 || ^8.0.0 || ^9.0.0 - typescript: '>=3.3.1' - peerDependenciesMeta: - typescript: - optional: true - - eslint-config-prettier@9.1.2: - resolution: {integrity: sha512-iI1f+D2ViGn+uvv5HuHVUamg8ll4tN+JRHGc6IJi4TP9Kl976C57fzPXgseXNs8v0iA8aSJpHsTWjDb9QJamGQ==} - hasBin: true - peerDependencies: - eslint: '>=7.0.0' - - eslint-import-resolver-node@0.3.9: - resolution: {integrity: sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==} - - eslint-import-resolver-typescript@3.10.1: - resolution: {integrity: sha512-A1rHYb06zjMGAxdLSkN2fXPBwuSaQ0iO5M/hdyS0Ajj1VBaRp0sPD3dn1FhME3c/JluGFbwSxyCfqdSbtQLAHQ==} - engines: {node: ^14.18.0 || >=16.0.0} - peerDependencies: - eslint: '*' - eslint-plugin-import: '*' - eslint-plugin-import-x: '*' - peerDependenciesMeta: - eslint-plugin-import: - optional: true - eslint-plugin-import-x: - optional: true + resolution: {integrity: sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==} + engines: {node: '>= 0.4'} - eslint-module-utils@2.12.1: - resolution: {integrity: sha512-L8jSWTze7K2mTg0vos/RuLRS5soomksDPoJLXIslC7c8Wmut3bx7CPpJijDcBZtxQ5lrbUdM+s0OlNbz0DCDNw==} - engines: {node: '>=4'} - peerDependencies: - '@typescript-eslint/parser': '*' - eslint: '*' - eslint-import-resolver-node: '*' - eslint-import-resolver-typescript: '*' - eslint-import-resolver-webpack: '*' - peerDependenciesMeta: - '@typescript-eslint/parser': - optional: true - eslint: - optional: true - eslint-import-resolver-node: - optional: true - eslint-import-resolver-typescript: - optional: true - eslint-import-resolver-webpack: - optional: true + esbuild@0.21.5: + resolution: {integrity: sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==} + engines: {node: '>=12'} + hasBin: true - eslint-plugin-import@2.32.0: - resolution: {integrity: sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA==} - engines: {node: '>=4'} - peerDependencies: - '@typescript-eslint/parser': '*' - eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9 - peerDependenciesMeta: - '@typescript-eslint/parser': - optional: true + esbuild@0.25.12: + resolution: {integrity: sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==} + engines: {node: '>=18'} + hasBin: true - eslint-plugin-jsx-a11y@6.10.2: - resolution: {integrity: sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q==} - engines: {node: '>=4.0'} - peerDependencies: - eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9 + escalade@3.2.0: + resolution: {integrity: sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==} + engines: {node: '>=6'} - eslint-plugin-react-hooks@5.2.0: - resolution: {integrity: sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg==} + escape-string-regexp@4.0.0: + resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} engines: {node: '>=10'} - peerDependencies: - eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 - eslint-plugin-react@7.37.5: - resolution: {integrity: sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA==} - engines: {node: '>=4'} + eslint-config-prettier@9.1.2: + resolution: {integrity: sha512-iI1f+D2ViGn+uvv5HuHVUamg8ll4tN+JRHGc6IJi4TP9Kl976C57fzPXgseXNs8v0iA8aSJpHsTWjDb9QJamGQ==} + hasBin: true peerDependencies: - eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7 + eslint: '>=7.0.0' eslint-scope@8.4.0: resolution: {integrity: sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==} @@ -1571,6 +1721,9 @@ packages: resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} engines: {node: '>=4.0'} + estree-walker@3.0.3: + resolution: {integrity: sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==} + esutils@2.0.3: resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} engines: {node: '>=0.10.0'} @@ -1578,6 +1731,10 @@ packages: eventemitter3@4.0.7: resolution: {integrity: sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==} + expect-type@1.3.0: + resolution: {integrity: sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==} + engines: {node: '>=12.0.0'} + fast-deep-equal@3.1.3: resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} @@ -1585,10 +1742,6 @@ packages: resolution: {integrity: sha512-jt2DW/aNFNwke7AUd+Z+e6pz39KO5rzdbbFCg2sGafS4mk13MI7Z8O5z9cADNn5lhGODIgLwug6TZO2ctf7kcw==} engines: {node: '>=6.0.0'} - fast-glob@3.3.1: - resolution: {integrity: sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==} - engines: {node: '>=8.6.0'} - fast-glob@3.3.3: resolution: {integrity: sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==} engines: {node: '>=8.6.0'} @@ -1630,13 +1783,9 @@ packages: flatted@3.3.3: resolution: {integrity: sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==} - for-each@0.3.5: - resolution: {integrity: sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==} - engines: {node: '>= 0.4'} - - foreground-child@3.3.1: - resolution: {integrity: sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==} - engines: {node: '>=14'} + form-data@4.0.5: + resolution: {integrity: sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==} + engines: {node: '>= 6'} fsevents@2.3.3: resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} @@ -1646,16 +1795,9 @@ packages: function-bind@1.1.2: resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - function.prototype.name@1.1.8: - resolution: {integrity: sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==} - engines: {node: '>= 0.4'} - - functions-have-names@1.2.3: - resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==} - - generator-function@2.0.1: - resolution: {integrity: sha512-SFdFmIJi+ybC0vjlHN0ZGVGHc3lgE0DxPAT0djjVg+kjOnSqclqmj0KQ7ykTOLP6YxoqOvuAODGdcHJn+43q3g==} - engines: {node: '>= 0.4'} + gensync@1.0.0-beta.2: + resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} + engines: {node: '>=6.9.0'} get-intrinsic@1.3.0: resolution: {integrity: sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==} @@ -1669,13 +1811,6 @@ packages: resolution: {integrity: sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==} engines: {node: '>= 0.4'} - get-symbol-description@1.1.0: - resolution: {integrity: sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==} - engines: {node: '>= 0.4'} - - get-tsconfig@4.13.1: - resolution: {integrity: sha512-EoY1N2xCn44xU6750Sx7OjOIT59FkmstNc3X6y5xpz7D5cBtZRe/3pSlTkDJgqsOk3WwZPkWfonhhUJfttQo3w==} - glob-parent@5.1.2: resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} engines: {node: '>= 6'} @@ -1684,38 +1819,18 @@ packages: resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} engines: {node: '>=10.13.0'} - glob@11.1.0: - resolution: {integrity: sha512-vuNwKSaKiqm7g0THUBu2x7ckSs3XJLXE+2ssL7/MfTGPLLcrJQ/4Uq1CjPTtO5cCIiRxqvN6Twy1qOwhL0Xjcw==} - engines: {node: 20 || >=22} - hasBin: true - globals@14.0.0: resolution: {integrity: sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==} engines: {node: '>=18'} - globalthis@1.0.4: - resolution: {integrity: sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==} - engines: {node: '>= 0.4'} - gopd@1.2.0: resolution: {integrity: sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==} engines: {node: '>= 0.4'} - has-bigints@1.1.0: - resolution: {integrity: sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==} - engines: {node: '>= 0.4'} - has-flag@4.0.0: resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} engines: {node: '>=8'} - has-property-descriptors@1.0.2: - resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} - - has-proto@1.2.0: - resolution: {integrity: sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==} - engines: {node: '>= 0.4'} - has-symbols@1.1.0: resolution: {integrity: sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==} engines: {node: '>= 0.4'} @@ -1728,14 +1843,26 @@ packages: resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} engines: {node: '>= 0.4'} + html-encoding-sniffer@4.0.0: + resolution: {integrity: sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ==} + engines: {node: '>=18'} + + http-proxy-agent@7.0.2: + resolution: {integrity: sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==} + engines: {node: '>= 14'} + + https-proxy-agent@7.0.6: + resolution: {integrity: sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==} + engines: {node: '>= 14'} + + iconv-lite@0.6.3: + resolution: {integrity: sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==} + engines: {node: '>=0.10.0'} + ignore@5.3.2: resolution: {integrity: sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==} engines: {node: '>= 4'} - ignore@7.0.5: - resolution: {integrity: sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==} - engines: {node: '>= 4'} - import-fresh@3.3.1: resolution: {integrity: sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==} engines: {node: '>=6'} @@ -1744,139 +1871,40 @@ packages: resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} engines: {node: '>=0.8.19'} - internal-slot@1.1.0: - resolution: {integrity: sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==} - engines: {node: '>= 0.4'} + indent-string@4.0.0: + resolution: {integrity: sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==} + engines: {node: '>=8'} internmap@2.0.3: resolution: {integrity: sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==} engines: {node: '>=12'} - is-array-buffer@3.0.5: - resolution: {integrity: sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==} - engines: {node: '>= 0.4'} - - is-async-function@2.1.1: - resolution: {integrity: sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==} - engines: {node: '>= 0.4'} - - is-bigint@1.1.0: - resolution: {integrity: sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==} - engines: {node: '>= 0.4'} - is-binary-path@2.1.0: resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} engines: {node: '>=8'} - is-boolean-object@1.2.2: - resolution: {integrity: sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==} - engines: {node: '>= 0.4'} - - is-bun-module@2.0.0: - resolution: {integrity: sha512-gNCGbnnnnFAUGKeZ9PdbyeGYJqewpmc2aKHUEMO5nQPWU9lOmv7jcmQIv+qHD8fXW6W7qfuCwX4rY9LNRjXrkQ==} - - is-callable@1.2.7: - resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} - engines: {node: '>= 0.4'} - is-core-module@2.16.1: resolution: {integrity: sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==} engines: {node: '>= 0.4'} - is-data-view@1.0.2: - resolution: {integrity: sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==} - engines: {node: '>= 0.4'} - - is-date-object@1.1.0: - resolution: {integrity: sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==} - engines: {node: '>= 0.4'} - is-extglob@2.1.1: resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} engines: {node: '>=0.10.0'} - is-finalizationregistry@1.1.1: - resolution: {integrity: sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==} - engines: {node: '>= 0.4'} - - is-fullwidth-code-point@3.0.0: - resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} - engines: {node: '>=8'} - - is-generator-function@1.1.2: - resolution: {integrity: sha512-upqt1SkGkODW9tsGNG5mtXTXtECizwtS2kA161M+gJPc1xdb/Ax629af6YrTwcOeQHbewrPNlE5Dx7kzvXTizA==} - engines: {node: '>= 0.4'} - is-glob@4.0.3: resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} engines: {node: '>=0.10.0'} - is-map@2.0.3: - resolution: {integrity: sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==} - engines: {node: '>= 0.4'} - - is-negative-zero@2.0.3: - resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} - engines: {node: '>= 0.4'} - - is-number-object@1.1.1: - resolution: {integrity: sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==} - engines: {node: '>= 0.4'} - is-number@7.0.0: resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} engines: {node: '>=0.12.0'} - is-regex@1.2.1: - resolution: {integrity: sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==} - engines: {node: '>= 0.4'} - - is-set@2.0.3: - resolution: {integrity: sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==} - engines: {node: '>= 0.4'} - - is-shared-array-buffer@1.0.4: - resolution: {integrity: sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==} - engines: {node: '>= 0.4'} - - is-string@1.1.1: - resolution: {integrity: sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==} - engines: {node: '>= 0.4'} - - is-symbol@1.1.1: - resolution: {integrity: sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==} - engines: {node: '>= 0.4'} - - is-typed-array@1.1.15: - resolution: {integrity: sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==} - engines: {node: '>= 0.4'} - - is-weakmap@2.0.2: - resolution: {integrity: sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==} - engines: {node: '>= 0.4'} - - is-weakref@1.1.1: - resolution: {integrity: sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==} - engines: {node: '>= 0.4'} - - is-weakset@2.0.4: - resolution: {integrity: sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==} - engines: {node: '>= 0.4'} - - isarray@2.0.5: - resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + is-potential-custom-element-name@1.0.1: + resolution: {integrity: sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==} isexe@2.0.0: resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} - iterator.prototype@1.1.5: - resolution: {integrity: sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g==} - engines: {node: '>= 0.4'} - - jackspeak@4.1.1: - resolution: {integrity: sha512-zptv57P3GpL+O0I7VdMJNBZCu+BPHVQUk55Ft8/QCJjTVxrnJHuVuX/0Bl2A6/+2oyR/ZMEuFKwmzqqZ/U5nPQ==} - engines: {node: 20 || >=22} - jiti@1.21.7: resolution: {integrity: sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==} hasBin: true @@ -1888,6 +1916,20 @@ packages: resolution: {integrity: sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==} hasBin: true + jsdom@25.0.1: + resolution: {integrity: sha512-8i7LzZj7BF8uplX+ZyOlIz86V6TAsSs+np6m1kpW9u0JWi4z/1t+FzcK1aek+ybTnAC4KhBL4uXCNT0wcUIeCw==} + engines: {node: '>=18'} + peerDependencies: + canvas: ^2.11.2 + peerDependenciesMeta: + canvas: + optional: true + + jsesc@3.1.0: + resolution: {integrity: sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==} + engines: {node: '>=6'} + hasBin: true + json-buffer@3.0.1: resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} @@ -1897,24 +1939,14 @@ packages: json-stable-stringify-without-jsonify@1.0.1: resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} - json5@1.0.2: - resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==} + json5@2.2.3: + resolution: {integrity: sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==} + engines: {node: '>=6'} hasBin: true - jsx-ast-utils@3.3.5: - resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==} - engines: {node: '>=4.0'} - keyv@4.5.4: resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} - language-subtag-registry@0.3.23: - resolution: {integrity: sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ==} - - language-tags@1.0.9: - resolution: {integrity: sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA==} - engines: {node: '>=0.10'} - levn@0.4.1: resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} engines: {node: '>= 0.8.0'} @@ -1940,15 +1972,27 @@ packages: resolution: {integrity: sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==} hasBin: true - lru-cache@11.2.5: - resolution: {integrity: sha512-vFrFJkWtJvJnD5hg+hJvVE8Lh/TcMzKnTgCWmtBipwI5yLX/iX+5UB2tfuyODF5E7k9xEzMdYgGqaSb1c0c5Yw==} - engines: {node: 20 || >=22} + loupe@3.2.1: + resolution: {integrity: sha512-CdzqowRJCeLU72bHvWqwRBBlLcMEtIvGrlvef74kMnV2AolS9Y8xUv1I0U/MNAWMhBlKIoyuEgoJ0t/bbwHbLQ==} + + lru-cache@10.4.3: + resolution: {integrity: sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==} + + lru-cache@5.1.1: + resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} lucide-react@0.411.0: resolution: {integrity: sha512-bDRvLt/jIIjsq4JVYB3EjyOtLHu8uQGzv7usri2DnVpOtfIRuLln96srS+d8WJsmJ52LBwDnYx7me/TSjZ6AcA==} peerDependencies: react: ^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0 + lz-string@1.5.0: + resolution: {integrity: sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==} + hasBin: true + + magic-string@0.30.21: + resolution: {integrity: sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==} + math-intrinsics@1.1.0: resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==} engines: {node: '>= 0.4'} @@ -1961,23 +2005,20 @@ packages: resolution: {integrity: sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==} engines: {node: '>=8.6'} - minimatch@10.1.1: - resolution: {integrity: sha512-enIvLvRAFZYXJzkCYG5RKmPfrFArdLv+R+lbQ53BmIMLIry74bjKzX6iHAm8WYamJkhSSEabrWN5D97XnKObjQ==} - engines: {node: 20 || >=22} - - minimatch@3.1.2: - resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + mime-db@1.52.0: + resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} + engines: {node: '>= 0.6'} - minimatch@9.0.5: - resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} - engines: {node: '>=16 || 14 >=14.17'} + mime-types@2.1.35: + resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} + engines: {node: '>= 0.6'} - minimist@1.2.8: - resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + min-indent@1.0.1: + resolution: {integrity: sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==} + engines: {node: '>=4'} - minipass@7.1.2: - resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} - engines: {node: '>=16 || 14 >=14.17'} + minimatch@3.1.2: + resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} ms@2.1.3: resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} @@ -1990,45 +2031,19 @@ packages: engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} hasBin: true - napi-postinstall@0.3.4: - resolution: {integrity: sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ==} - engines: {node: ^12.20.0 || ^14.18.0 || >=16.0.0} - hasBin: true - natural-compare@1.4.0: resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} - next-themes@0.4.6: - resolution: {integrity: sha512-pZvgD5L0IEvX5/9GWyHMf3m8BKiVQwsCMHfoFosXtXBMnaS0ZnIJ9ST4b4NqLVKDEm8QBxoNNGNaBv2JNF6XNA==} - peerDependencies: - react: ^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc - react-dom: ^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc - - next@15.4.10: - resolution: {integrity: sha512-itVlc79QjpKMFMRhP+kbGKaSG/gZM6RCvwhEbwmCNF06CdDiNaoHcbeg0PqkEa2GOcn8KJ0nnc7+yL7EjoYLHQ==} - engines: {node: ^18.18.0 || ^19.8.0 || >= 20.0.0} - hasBin: true - peerDependencies: - '@opentelemetry/api': ^1.1.0 - '@playwright/test': ^1.51.1 - babel-plugin-react-compiler: '*' - react: ^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0 - react-dom: ^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0 - sass: ^1.3.0 - peerDependenciesMeta: - '@opentelemetry/api': - optional: true - '@playwright/test': - optional: true - babel-plugin-react-compiler: - optional: true - sass: - optional: true + node-releases@2.0.27: + resolution: {integrity: sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==} normalize-path@3.0.0: resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} engines: {node: '>=0.10.0'} + nwsapi@2.2.23: + resolution: {integrity: sha512-7wfH4sLbt4M0gCDzGE6vzQBo0bfTKjU7Sfpqy/7gs1qBfYz2vEJH6vXcBKpO3+6Yu1telwd0t9HpyOoLEQQbIQ==} + object-assign@4.1.1: resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} engines: {node: '>=0.10.0'} @@ -2037,42 +2052,10 @@ packages: resolution: {integrity: sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==} engines: {node: '>= 6'} - object-inspect@1.13.4: - resolution: {integrity: sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==} - engines: {node: '>= 0.4'} - - object-keys@1.1.1: - resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==} - engines: {node: '>= 0.4'} - - object.assign@4.1.7: - resolution: {integrity: sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==} - engines: {node: '>= 0.4'} - - object.entries@1.1.9: - resolution: {integrity: sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw==} - engines: {node: '>= 0.4'} - - object.fromentries@2.0.8: - resolution: {integrity: sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==} - engines: {node: '>= 0.4'} - - object.groupby@1.0.3: - resolution: {integrity: sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==} - engines: {node: '>= 0.4'} - - object.values@1.2.1: - resolution: {integrity: sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==} - engines: {node: '>= 0.4'} - optionator@0.9.4: resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} engines: {node: '>= 0.8.0'} - own-keys@1.0.1: - resolution: {integrity: sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==} - engines: {node: '>= 0.4'} - p-limit@3.1.0: resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} engines: {node: '>=10'} @@ -2081,13 +2064,13 @@ packages: resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} engines: {node: '>=10'} - package-json-from-dist@1.0.1: - resolution: {integrity: sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==} - parent-module@1.0.1: resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} engines: {node: '>=6'} + parse5@7.3.0: + resolution: {integrity: sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw==} + path-exists@4.0.0: resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} engines: {node: '>=8'} @@ -2099,9 +2082,12 @@ packages: path-parse@1.0.7: resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} - path-scurry@2.0.1: - resolution: {integrity: sha512-oWyT4gICAu+kaA7QWk/jvCHWarMKNs6pXOGWKDTr7cw4IGcUbW+PeTfbaQiLGheFRpjo6O9J0PmyMfQPjH71oA==} - engines: {node: 20 || >=22} + pathe@1.1.2: + resolution: {integrity: sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ==} + + pathval@2.0.1: + resolution: {integrity: sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ==} + engines: {node: '>= 14.16'} picocolors@1.1.1: resolution: {integrity: sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==} @@ -2122,10 +2108,6 @@ packages: resolution: {integrity: sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==} engines: {node: '>= 6'} - possible-typed-array-names@1.1.0: - resolution: {integrity: sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==} - engines: {node: '>= 0.4'} - postcss-import@15.1.0: resolution: {integrity: sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==} engines: {node: '>=14.0.0'} @@ -2169,10 +2151,6 @@ packages: postcss-value-parser@4.2.0: resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} - postcss@8.4.31: - resolution: {integrity: sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==} - engines: {node: ^10 || ^12 || >=14} - postcss@8.5.6: resolution: {integrity: sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==} engines: {node: ^10 || ^12 || >=14} @@ -2186,6 +2164,10 @@ packages: engines: {node: '>=14'} hasBin: true + pretty-format@27.5.1: + resolution: {integrity: sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==} + engines: {node: ^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0} + prop-types@15.8.1: resolution: {integrity: sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==} @@ -2210,9 +2192,16 @@ packages: react-is@16.13.1: resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==} + react-is@17.0.2: + resolution: {integrity: sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==} + react-is@18.3.1: resolution: {integrity: sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==} + react-refresh@0.17.0: + resolution: {integrity: sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==} + engines: {node: '>=0.10.0'} + react-remove-scroll-bar@2.3.8: resolution: {integrity: sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q==} engines: {node: '>=10'} @@ -2233,6 +2222,19 @@ packages: '@types/react': optional: true + react-router-dom@6.30.3: + resolution: {integrity: sha512-pxPcv1AczD4vso7G4Z3TKcvlxK7g7TNt3/FNGMhfqyntocvYKj+GCatfigGDjbLozC4baguJ0ReCigoDJXb0ag==} + engines: {node: '>=14.0.0'} + peerDependencies: + react: '>=16.8' + react-dom: '>=16.8' + + react-router@6.30.3: + resolution: {integrity: sha512-XRnlbKMTmktBkjCLE8/XcZFlnHvr2Ltdr1eJX4idL55/9BbORzyZEaIkBFDhFGCEWBBItsVrDxwx3gnisMitdw==} + engines: {node: '>=14.0.0'} + peerDependencies: + react: '>=16.8' + react-smooth@4.0.4: resolution: {integrity: sha512-gnGKTpYwqL0Iii09gHobNolvX4Kiq4PKx6eWBCYYix+8cdw+cGo3do906l1NBPKkSWx1DghC1dlWG9L2uGd61Q==} peerDependencies: @@ -2276,48 +2278,43 @@ packages: react: ^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 react-dom: ^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 - reflect.getprototypeof@1.0.10: - resolution: {integrity: sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==} - engines: {node: '>= 0.4'} - - regexp.prototype.flags@1.5.4: - resolution: {integrity: sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==} - engines: {node: '>= 0.4'} + redent@3.0.0: + resolution: {integrity: sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==} + engines: {node: '>=8'} resolve-from@4.0.0: resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} engines: {node: '>=4'} - resolve-pkg-maps@1.0.0: - resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} - resolve@1.22.11: resolution: {integrity: sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==} engines: {node: '>= 0.4'} hasBin: true - resolve@2.0.0-next.5: - resolution: {integrity: sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==} - hasBin: true - reusify@1.1.0: resolution: {integrity: sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==} engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + rollup@4.59.0: + resolution: {integrity: sha512-2oMpl67a3zCH9H79LeMcbDhXW/UmWG/y2zuqnF2jQq5uq9TbM9TVyXvA4+t+ne2IIkBdrLpAaRQAvo7YI/Yyeg==} + engines: {node: '>=18.0.0', npm: '>=8.0.0'} + hasBin: true + + rrweb-cssom@0.7.1: + resolution: {integrity: sha512-TrEMa7JGdVm0UThDJSx7ddw5nVm3UJS9o9CCIZ72B1vSyEZoziDqBYP3XIoi/12lKrJR8rE3jeFHMok2F/Mnsg==} + + rrweb-cssom@0.8.0: + resolution: {integrity: sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw==} + run-parallel@1.2.0: resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} - safe-array-concat@1.1.3: - resolution: {integrity: sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==} - engines: {node: '>=0.4'} - - safe-push-apply@1.0.0: - resolution: {integrity: sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==} - engines: {node: '>= 0.4'} + safer-buffer@2.1.2: + resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} - safe-regex-test@1.1.0: - resolution: {integrity: sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==} - engines: {node: '>= 0.4'} + saxes@6.0.0: + resolution: {integrity: sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==} + engines: {node: '>=v12.22.7'} scheduler@0.26.0: resolution: {integrity: sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA==} @@ -2326,27 +2323,6 @@ packages: resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} hasBin: true - semver@7.7.3: - resolution: {integrity: sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==} - engines: {node: '>=10'} - hasBin: true - - set-function-length@1.2.2: - resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} - engines: {node: '>= 0.4'} - - set-function-name@2.0.2: - resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==} - engines: {node: '>= 0.4'} - - set-proto@1.0.0: - resolution: {integrity: sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==} - engines: {node: '>= 0.4'} - - sharp@0.34.5: - resolution: {integrity: sha512-Ou9I5Ft9WNcCbXrU9cMgPBcCK8LiwLqcbywW3t4oDV37n1pzpuNLsYiAV8eODnjbtQlSDwZ2cUEeQz4E54Hltg==} - engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} - shebang-command@2.0.0: resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} engines: {node: '>=8'} @@ -2355,25 +2331,8 @@ packages: resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} engines: {node: '>=8'} - side-channel-list@1.0.0: - resolution: {integrity: sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==} - engines: {node: '>= 0.4'} - - side-channel-map@1.0.1: - resolution: {integrity: sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==} - engines: {node: '>= 0.4'} - - side-channel-weakmap@1.0.2: - resolution: {integrity: sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==} - engines: {node: '>= 0.4'} - - side-channel@1.1.0: - resolution: {integrity: sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==} - engines: {node: '>= 0.4'} - - signal-exit@4.1.0: - resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} - engines: {node: '>=14'} + siginfo@2.0.0: + resolution: {integrity: sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==} sonner@2.0.7: resolution: {integrity: sha512-W6ZN4p58k8aDKA4XPcx2hpIQXBRAgyiWVkYhT7CvK6D3iAu7xjvVyhQHg2/iaKJZ1XVJ4r7XuwGL+WGEK37i9w==} @@ -2385,73 +2344,20 @@ packages: resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} engines: {node: '>=0.10.0'} - stable-hash@0.0.5: - resolution: {integrity: sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA==} - - stop-iteration-iterator@1.1.0: - resolution: {integrity: sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==} - engines: {node: '>= 0.4'} - - string-width@4.2.3: - resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} - engines: {node: '>=8'} - - string-width@5.1.2: - resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} - engines: {node: '>=12'} - - string.prototype.includes@2.0.1: - resolution: {integrity: sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg==} - engines: {node: '>= 0.4'} - - string.prototype.matchall@4.0.12: - resolution: {integrity: sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA==} - engines: {node: '>= 0.4'} - - string.prototype.repeat@1.0.0: - resolution: {integrity: sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==} - - string.prototype.trim@1.2.10: - resolution: {integrity: sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==} - engines: {node: '>= 0.4'} - - string.prototype.trimend@1.0.9: - resolution: {integrity: sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==} - engines: {node: '>= 0.4'} + stackback@0.0.2: + resolution: {integrity: sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==} - string.prototype.trimstart@1.0.8: - resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==} - engines: {node: '>= 0.4'} + std-env@3.10.0: + resolution: {integrity: sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==} - strip-ansi@6.0.1: - resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} + strip-indent@3.0.0: + resolution: {integrity: sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==} engines: {node: '>=8'} - strip-ansi@7.1.2: - resolution: {integrity: sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA==} - engines: {node: '>=12'} - - strip-bom@3.0.0: - resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==} - engines: {node: '>=4'} - strip-json-comments@3.1.1: resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} engines: {node: '>=8'} - styled-jsx@5.1.6: - resolution: {integrity: sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA==} - engines: {node: '>= 12.0.0'} - peerDependencies: - '@babel/core': '*' - babel-plugin-macros: '*' - react: '>= 16.8.0 || 17.x.x || ^18.0.0-0 || ^19.0.0-0' - peerDependenciesMeta: - '@babel/core': - optional: true - babel-plugin-macros: - optional: true - sucrase@3.35.1: resolution: {integrity: sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==} engines: {node: '>=16 || 14 >=14.17'} @@ -2470,6 +2376,9 @@ packages: peerDependencies: react: ^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + symbol-tree@3.2.4: + resolution: {integrity: sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==} + tailwind-merge@2.6.1: resolution: {integrity: sha512-Oo6tHdpZsGpkKG88HJ8RR1rg/RdnEkQEfMoEk2x1XRI3F1AxeU+ijRXpiVUF4UbLfcxxRGw6TbUINKYdWVsQTQ==} @@ -2493,10 +2402,35 @@ packages: tiny-invariant@1.3.3: resolution: {integrity: sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==} + tinybench@2.9.0: + resolution: {integrity: sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==} + + tinyexec@0.3.2: + resolution: {integrity: sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==} + tinyglobby@0.2.15: resolution: {integrity: sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==} engines: {node: '>=12.0.0'} + tinypool@1.1.1: + resolution: {integrity: sha512-Zba82s87IFq9A9XmjiX5uZA/ARWDrB03OHlq+Vw1fSdt0I+4/Kutwy8BP4Y/y/aORMo61FQ0vIb5j44vSo5Pkg==} + engines: {node: ^18.0.0 || >=20.0.0} + + tinyrainbow@1.2.0: + resolution: {integrity: sha512-weEDEq7Z5eTHPDh4xjX789+fHfF+P8boiFB+0vbWzpbnbsEr/GRaohi/uMKxg8RZMXnl1ItAi/IUHWMsjDV7kQ==} + engines: {node: '>=14.0.0'} + + tinyspy@3.0.2: + resolution: {integrity: sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q==} + engines: {node: '>=14.0.0'} + + tldts-core@6.1.86: + resolution: {integrity: sha512-Je6p7pkk+KMzMv2XXKmAE3McmolOQFdxkKw0R8EYNr7sELW46JqnNeTX8ybPiQgvg1ymCoF8LXs5fzFaZvJPTA==} + + tldts@6.1.86: + resolution: {integrity: sha512-WMi/OQ2axVTf/ykqCQgXiIct+mSQDFdH2fkwhPwgEwvJ1kSzZRiinb0zF2Xb8u4+OqPChmyI6MEu4EezNJz+FQ==} + hasBin: true + to-regex-range@5.0.1: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} @@ -2504,18 +2438,17 @@ packages: toggle-selection@1.0.6: resolution: {integrity: sha512-BiZS+C1OS8g/q2RRbJmy59xpyghNBqrr6k5L/uKBGRsTfxmu3ffiRnd8mlGPUVayg8pvfi5urfnu8TU7DVOkLQ==} - ts-api-utils@2.4.0: - resolution: {integrity: sha512-3TaVTaAv2gTiMB35i3FiGJaRfwb3Pyn/j3m/bfAvGe8FB7CF6u+LMYqYlDh7reQf7UNvoTvdfAqHGmPGOSsPmA==} - engines: {node: '>=18.12'} - peerDependencies: - typescript: '>=4.8.4' + tough-cookie@5.1.2: + resolution: {integrity: sha512-FVDYdxtnj0G6Qm/DhNPSb8Ju59ULcup3tuJxkFb5K8Bv2pUXILbf0xZWU8PX8Ov19OXljbUyveOFwRMwkXzO+A==} + engines: {node: '>=16'} + + tr46@5.1.1: + resolution: {integrity: sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw==} + engines: {node: '>=18'} ts-interface-checker@0.1.13: resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} - tsconfig-paths@3.15.0: - resolution: {integrity: sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==} - tslib@2.8.1: resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} @@ -2523,36 +2456,19 @@ packages: resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} engines: {node: '>= 0.8.0'} - typed-array-buffer@1.0.3: - resolution: {integrity: sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==} - engines: {node: '>= 0.4'} - - typed-array-byte-length@1.0.3: - resolution: {integrity: sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==} - engines: {node: '>= 0.4'} - - typed-array-byte-offset@1.0.4: - resolution: {integrity: sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==} - engines: {node: '>= 0.4'} - - typed-array-length@1.0.7: - resolution: {integrity: sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==} - engines: {node: '>= 0.4'} - typescript@5.9.3: resolution: {integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==} engines: {node: '>=14.17'} hasBin: true - unbox-primitive@1.1.0: - resolution: {integrity: sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==} - engines: {node: '>= 0.4'} - undici-types@6.21.0: resolution: {integrity: sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==} - unrs-resolver@1.11.1: - resolution: {integrity: sha512-bSjt9pjaEBnNiGgc9rUiHGKv5l4/TGzDmYw3RhnkJGtLhbnnA/5qJj7x3dNDCRx/PJxu774LlH8lCOlB4hEfKg==} + update-browserslist-db@1.2.3: + resolution: {integrity: sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==} + hasBin: true + peerDependencies: + browserslist: '>= 4.21.0' uri-js@4.4.1: resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} @@ -2588,38 +2504,163 @@ packages: victory-vendor@36.9.2: resolution: {integrity: sha512-PnpQQMuxlwYdocC8fIJqVXvkeViHYzotI+NJrCuav0ZYFoq912ZHBk3mCeuj+5/VpodOjPe1z0Fk2ihgzlXqjQ==} - which-boxed-primitive@1.1.1: - resolution: {integrity: sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==} - engines: {node: '>= 0.4'} + vite-node@2.1.9: + resolution: {integrity: sha512-AM9aQ/IPrW/6ENLQg3AGY4K1N2TGZdR5e4gu/MmmR2xR3Ll1+dib+nook92g4TV3PXVyeyxdWwtaCAiUL0hMxA==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true - which-builtin-type@1.2.1: - resolution: {integrity: sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==} - engines: {node: '>= 0.4'} + vite@5.4.21: + resolution: {integrity: sha512-o5a9xKjbtuhY6Bi5S3+HvbRERmouabWbyUcpXXUA1u+GNUKoROi9byOJ8M0nHbHYHkYICiMlqxkg1KkYmm25Sw==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@types/node': ^18.0.0 || >=20.0.0 + less: '*' + lightningcss: ^1.21.0 + sass: '*' + sass-embedded: '*' + stylus: '*' + sugarss: '*' + terser: ^5.4.0 + peerDependenciesMeta: + '@types/node': + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + sass-embedded: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true - which-collection@1.0.2: - resolution: {integrity: sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==} - engines: {node: '>= 0.4'} + vite@6.4.1: + resolution: {integrity: sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==} + engines: {node: ^18.0.0 || ^20.0.0 || >=22.0.0} + hasBin: true + peerDependencies: + '@types/node': ^18.0.0 || ^20.0.0 || >=22.0.0 + jiti: '>=1.21.0' + less: '*' + lightningcss: ^1.21.0 + sass: '*' + sass-embedded: '*' + stylus: '*' + sugarss: '*' + terser: ^5.16.0 + tsx: ^4.8.1 + yaml: ^2.4.2 + peerDependenciesMeta: + '@types/node': + optional: true + jiti: + optional: true + less: + optional: true + lightningcss: + optional: true + sass: + optional: true + sass-embedded: + optional: true + stylus: + optional: true + sugarss: + optional: true + terser: + optional: true + tsx: + optional: true + yaml: + optional: true - which-typed-array@1.1.20: - resolution: {integrity: sha512-LYfpUkmqwl0h9A2HL09Mms427Q1RZWuOHsukfVcKRq9q95iQxdw0ix1JQrqbcDR9PH1QDwf5Qo8OZb5lksZ8Xg==} - engines: {node: '>= 0.4'} + vitest@2.1.9: + resolution: {integrity: sha512-MSmPM9REYqDGBI8439mA4mWhV5sKmDlBKWIYbA3lRb2PTHACE0mgKwA8yQ2xq9vxDTuk4iPrECBAEW2aoFXY0Q==} + engines: {node: ^18.0.0 || >=20.0.0} + hasBin: true + peerDependencies: + '@edge-runtime/vm': '*' + '@types/node': ^18.0.0 || >=20.0.0 + '@vitest/browser': 2.1.9 + '@vitest/ui': 2.1.9 + happy-dom: '*' + jsdom: '*' + peerDependenciesMeta: + '@edge-runtime/vm': + optional: true + '@types/node': + optional: true + '@vitest/browser': + optional: true + '@vitest/ui': + optional: true + happy-dom: + optional: true + jsdom: + optional: true + + w3c-xmlserializer@5.0.0: + resolution: {integrity: sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==} + engines: {node: '>=18'} + + webidl-conversions@7.0.0: + resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==} + engines: {node: '>=12'} + + whatwg-encoding@3.1.1: + resolution: {integrity: sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==} + engines: {node: '>=18'} + deprecated: Use @exodus/bytes instead for a more spec-conformant and faster implementation + + whatwg-mimetype@4.0.0: + resolution: {integrity: sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==} + engines: {node: '>=18'} + + whatwg-url@14.2.0: + resolution: {integrity: sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==} + engines: {node: '>=18'} which@2.0.2: resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} engines: {node: '>= 8'} hasBin: true + why-is-node-running@2.3.0: + resolution: {integrity: sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==} + engines: {node: '>=8'} + hasBin: true + word-wrap@1.2.5: resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} engines: {node: '>=0.10.0'} - wrap-ansi@7.0.0: - resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} - engines: {node: '>=10'} + ws@8.19.0: + resolution: {integrity: sha512-blAT2mjOEIi0ZzruJfIhb3nps74PRWTCz1IjglWEEpQl5XS/UNama6u2/rjFkDDouqr4L67ry+1aGIALViWjDg==} + engines: {node: '>=10.0.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: '>=5.0.2' + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true - wrap-ansi@8.1.0: - resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} - engines: {node: '>=12'} + xml-name-validator@5.0.0: + resolution: {integrity: sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==} + engines: {node: '>=18'} + + xmlchars@2.2.0: + resolution: {integrity: sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==} + + yallist@3.1.1: + resolution: {integrity: sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==} yaml@2.8.0: resolution: {integrity: sha512-4lLa/EcQCB0cJkyts+FpIRx5G/llPxfP6VQU5KByHEhLxY3IJCH0f0Hy1MHI8sClTvsIb8qwRJ6R/ZdlDJ/leQ==} @@ -2635,219 +2676,385 @@ packages: snapshots: - '@alloc/quick-lru@5.2.0': {} - - '@babel/runtime@7.28.6': {} + '@adobe/css-tools@4.4.4': {} - '@date-fns/tz@1.4.1': {} + '@alloc/quick-lru@5.2.0': {} - '@emnapi/core@1.8.1': + '@asamuzakjp/css-color@3.2.0': dependencies: - '@emnapi/wasi-threads': 1.1.0 - tslib: 2.8.1 - optional: true + '@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4) + '@csstools/css-color-parser': 3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4) + '@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4) + '@csstools/css-tokenizer': 3.0.4 + lru-cache: 10.4.3 - '@emnapi/runtime@1.8.1': + '@babel/code-frame@7.29.0': dependencies: - tslib: 2.8.1 - optional: true + '@babel/helper-validator-identifier': 7.28.5 + js-tokens: 4.0.0 + picocolors: 1.1.1 + + '@babel/compat-data@7.29.0': {} + + '@babel/core@7.29.0': + dependencies: + '@babel/code-frame': 7.29.0 + '@babel/generator': 7.29.1 + '@babel/helper-compilation-targets': 7.28.6 + '@babel/helper-module-transforms': 7.28.6(@babel/core@7.29.0) + '@babel/helpers': 7.28.6 + '@babel/parser': 7.29.0 + '@babel/template': 7.28.6 + '@babel/traverse': 7.29.0 + '@babel/types': 7.29.0 + '@jridgewell/remapping': 2.3.5 + convert-source-map: 2.0.0 + debug: 4.4.3 + gensync: 1.0.0-beta.2 + json5: 2.2.3 + semver: 6.3.1 + transitivePeerDependencies: + - supports-color - '@emnapi/wasi-threads@1.1.0': + '@babel/generator@7.29.1': dependencies: - tslib: 2.8.1 - optional: true + '@babel/parser': 7.29.0 + '@babel/types': 7.29.0 + '@jridgewell/gen-mapping': 0.3.13 + '@jridgewell/trace-mapping': 0.3.31 + jsesc: 3.1.0 - '@eslint-community/eslint-utils@4.9.1(eslint@9.39.2(jiti@1.21.7))': + '@babel/helper-compilation-targets@7.28.6': dependencies: - eslint: 9.39.2(jiti@1.21.7) - eslint-visitor-keys: 3.4.3 + '@babel/compat-data': 7.29.0 + '@babel/helper-validator-option': 7.27.1 + browserslist: 4.28.1 + lru-cache: 5.1.1 + semver: 6.3.1 - '@eslint-community/regexpp@4.12.2': {} + '@babel/helper-globals@7.28.0': {} - '@eslint/config-array@0.21.1': + '@babel/helper-module-imports@7.28.6': dependencies: - '@eslint/object-schema': 2.1.7 - debug: 4.4.3 - minimatch: 3.1.2 + '@babel/traverse': 7.29.0 + '@babel/types': 7.29.0 transitivePeerDependencies: - supports-color - '@eslint/config-helpers@0.4.2': + '@babel/helper-module-transforms@7.28.6(@babel/core@7.29.0)': dependencies: - '@eslint/core': 0.17.0 + '@babel/core': 7.29.0 + '@babel/helper-module-imports': 7.28.6 + '@babel/helper-validator-identifier': 7.28.5 + '@babel/traverse': 7.29.0 + transitivePeerDependencies: + - supports-color - '@eslint/core@0.17.0': + '@babel/helper-plugin-utils@7.28.6': {} + + '@babel/helper-string-parser@7.27.1': {} + + '@babel/helper-validator-identifier@7.28.5': {} + + '@babel/helper-validator-option@7.27.1': {} + + '@babel/helpers@7.28.6': dependencies: - '@types/json-schema': 7.0.15 + '@babel/template': 7.28.6 + '@babel/types': 7.29.0 - '@eslint/eslintrc@3.3.3': + '@babel/parser@7.29.0': dependencies: - ajv: 6.12.6 + '@babel/types': 7.29.0 + + '@babel/plugin-transform-react-jsx-self@7.27.1(@babel/core@7.29.0)': + dependencies: + '@babel/core': 7.29.0 + '@babel/helper-plugin-utils': 7.28.6 + + '@babel/plugin-transform-react-jsx-source@7.27.1(@babel/core@7.29.0)': + dependencies: + '@babel/core': 7.29.0 + '@babel/helper-plugin-utils': 7.28.6 + + '@babel/runtime@7.28.6': {} + + '@babel/template@7.28.6': + dependencies: + '@babel/code-frame': 7.29.0 + '@babel/parser': 7.29.0 + '@babel/types': 7.29.0 + + '@babel/traverse@7.29.0': + dependencies: + '@babel/code-frame': 7.29.0 + '@babel/generator': 7.29.1 + '@babel/helper-globals': 7.28.0 + '@babel/parser': 7.29.0 + '@babel/template': 7.28.6 + '@babel/types': 7.29.0 debug: 4.4.3 - espree: 10.4.0 - globals: 14.0.0 - ignore: 5.3.2 - import-fresh: 3.3.1 - js-yaml: 4.1.1 - minimatch: 3.1.2 - strip-json-comments: 3.1.1 transitivePeerDependencies: - supports-color - '@eslint/js@9.39.2': {} + '@babel/types@7.29.0': + dependencies: + '@babel/helper-string-parser': 7.27.1 + '@babel/helper-validator-identifier': 7.28.5 - '@eslint/object-schema@2.1.7': {} + '@csstools/color-helpers@5.1.0': {} - '@eslint/plugin-kit@0.4.1': + '@csstools/css-calc@2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)': dependencies: - '@eslint/core': 0.17.0 - levn: 0.4.1 + '@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4) + '@csstools/css-tokenizer': 3.0.4 - '@floating-ui/core@1.7.4': + '@csstools/css-color-parser@3.1.0(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4)': dependencies: - '@floating-ui/utils': 0.2.10 + '@csstools/color-helpers': 5.1.0 + '@csstools/css-calc': 2.1.4(@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4))(@csstools/css-tokenizer@3.0.4) + '@csstools/css-parser-algorithms': 3.0.5(@csstools/css-tokenizer@3.0.4) + '@csstools/css-tokenizer': 3.0.4 - '@floating-ui/dom@1.7.5': + '@csstools/css-parser-algorithms@3.0.5(@csstools/css-tokenizer@3.0.4)': dependencies: - '@floating-ui/core': 1.7.4 - '@floating-ui/utils': 0.2.10 + '@csstools/css-tokenizer': 3.0.4 - '@floating-ui/react-dom@2.1.7(react-dom@19.1.0(react@19.1.0))(react@19.1.0)': - dependencies: - '@floating-ui/dom': 1.7.5 - react: 19.1.0 - react-dom: 19.1.0(react@19.1.0) + '@csstools/css-tokenizer@3.0.4': {} - '@floating-ui/utils@0.2.10': {} + '@date-fns/tz@1.4.1': {} - '@humanfs/core@0.19.1': {} + '@esbuild/aix-ppc64@0.21.5': + optional: true - '@humanfs/node@0.16.7': - dependencies: - '@humanfs/core': 0.19.1 - '@humanwhocodes/retry': 0.4.3 + '@esbuild/aix-ppc64@0.25.12': + optional: true - '@humanwhocodes/module-importer@1.0.1': {} + '@esbuild/android-arm64@0.21.5': + optional: true - '@humanwhocodes/retry@0.4.3': {} + '@esbuild/android-arm64@0.25.12': + optional: true - '@img/colour@1.0.0': + '@esbuild/android-arm@0.21.5': optional: true - '@img/sharp-darwin-arm64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-darwin-arm64': 1.2.4 + '@esbuild/android-arm@0.25.12': optional: true - '@img/sharp-darwin-x64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-darwin-x64': 1.2.4 + '@esbuild/android-x64@0.21.5': optional: true - '@img/sharp-libvips-darwin-arm64@1.2.4': + '@esbuild/android-x64@0.25.12': optional: true - '@img/sharp-libvips-darwin-x64@1.2.4': + '@esbuild/darwin-arm64@0.21.5': optional: true - '@img/sharp-libvips-linux-arm64@1.2.4': + '@esbuild/darwin-arm64@0.25.12': optional: true - '@img/sharp-libvips-linux-arm@1.2.4': + '@esbuild/darwin-x64@0.21.5': optional: true - '@img/sharp-libvips-linux-ppc64@1.2.4': + '@esbuild/darwin-x64@0.25.12': optional: true - '@img/sharp-libvips-linux-riscv64@1.2.4': + '@esbuild/freebsd-arm64@0.21.5': optional: true - '@img/sharp-libvips-linux-s390x@1.2.4': + '@esbuild/freebsd-arm64@0.25.12': optional: true - '@img/sharp-libvips-linux-x64@1.2.4': + '@esbuild/freebsd-x64@0.21.5': optional: true - '@img/sharp-libvips-linuxmusl-arm64@1.2.4': + '@esbuild/freebsd-x64@0.25.12': optional: true - '@img/sharp-libvips-linuxmusl-x64@1.2.4': + '@esbuild/linux-arm64@0.21.5': optional: true - '@img/sharp-linux-arm64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm64': 1.2.4 + '@esbuild/linux-arm64@0.25.12': optional: true - '@img/sharp-linux-arm@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-arm': 1.2.4 + '@esbuild/linux-arm@0.21.5': optional: true - '@img/sharp-linux-ppc64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-ppc64': 1.2.4 + '@esbuild/linux-arm@0.25.12': + optional: true + + '@esbuild/linux-ia32@0.21.5': + optional: true + + '@esbuild/linux-ia32@0.25.12': + optional: true + + '@esbuild/linux-loong64@0.21.5': + optional: true + + '@esbuild/linux-loong64@0.25.12': + optional: true + + '@esbuild/linux-mips64el@0.21.5': + optional: true + + '@esbuild/linux-mips64el@0.25.12': + optional: true + + '@esbuild/linux-ppc64@0.21.5': + optional: true + + '@esbuild/linux-ppc64@0.25.12': + optional: true + + '@esbuild/linux-riscv64@0.21.5': + optional: true + + '@esbuild/linux-riscv64@0.25.12': + optional: true + + '@esbuild/linux-s390x@0.21.5': + optional: true + + '@esbuild/linux-s390x@0.25.12': + optional: true + + '@esbuild/linux-x64@0.21.5': + optional: true + + '@esbuild/linux-x64@0.25.12': + optional: true + + '@esbuild/netbsd-arm64@0.25.12': + optional: true + + '@esbuild/netbsd-x64@0.21.5': + optional: true + + '@esbuild/netbsd-x64@0.25.12': + optional: true + + '@esbuild/openbsd-arm64@0.25.12': + optional: true + + '@esbuild/openbsd-x64@0.21.5': + optional: true + + '@esbuild/openbsd-x64@0.25.12': + optional: true + + '@esbuild/openharmony-arm64@0.25.12': + optional: true + + '@esbuild/sunos-x64@0.21.5': + optional: true + + '@esbuild/sunos-x64@0.25.12': + optional: true + + '@esbuild/win32-arm64@0.21.5': + optional: true + + '@esbuild/win32-arm64@0.25.12': + optional: true + + '@esbuild/win32-ia32@0.21.5': + optional: true + + '@esbuild/win32-ia32@0.25.12': + optional: true + + '@esbuild/win32-x64@0.21.5': + optional: true + + '@esbuild/win32-x64@0.25.12': optional: true - '@img/sharp-linux-riscv64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-riscv64': 1.2.4 - optional: true + '@eslint-community/eslint-utils@4.9.1(eslint@9.39.2(jiti@1.21.7))': + dependencies: + eslint: 9.39.2(jiti@1.21.7) + eslint-visitor-keys: 3.4.3 + + '@eslint-community/regexpp@4.12.2': {} + + '@eslint/config-array@0.21.1': + dependencies: + '@eslint/object-schema': 2.1.7 + debug: 4.4.3 + minimatch: 3.1.2 + transitivePeerDependencies: + - supports-color + + '@eslint/config-helpers@0.4.2': + dependencies: + '@eslint/core': 0.17.0 + + '@eslint/core@0.17.0': + dependencies: + '@types/json-schema': 7.0.15 - '@img/sharp-linux-s390x@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-s390x': 1.2.4 - optional: true + '@eslint/eslintrc@3.3.3': + dependencies: + ajv: 6.12.6 + debug: 4.4.3 + espree: 10.4.0 + globals: 14.0.0 + ignore: 5.3.2 + import-fresh: 3.3.1 + js-yaml: 4.1.1 + minimatch: 3.1.2 + strip-json-comments: 3.1.1 + transitivePeerDependencies: + - supports-color - '@img/sharp-linux-x64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linux-x64': 1.2.4 - optional: true + '@eslint/js@9.39.2': {} - '@img/sharp-linuxmusl-arm64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-arm64': 1.2.4 - optional: true + '@eslint/object-schema@2.1.7': {} - '@img/sharp-linuxmusl-x64@0.34.5': - optionalDependencies: - '@img/sharp-libvips-linuxmusl-x64': 1.2.4 - optional: true + '@eslint/plugin-kit@0.4.1': + dependencies: + '@eslint/core': 0.17.0 + levn: 0.4.1 - '@img/sharp-wasm32@0.34.5': + '@floating-ui/core@1.7.4': dependencies: - '@emnapi/runtime': 1.8.1 - optional: true + '@floating-ui/utils': 0.2.10 - '@img/sharp-win32-arm64@0.34.5': - optional: true + '@floating-ui/dom@1.7.5': + dependencies: + '@floating-ui/core': 1.7.4 + '@floating-ui/utils': 0.2.10 - '@img/sharp-win32-ia32@0.34.5': - optional: true + '@floating-ui/react-dom@2.1.7(react-dom@19.1.0(react@19.1.0))(react@19.1.0)': + dependencies: + '@floating-ui/dom': 1.7.5 + react: 19.1.0 + react-dom: 19.1.0(react@19.1.0) - '@img/sharp-win32-x64@0.34.5': - optional: true + '@floating-ui/utils@0.2.10': {} - '@isaacs/balanced-match@4.0.1': {} + '@humanfs/core@0.19.1': {} - '@isaacs/brace-expansion@5.0.0': + '@humanfs/node@0.16.7': dependencies: - '@isaacs/balanced-match': 4.0.1 + '@humanfs/core': 0.19.1 + '@humanwhocodes/retry': 0.4.3 - '@isaacs/cliui@8.0.2': - dependencies: - string-width: 5.1.2 - string-width-cjs: string-width@4.2.3 - strip-ansi: 7.1.2 - strip-ansi-cjs: strip-ansi@6.0.1 - wrap-ansi: 8.1.0 - wrap-ansi-cjs: wrap-ansi@7.0.0 + '@humanwhocodes/module-importer@1.0.1': {} + + '@humanwhocodes/retry@0.4.3': {} '@jridgewell/gen-mapping@0.3.13': dependencies: '@jridgewell/sourcemap-codec': 1.5.5 '@jridgewell/trace-mapping': 0.3.31 + '@jridgewell/remapping@2.3.5': + dependencies: + '@jridgewell/gen-mapping': 0.3.13 + '@jridgewell/trace-mapping': 0.3.31 + '@jridgewell/resolve-uri@3.1.2': {} '@jridgewell/sourcemap-codec@1.5.5': {} @@ -2857,43 +3064,6 @@ snapshots: '@jridgewell/resolve-uri': 3.1.2 '@jridgewell/sourcemap-codec': 1.5.5 - '@napi-rs/wasm-runtime@0.2.12': - dependencies: - '@emnapi/core': 1.8.1 - '@emnapi/runtime': 1.8.1 - '@tybys/wasm-util': 0.10.1 - optional: true - - '@next/env@15.4.10': {} - - '@next/eslint-plugin-next@15.2.4': - dependencies: - fast-glob: 3.3.1 - - '@next/swc-darwin-arm64@15.4.8': - optional: true - - '@next/swc-darwin-x64@15.4.8': - optional: true - - '@next/swc-linux-arm64-gnu@15.4.8': - optional: true - - '@next/swc-linux-arm64-musl@15.4.8': - optional: true - - '@next/swc-linux-x64-gnu@15.4.8': - optional: true - - '@next/swc-linux-x64-musl@15.4.8': - optional: true - - '@next/swc-win32-arm64-msvc@15.4.8': - optional: true - - '@next/swc-win32-x64-msvc@15.4.8': - optional: true - '@nodelib/fs.scandir@2.1.5': dependencies: '@nodelib/fs.stat': 2.0.5 @@ -2906,8 +3076,6 @@ snapshots: '@nodelib/fs.scandir': 2.1.5 fastq: 1.20.1 - '@nolyfill/is-core-module@1.0.39': {} - '@radix-ui/number@1.1.1': {} '@radix-ui/primitive@1.1.3': {} @@ -3346,18 +3514,141 @@ snapshots: '@radix-ui/rect@1.1.1': {} - '@rtsao/scc@1.1.0': {} + '@remix-run/router@1.23.2': {} + + '@rolldown/pluginutils@1.0.0-beta.27': {} + + '@rollup/rollup-android-arm-eabi@4.59.0': + optional: true + + '@rollup/rollup-android-arm64@4.59.0': + optional: true + + '@rollup/rollup-darwin-arm64@4.59.0': + optional: true + + '@rollup/rollup-darwin-x64@4.59.0': + optional: true + + '@rollup/rollup-freebsd-arm64@4.59.0': + optional: true + + '@rollup/rollup-freebsd-x64@4.59.0': + optional: true + + '@rollup/rollup-linux-arm-gnueabihf@4.59.0': + optional: true + + '@rollup/rollup-linux-arm-musleabihf@4.59.0': + optional: true + + '@rollup/rollup-linux-arm64-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-arm64-musl@4.59.0': + optional: true + + '@rollup/rollup-linux-loong64-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-loong64-musl@4.59.0': + optional: true + + '@rollup/rollup-linux-ppc64-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-ppc64-musl@4.59.0': + optional: true + + '@rollup/rollup-linux-riscv64-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-riscv64-musl@4.59.0': + optional: true + + '@rollup/rollup-linux-s390x-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-x64-gnu@4.59.0': + optional: true + + '@rollup/rollup-linux-x64-musl@4.59.0': + optional: true + + '@rollup/rollup-openbsd-x64@4.59.0': + optional: true + + '@rollup/rollup-openharmony-arm64@4.59.0': + optional: true + + '@rollup/rollup-win32-arm64-msvc@4.59.0': + optional: true + + '@rollup/rollup-win32-ia32-msvc@4.59.0': + optional: true + + '@rollup/rollup-win32-x64-gnu@4.59.0': + optional: true - '@rushstack/eslint-patch@1.15.0': {} + '@rollup/rollup-win32-x64-msvc@4.59.0': + optional: true - '@swc/helpers@0.5.15': + '@testing-library/dom@10.4.1': dependencies: - tslib: 2.8.1 + '@babel/code-frame': 7.29.0 + '@babel/runtime': 7.28.6 + '@types/aria-query': 5.0.4 + aria-query: 5.3.0 + dom-accessibility-api: 0.5.16 + lz-string: 1.5.0 + picocolors: 1.1.1 + pretty-format: 27.5.1 - '@tybys/wasm-util@0.10.1': + '@testing-library/jest-dom@6.9.1': dependencies: - tslib: 2.8.1 - optional: true + '@adobe/css-tools': 4.4.4 + aria-query: 5.3.2 + css.escape: 1.5.1 + dom-accessibility-api: 0.6.3 + picocolors: 1.1.1 + redent: 3.0.0 + + '@testing-library/react@16.3.2(@testing-library/dom@10.4.1)(@types/react-dom@19.1.1(@types/react@19.1.0))(@types/react@19.1.0)(react-dom@19.1.0(react@19.1.0))(react@19.1.0)': + dependencies: + '@babel/runtime': 7.28.6 + '@testing-library/dom': 10.4.1 + react: 19.1.0 + react-dom: 19.1.0(react@19.1.0) + optionalDependencies: + '@types/react': 19.1.0 + '@types/react-dom': 19.1.1(@types/react@19.1.0) + + '@testing-library/user-event@14.6.1(@testing-library/dom@10.4.1)': + dependencies: + '@testing-library/dom': 10.4.1 + + '@types/aria-query@5.0.4': {} + + '@types/babel__core@7.20.5': + dependencies: + '@babel/parser': 7.29.0 + '@babel/types': 7.29.0 + '@types/babel__generator': 7.27.0 + '@types/babel__template': 7.4.4 + '@types/babel__traverse': 7.28.0 + + '@types/babel__generator@7.27.0': + dependencies: + '@babel/types': 7.29.0 + + '@types/babel__template@7.4.4': + dependencies: + '@babel/parser': 7.29.0 + '@babel/types': 7.29.0 + + '@types/babel__traverse@7.28.0': + dependencies: + '@babel/types': 7.29.0 '@types/d3-array@3.2.2': {} @@ -3387,8 +3678,6 @@ snapshots: '@types/json-schema@7.0.15': {} - '@types/json5@0.0.29': {} - '@types/node@20.19.30': dependencies: undici-types: 6.21.0 @@ -3401,155 +3690,57 @@ snapshots: dependencies: csstype: 3.2.3 - '@typescript-eslint/eslint-plugin@8.54.0(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3)': - dependencies: - '@eslint-community/regexpp': 4.12.2 - '@typescript-eslint/parser': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/scope-manager': 8.54.0 - '@typescript-eslint/type-utils': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/utils': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/visitor-keys': 8.54.0 - eslint: 9.39.2(jiti@1.21.7) - ignore: 7.0.5 - natural-compare: 1.4.0 - ts-api-utils: 2.4.0(typescript@5.9.3) - typescript: 5.9.3 - transitivePeerDependencies: - - supports-color - - '@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3)': - dependencies: - '@typescript-eslint/scope-manager': 8.54.0 - '@typescript-eslint/types': 8.54.0 - '@typescript-eslint/typescript-estree': 8.54.0(typescript@5.9.3) - '@typescript-eslint/visitor-keys': 8.54.0 - debug: 4.4.3 - eslint: 9.39.2(jiti@1.21.7) - typescript: 5.9.3 - transitivePeerDependencies: - - supports-color - - '@typescript-eslint/project-service@8.54.0(typescript@5.9.3)': + '@vitejs/plugin-react@4.7.0(vite@6.4.1(@types/node@20.19.30)(jiti@1.21.7)(yaml@2.8.0))': dependencies: - '@typescript-eslint/tsconfig-utils': 8.54.0(typescript@5.9.3) - '@typescript-eslint/types': 8.54.0 - debug: 4.4.3 - typescript: 5.9.3 + '@babel/core': 7.29.0 + '@babel/plugin-transform-react-jsx-self': 7.27.1(@babel/core@7.29.0) + '@babel/plugin-transform-react-jsx-source': 7.27.1(@babel/core@7.29.0) + '@rolldown/pluginutils': 1.0.0-beta.27 + '@types/babel__core': 7.20.5 + react-refresh: 0.17.0 + vite: 6.4.1(@types/node@20.19.30)(jiti@1.21.7)(yaml@2.8.0) transitivePeerDependencies: - supports-color - '@typescript-eslint/scope-manager@8.54.0': + '@vitest/expect@2.1.9': dependencies: - '@typescript-eslint/types': 8.54.0 - '@typescript-eslint/visitor-keys': 8.54.0 + '@vitest/spy': 2.1.9 + '@vitest/utils': 2.1.9 + chai: 5.3.3 + tinyrainbow: 1.2.0 - '@typescript-eslint/tsconfig-utils@8.54.0(typescript@5.9.3)': + '@vitest/mocker@2.1.9(vite@5.4.21(@types/node@20.19.30))': dependencies: - typescript: 5.9.3 + '@vitest/spy': 2.1.9 + estree-walker: 3.0.3 + magic-string: 0.30.21 + optionalDependencies: + vite: 5.4.21(@types/node@20.19.30) - '@typescript-eslint/type-utils@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3)': + '@vitest/pretty-format@2.1.9': dependencies: - '@typescript-eslint/types': 8.54.0 - '@typescript-eslint/typescript-estree': 8.54.0(typescript@5.9.3) - '@typescript-eslint/utils': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - debug: 4.4.3 - eslint: 9.39.2(jiti@1.21.7) - ts-api-utils: 2.4.0(typescript@5.9.3) - typescript: 5.9.3 - transitivePeerDependencies: - - supports-color - - '@typescript-eslint/types@8.54.0': {} + tinyrainbow: 1.2.0 - '@typescript-eslint/typescript-estree@8.54.0(typescript@5.9.3)': + '@vitest/runner@2.1.9': dependencies: - '@typescript-eslint/project-service': 8.54.0(typescript@5.9.3) - '@typescript-eslint/tsconfig-utils': 8.54.0(typescript@5.9.3) - '@typescript-eslint/types': 8.54.0 - '@typescript-eslint/visitor-keys': 8.54.0 - debug: 4.4.3 - minimatch: 9.0.5 - semver: 7.7.3 - tinyglobby: 0.2.15 - ts-api-utils: 2.4.0(typescript@5.9.3) - typescript: 5.9.3 - transitivePeerDependencies: - - supports-color + '@vitest/utils': 2.1.9 + pathe: 1.1.2 - '@typescript-eslint/utils@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3)': + '@vitest/snapshot@2.1.9': dependencies: - '@eslint-community/eslint-utils': 4.9.1(eslint@9.39.2(jiti@1.21.7)) - '@typescript-eslint/scope-manager': 8.54.0 - '@typescript-eslint/types': 8.54.0 - '@typescript-eslint/typescript-estree': 8.54.0(typescript@5.9.3) - eslint: 9.39.2(jiti@1.21.7) - typescript: 5.9.3 - transitivePeerDependencies: - - supports-color + '@vitest/pretty-format': 2.1.9 + magic-string: 0.30.21 + pathe: 1.1.2 - '@typescript-eslint/visitor-keys@8.54.0': + '@vitest/spy@2.1.9': dependencies: - '@typescript-eslint/types': 8.54.0 - eslint-visitor-keys: 4.2.1 - - '@unrs/resolver-binding-android-arm-eabi@1.11.1': - optional: true - - '@unrs/resolver-binding-android-arm64@1.11.1': - optional: true - - '@unrs/resolver-binding-darwin-arm64@1.11.1': - optional: true - - '@unrs/resolver-binding-darwin-x64@1.11.1': - optional: true - - '@unrs/resolver-binding-freebsd-x64@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-arm-gnueabihf@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-arm-musleabihf@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-arm64-gnu@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-arm64-musl@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-ppc64-gnu@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-riscv64-gnu@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-riscv64-musl@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-s390x-gnu@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-x64-gnu@1.11.1': - optional: true - - '@unrs/resolver-binding-linux-x64-musl@1.11.1': - optional: true + tinyspy: 3.0.2 - '@unrs/resolver-binding-wasm32-wasi@1.11.1': + '@vitest/utils@2.1.9': dependencies: - '@napi-rs/wasm-runtime': 0.2.12 - optional: true - - '@unrs/resolver-binding-win32-arm64-msvc@1.11.1': - optional: true - - '@unrs/resolver-binding-win32-ia32-msvc@1.11.1': - optional: true - - '@unrs/resolver-binding-win32-x64-msvc@1.11.1': - optional: true + '@vitest/pretty-format': 2.1.9 + loupe: 3.2.1 + tinyrainbow: 1.2.0 acorn-jsx@5.3.2(acorn@8.15.0): dependencies: @@ -3557,6 +3748,8 @@ snapshots: acorn@8.15.0: {} + agent-base@7.1.4: {} + ajv@6.12.6: dependencies: fast-deep-equal: 3.1.3 @@ -3566,13 +3759,11 @@ snapshots: ansi-regex@5.0.1: {} - ansi-regex@6.2.2: {} - ansi-styles@4.3.0: dependencies: color-convert: 2.0.1 - ansi-styles@6.2.3: {} + ansi-styles@5.2.0: {} any-promise@1.3.0: {} @@ -3581,97 +3772,28 @@ snapshots: normalize-path: 3.0.0 picomatch: 2.3.1 - arg@5.0.2: {} - - argparse@2.0.1: {} - - aria-hidden@1.2.6: - dependencies: - tslib: 2.8.1 - - aria-query@5.3.2: {} - - array-buffer-byte-length@1.0.2: - dependencies: - call-bound: 1.0.4 - is-array-buffer: 3.0.5 - - array-includes@3.1.9: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-object-atoms: 1.1.1 - get-intrinsic: 1.3.0 - is-string: 1.1.1 - math-intrinsics: 1.1.0 - - array.prototype.findlast@1.2.5: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - es-shim-unscopables: 1.1.0 - - array.prototype.findlastindex@1.2.6: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - es-shim-unscopables: 1.1.0 - - array.prototype.flat@1.3.3: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-shim-unscopables: 1.1.0 + arg@5.0.2: {} - array.prototype.flatmap@1.3.3: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-shim-unscopables: 1.1.0 + argparse@2.0.1: {} - array.prototype.tosorted@1.1.4: + aria-hidden@1.2.6: dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-shim-unscopables: 1.1.0 + tslib: 2.8.1 - arraybuffer.prototype.slice@1.0.4: + aria-query@5.3.0: dependencies: - array-buffer-byte-length: 1.0.2 - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - get-intrinsic: 1.3.0 - is-array-buffer: 3.0.5 - - ast-types-flow@0.0.8: {} - - async-function@1.0.0: {} + dequal: 2.0.3 - available-typed-arrays@1.0.7: - dependencies: - possible-typed-array-names: 1.1.0 + aria-query@5.3.2: {} - axe-core@4.11.1: {} + assertion-error@2.0.1: {} - axobject-query@4.1.0: {} + asynckit@0.4.0: {} balanced-match@1.0.2: {} + baseline-browser-mapping@2.10.0: {} + binary-extensions@2.3.0: {} brace-expansion@1.1.12: @@ -3679,30 +3801,24 @@ snapshots: balanced-match: 1.0.2 concat-map: 0.0.1 - brace-expansion@2.0.2: - dependencies: - balanced-match: 1.0.2 - braces@3.0.3: dependencies: fill-range: 7.1.1 - call-bind-apply-helpers@1.0.2: + browserslist@4.28.1: dependencies: - es-errors: 1.3.0 - function-bind: 1.1.2 + baseline-browser-mapping: 2.10.0 + caniuse-lite: 1.0.30001766 + electron-to-chromium: 1.5.302 + node-releases: 2.0.27 + update-browserslist-db: 1.2.3(browserslist@4.28.1) - call-bind@1.0.8: - dependencies: - call-bind-apply-helpers: 1.0.2 - es-define-property: 1.0.1 - get-intrinsic: 1.3.0 - set-function-length: 1.2.2 + cac@6.7.14: {} - call-bound@1.0.4: + call-bind-apply-helpers@1.0.2: dependencies: - call-bind-apply-helpers: 1.0.2 - get-intrinsic: 1.3.0 + es-errors: 1.3.0 + function-bind: 1.1.2 callsites@3.1.0: {} @@ -3710,11 +3826,21 @@ snapshots: caniuse-lite@1.0.30001766: {} + chai@5.3.3: + dependencies: + assertion-error: 2.0.1 + check-error: 2.1.3 + deep-eql: 5.0.2 + loupe: 3.2.1 + pathval: 2.0.1 + chalk@4.1.2: dependencies: ansi-styles: 4.3.0 supports-color: 7.2.0 + check-error@2.1.3: {} + chokidar@3.6.0: dependencies: anymatch: 3.1.3 @@ -3731,8 +3857,6 @@ snapshots: dependencies: clsx: 2.1.1 - client-only@0.0.1: {} - clsx@2.1.1: {} color-convert@2.0.1: @@ -3741,10 +3865,16 @@ snapshots: color-name@1.1.4: {} + combined-stream@1.0.8: + dependencies: + delayed-stream: 1.0.0 + commander@4.1.1: {} concat-map@0.0.1: {} + convert-source-map@2.0.0: {} + copy-to-clipboard@3.3.3: dependencies: toggle-selection: 1.0.6 @@ -3755,8 +3885,15 @@ snapshots: shebang-command: 2.0.0 which: 2.0.2 + css.escape@1.5.1: {} + cssesc@3.0.0: {} + cssstyle@4.6.0: + dependencies: + '@asamuzakjp/css-color': 3.2.0 + rrweb-cssom: 0.8.0 + csstype@3.2.3: {} d3-array@3.2.4: @@ -3797,25 +3934,10 @@ snapshots: d3-timer@3.0.1: {} - damerau-levenshtein@1.0.8: {} - - data-view-buffer@1.0.2: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - is-data-view: 1.0.2 - - data-view-byte-length@1.0.2: + data-urls@5.0.0: dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - is-data-view: 1.0.2 - - data-view-byte-offset@1.0.1: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - is-data-view: 1.0.2 + whatwg-mimetype: 4.0.0 + whatwg-url: 14.2.0 date-fns-jalali@4.1.0-0: {} @@ -3823,34 +3945,21 @@ snapshots: date-fns@4.1.0: {} - debug@3.2.7: - dependencies: - ms: 2.1.3 - debug@4.4.3: dependencies: ms: 2.1.3 decimal.js-light@2.5.1: {} - deep-is@0.1.4: {} + decimal.js@10.6.0: {} - define-data-property@1.1.4: - dependencies: - es-define-property: 1.0.1 - es-errors: 1.3.0 - gopd: 1.2.0 + deep-eql@5.0.2: {} - define-properties@1.2.1: - dependencies: - define-data-property: 1.1.4 - has-property-descriptors: 1.0.2 - object-keys: 1.1.1 + deep-is@0.1.4: {} - dequal@2.0.3: {} + delayed-stream@1.0.0: {} - detect-libc@2.1.2: - optional: true + dequal@2.0.3: {} detect-node-es@1.1.0: {} @@ -3858,9 +3967,9 @@ snapshots: dlv@1.1.3: {} - doctrine@2.1.0: - dependencies: - esutils: 2.0.3 + dom-accessibility-api@0.5.16: {} + + dom-accessibility-api@0.6.3: {} dom-helpers@5.2.1: dependencies: @@ -3873,91 +3982,15 @@ snapshots: es-errors: 1.3.0 gopd: 1.2.0 - eastasianwidth@0.2.0: {} + electron-to-chromium@1.5.302: {} - emoji-regex@8.0.0: {} - - emoji-regex@9.2.2: {} - - es-abstract@1.24.1: - dependencies: - array-buffer-byte-length: 1.0.2 - arraybuffer.prototype.slice: 1.0.4 - available-typed-arrays: 1.0.7 - call-bind: 1.0.8 - call-bound: 1.0.4 - data-view-buffer: 1.0.2 - data-view-byte-length: 1.0.2 - data-view-byte-offset: 1.0.1 - es-define-property: 1.0.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - es-set-tostringtag: 2.1.0 - es-to-primitive: 1.3.0 - function.prototype.name: 1.1.8 - get-intrinsic: 1.3.0 - get-proto: 1.0.1 - get-symbol-description: 1.1.0 - globalthis: 1.0.4 - gopd: 1.2.0 - has-property-descriptors: 1.0.2 - has-proto: 1.2.0 - has-symbols: 1.1.0 - hasown: 2.0.2 - internal-slot: 1.1.0 - is-array-buffer: 3.0.5 - is-callable: 1.2.7 - is-data-view: 1.0.2 - is-negative-zero: 2.0.3 - is-regex: 1.2.1 - is-set: 2.0.3 - is-shared-array-buffer: 1.0.4 - is-string: 1.1.1 - is-typed-array: 1.1.15 - is-weakref: 1.1.1 - math-intrinsics: 1.1.0 - object-inspect: 1.13.4 - object-keys: 1.1.1 - object.assign: 4.1.7 - own-keys: 1.0.1 - regexp.prototype.flags: 1.5.4 - safe-array-concat: 1.1.3 - safe-push-apply: 1.0.0 - safe-regex-test: 1.1.0 - set-proto: 1.0.0 - stop-iteration-iterator: 1.1.0 - string.prototype.trim: 1.2.10 - string.prototype.trimend: 1.0.9 - string.prototype.trimstart: 1.0.8 - typed-array-buffer: 1.0.3 - typed-array-byte-length: 1.0.3 - typed-array-byte-offset: 1.0.4 - typed-array-length: 1.0.7 - unbox-primitive: 1.1.0 - which-typed-array: 1.1.20 + entities@6.0.1: {} es-define-property@1.0.1: {} es-errors@1.3.0: {} - es-iterator-helpers@1.2.2: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-set-tostringtag: 2.1.0 - function-bind: 1.1.2 - get-intrinsic: 1.3.0 - globalthis: 1.0.4 - gopd: 1.2.0 - has-property-descriptors: 1.0.2 - has-proto: 1.2.0 - has-symbols: 1.1.0 - internal-slot: 1.1.0 - iterator.prototype: 1.1.5 - safe-array-concat: 1.1.3 + es-module-lexer@1.7.0: {} es-object-atoms@1.1.1: dependencies: @@ -3970,149 +4003,68 @@ snapshots: has-tostringtag: 1.0.2 hasown: 2.0.2 - es-shim-unscopables@1.1.0: - dependencies: - hasown: 2.0.2 - - es-to-primitive@1.3.0: - dependencies: - is-callable: 1.2.7 - is-date-object: 1.1.0 - is-symbol: 1.1.1 - - escape-string-regexp@4.0.0: {} - - eslint-config-next@15.2.4(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3): - dependencies: - '@next/eslint-plugin-next': 15.2.4 - '@rushstack/eslint-patch': 1.15.0 - '@typescript-eslint/eslint-plugin': 8.54.0(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - '@typescript-eslint/parser': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - eslint: 9.39.2(jiti@1.21.7) - eslint-import-resolver-node: 0.3.9 - eslint-import-resolver-typescript: 3.10.1(eslint-plugin-import@2.32.0)(eslint@9.39.2(jiti@1.21.7)) - eslint-plugin-import: 2.32.0(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.39.2(jiti@1.21.7)) - eslint-plugin-jsx-a11y: 6.10.2(eslint@9.39.2(jiti@1.21.7)) - eslint-plugin-react: 7.37.5(eslint@9.39.2(jiti@1.21.7)) - eslint-plugin-react-hooks: 5.2.0(eslint@9.39.2(jiti@1.21.7)) - optionalDependencies: - typescript: 5.9.3 - transitivePeerDependencies: - - eslint-import-resolver-webpack - - eslint-plugin-import-x - - supports-color - - eslint-config-prettier@9.1.2(eslint@9.39.2(jiti@1.21.7)): - dependencies: - eslint: 9.39.2(jiti@1.21.7) - - eslint-import-resolver-node@0.3.9: - dependencies: - debug: 3.2.7 - is-core-module: 2.16.1 - resolve: 1.22.11 - transitivePeerDependencies: - - supports-color - - eslint-import-resolver-typescript@3.10.1(eslint-plugin-import@2.32.0)(eslint@9.39.2(jiti@1.21.7)): - dependencies: - '@nolyfill/is-core-module': 1.0.39 - debug: 4.4.3 - eslint: 9.39.2(jiti@1.21.7) - get-tsconfig: 4.13.1 - is-bun-module: 2.0.0 - stable-hash: 0.0.5 - tinyglobby: 0.2.15 - unrs-resolver: 1.11.1 - optionalDependencies: - eslint-plugin-import: 2.32.0(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.39.2(jiti@1.21.7)) - transitivePeerDependencies: - - supports-color - - eslint-module-utils@2.12.1(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.10.1)(eslint@9.39.2(jiti@1.21.7)): - dependencies: - debug: 3.2.7 + esbuild@0.21.5: optionalDependencies: - '@typescript-eslint/parser': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - eslint: 9.39.2(jiti@1.21.7) - eslint-import-resolver-node: 0.3.9 - eslint-import-resolver-typescript: 3.10.1(eslint-plugin-import@2.32.0)(eslint@9.39.2(jiti@1.21.7)) - transitivePeerDependencies: - - supports-color - - eslint-plugin-import@2.32.0(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint-import-resolver-typescript@3.10.1)(eslint@9.39.2(jiti@1.21.7)): - dependencies: - '@rtsao/scc': 1.1.0 - array-includes: 3.1.9 - array.prototype.findlastindex: 1.2.6 - array.prototype.flat: 1.3.3 - array.prototype.flatmap: 1.3.3 - debug: 3.2.7 - doctrine: 2.1.0 - eslint: 9.39.2(jiti@1.21.7) - eslint-import-resolver-node: 0.3.9 - eslint-module-utils: 2.12.1(@typescript-eslint/parser@8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3))(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.10.1)(eslint@9.39.2(jiti@1.21.7)) - hasown: 2.0.2 - is-core-module: 2.16.1 - is-glob: 4.0.3 - minimatch: 3.1.2 - object.fromentries: 2.0.8 - object.groupby: 1.0.3 - object.values: 1.2.1 - semver: 6.3.1 - string.prototype.trimend: 1.0.9 - tsconfig-paths: 3.15.0 + '@esbuild/aix-ppc64': 0.21.5 + '@esbuild/android-arm': 0.21.5 + '@esbuild/android-arm64': 0.21.5 + '@esbuild/android-x64': 0.21.5 + '@esbuild/darwin-arm64': 0.21.5 + '@esbuild/darwin-x64': 0.21.5 + '@esbuild/freebsd-arm64': 0.21.5 + '@esbuild/freebsd-x64': 0.21.5 + '@esbuild/linux-arm': 0.21.5 + '@esbuild/linux-arm64': 0.21.5 + '@esbuild/linux-ia32': 0.21.5 + '@esbuild/linux-loong64': 0.21.5 + '@esbuild/linux-mips64el': 0.21.5 + '@esbuild/linux-ppc64': 0.21.5 + '@esbuild/linux-riscv64': 0.21.5 + '@esbuild/linux-s390x': 0.21.5 + '@esbuild/linux-x64': 0.21.5 + '@esbuild/netbsd-x64': 0.21.5 + '@esbuild/openbsd-x64': 0.21.5 + '@esbuild/sunos-x64': 0.21.5 + '@esbuild/win32-arm64': 0.21.5 + '@esbuild/win32-ia32': 0.21.5 + '@esbuild/win32-x64': 0.21.5 + + esbuild@0.25.12: optionalDependencies: - '@typescript-eslint/parser': 8.54.0(eslint@9.39.2(jiti@1.21.7))(typescript@5.9.3) - transitivePeerDependencies: - - eslint-import-resolver-typescript - - eslint-import-resolver-webpack - - supports-color - - eslint-plugin-jsx-a11y@6.10.2(eslint@9.39.2(jiti@1.21.7)): - dependencies: - aria-query: 5.3.2 - array-includes: 3.1.9 - array.prototype.flatmap: 1.3.3 - ast-types-flow: 0.0.8 - axe-core: 4.11.1 - axobject-query: 4.1.0 - damerau-levenshtein: 1.0.8 - emoji-regex: 9.2.2 - eslint: 9.39.2(jiti@1.21.7) - hasown: 2.0.2 - jsx-ast-utils: 3.3.5 - language-tags: 1.0.9 - minimatch: 3.1.2 - object.fromentries: 2.0.8 - safe-regex-test: 1.1.0 - string.prototype.includes: 2.0.1 + '@esbuild/aix-ppc64': 0.25.12 + '@esbuild/android-arm': 0.25.12 + '@esbuild/android-arm64': 0.25.12 + '@esbuild/android-x64': 0.25.12 + '@esbuild/darwin-arm64': 0.25.12 + '@esbuild/darwin-x64': 0.25.12 + '@esbuild/freebsd-arm64': 0.25.12 + '@esbuild/freebsd-x64': 0.25.12 + '@esbuild/linux-arm': 0.25.12 + '@esbuild/linux-arm64': 0.25.12 + '@esbuild/linux-ia32': 0.25.12 + '@esbuild/linux-loong64': 0.25.12 + '@esbuild/linux-mips64el': 0.25.12 + '@esbuild/linux-ppc64': 0.25.12 + '@esbuild/linux-riscv64': 0.25.12 + '@esbuild/linux-s390x': 0.25.12 + '@esbuild/linux-x64': 0.25.12 + '@esbuild/netbsd-arm64': 0.25.12 + '@esbuild/netbsd-x64': 0.25.12 + '@esbuild/openbsd-arm64': 0.25.12 + '@esbuild/openbsd-x64': 0.25.12 + '@esbuild/openharmony-arm64': 0.25.12 + '@esbuild/sunos-x64': 0.25.12 + '@esbuild/win32-arm64': 0.25.12 + '@esbuild/win32-ia32': 0.25.12 + '@esbuild/win32-x64': 0.25.12 + + escalade@3.2.0: {} - eslint-plugin-react-hooks@5.2.0(eslint@9.39.2(jiti@1.21.7)): - dependencies: - eslint: 9.39.2(jiti@1.21.7) + escape-string-regexp@4.0.0: {} - eslint-plugin-react@7.37.5(eslint@9.39.2(jiti@1.21.7)): + eslint-config-prettier@9.1.2(eslint@9.39.2(jiti@1.21.7)): dependencies: - array-includes: 3.1.9 - array.prototype.findlast: 1.2.5 - array.prototype.flatmap: 1.3.3 - array.prototype.tosorted: 1.1.4 - doctrine: 2.1.0 - es-iterator-helpers: 1.2.2 eslint: 9.39.2(jiti@1.21.7) - estraverse: 5.3.0 - hasown: 2.0.2 - jsx-ast-utils: 3.3.5 - minimatch: 3.1.2 - object.entries: 1.1.9 - object.fromentries: 2.0.8 - object.values: 1.2.1 - prop-types: 15.8.1 - resolve: 2.0.0-next.5 - semver: 6.3.1 - string.prototype.matchall: 4.0.12 - string.prototype.repeat: 1.0.0 eslint-scope@8.4.0: dependencies: @@ -4180,22 +4132,20 @@ snapshots: estraverse@5.3.0: {} + estree-walker@3.0.3: + dependencies: + '@types/estree': 1.0.8 + esutils@2.0.3: {} eventemitter3@4.0.7: {} + expect-type@1.3.0: {} + fast-deep-equal@3.1.3: {} fast-equals@5.4.0: {} - fast-glob@3.3.1: - dependencies: - '@nodelib/fs.stat': 2.0.5 - '@nodelib/fs.walk': 1.2.8 - glob-parent: 5.1.2 - merge2: 1.4.1 - micromatch: 4.0.8 - fast-glob@3.3.3: dependencies: '@nodelib/fs.stat': 2.0.5 @@ -4236,32 +4186,20 @@ snapshots: flatted@3.3.3: {} - for-each@0.3.5: - dependencies: - is-callable: 1.2.7 - - foreground-child@3.3.1: + form-data@4.0.5: dependencies: - cross-spawn: 7.0.6 - signal-exit: 4.1.0 + asynckit: 0.4.0 + combined-stream: 1.0.8 + es-set-tostringtag: 2.1.0 + hasown: 2.0.2 + mime-types: 2.1.35 fsevents@2.3.3: optional: true function-bind@1.1.2: {} - function.prototype.name@1.1.8: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - functions-have-names: 1.2.3 - hasown: 2.0.2 - is-callable: 1.2.7 - - functions-have-names@1.2.3: {} - - generator-function@2.0.1: {} + gensync@1.0.0-beta.2: {} get-intrinsic@1.3.0: dependencies: @@ -4283,16 +4221,6 @@ snapshots: dunder-proto: 1.0.1 es-object-atoms: 1.1.1 - get-symbol-description@1.1.0: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - get-intrinsic: 1.3.0 - - get-tsconfig@4.13.1: - dependencies: - resolve-pkg-maps: 1.0.0 - glob-parent@5.1.2: dependencies: is-glob: 4.0.3 @@ -4301,36 +4229,12 @@ snapshots: dependencies: is-glob: 4.0.3 - glob@11.1.0: - dependencies: - foreground-child: 3.3.1 - jackspeak: 4.1.1 - minimatch: 10.1.1 - minipass: 7.1.2 - package-json-from-dist: 1.0.1 - path-scurry: 2.0.1 - globals@14.0.0: {} - globalthis@1.0.4: - dependencies: - define-properties: 1.2.1 - gopd: 1.2.0 - gopd@1.2.0: {} - has-bigints@1.1.0: {} - has-flag@4.0.0: {} - has-property-descriptors@1.0.2: - dependencies: - es-define-property: 1.0.1 - - has-proto@1.2.0: - dependencies: - dunder-proto: 1.0.1 - has-symbols@1.1.0: {} has-tostringtag@1.0.2: @@ -4341,9 +4245,29 @@ snapshots: dependencies: function-bind: 1.1.2 - ignore@5.3.2: {} + html-encoding-sniffer@4.0.0: + dependencies: + whatwg-encoding: 3.1.1 + + http-proxy-agent@7.0.2: + dependencies: + agent-base: 7.1.4 + debug: 4.4.3 + transitivePeerDependencies: + - supports-color + + https-proxy-agent@7.0.6: + dependencies: + agent-base: 7.1.4 + debug: 4.4.3 + transitivePeerDependencies: + - supports-color + + iconv-lite@0.6.3: + dependencies: + safer-buffer: 2.1.2 - ignore@7.0.5: {} + ignore@5.3.2: {} import-fresh@3.3.1: dependencies: @@ -4352,149 +4276,30 @@ snapshots: imurmurhash@0.1.4: {} - internal-slot@1.1.0: - dependencies: - es-errors: 1.3.0 - hasown: 2.0.2 - side-channel: 1.1.0 + indent-string@4.0.0: {} internmap@2.0.3: {} - is-array-buffer@3.0.5: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - get-intrinsic: 1.3.0 - - is-async-function@2.1.1: - dependencies: - async-function: 1.0.0 - call-bound: 1.0.4 - get-proto: 1.0.1 - has-tostringtag: 1.0.2 - safe-regex-test: 1.1.0 - - is-bigint@1.1.0: - dependencies: - has-bigints: 1.1.0 - is-binary-path@2.1.0: dependencies: binary-extensions: 2.3.0 - is-boolean-object@1.2.2: - dependencies: - call-bound: 1.0.4 - has-tostringtag: 1.0.2 - - is-bun-module@2.0.0: - dependencies: - semver: 7.7.3 - - is-callable@1.2.7: {} - is-core-module@2.16.1: dependencies: hasown: 2.0.2 - is-data-view@1.0.2: - dependencies: - call-bound: 1.0.4 - get-intrinsic: 1.3.0 - is-typed-array: 1.1.15 - - is-date-object@1.1.0: - dependencies: - call-bound: 1.0.4 - has-tostringtag: 1.0.2 - is-extglob@2.1.1: {} - is-finalizationregistry@1.1.1: - dependencies: - call-bound: 1.0.4 - - is-fullwidth-code-point@3.0.0: {} - - is-generator-function@1.1.2: - dependencies: - call-bound: 1.0.4 - generator-function: 2.0.1 - get-proto: 1.0.1 - has-tostringtag: 1.0.2 - safe-regex-test: 1.1.0 - - is-glob@4.0.3: - dependencies: - is-extglob: 2.1.1 - - is-map@2.0.3: {} - - is-negative-zero@2.0.3: {} - - is-number-object@1.1.1: - dependencies: - call-bound: 1.0.4 - has-tostringtag: 1.0.2 - - is-number@7.0.0: {} - - is-regex@1.2.1: - dependencies: - call-bound: 1.0.4 - gopd: 1.2.0 - has-tostringtag: 1.0.2 - hasown: 2.0.2 - - is-set@2.0.3: {} - - is-shared-array-buffer@1.0.4: - dependencies: - call-bound: 1.0.4 - - is-string@1.1.1: - dependencies: - call-bound: 1.0.4 - has-tostringtag: 1.0.2 - - is-symbol@1.1.1: - dependencies: - call-bound: 1.0.4 - has-symbols: 1.1.0 - safe-regex-test: 1.1.0 - - is-typed-array@1.1.15: - dependencies: - which-typed-array: 1.1.20 - - is-weakmap@2.0.2: {} - - is-weakref@1.1.1: + is-glob@4.0.3: dependencies: - call-bound: 1.0.4 + is-extglob: 2.1.1 - is-weakset@2.0.4: - dependencies: - call-bound: 1.0.4 - get-intrinsic: 1.3.0 + is-number@7.0.0: {} - isarray@2.0.5: {} + is-potential-custom-element-name@1.0.1: {} isexe@2.0.0: {} - iterator.prototype@1.1.5: - dependencies: - define-data-property: 1.1.4 - es-object-atoms: 1.1.1 - get-intrinsic: 1.3.0 - get-proto: 1.0.1 - has-symbols: 1.1.0 - set-function-name: 2.0.2 - - jackspeak@4.1.1: - dependencies: - '@isaacs/cliui': 8.0.2 - jiti@1.21.7: {} js-tokens@4.0.0: {} @@ -4503,33 +4308,48 @@ snapshots: dependencies: argparse: 2.0.1 + jsdom@25.0.1: + dependencies: + cssstyle: 4.6.0 + data-urls: 5.0.0 + decimal.js: 10.6.0 + form-data: 4.0.5 + html-encoding-sniffer: 4.0.0 + http-proxy-agent: 7.0.2 + https-proxy-agent: 7.0.6 + is-potential-custom-element-name: 1.0.1 + nwsapi: 2.2.23 + parse5: 7.3.0 + rrweb-cssom: 0.7.1 + saxes: 6.0.0 + symbol-tree: 3.2.4 + tough-cookie: 5.1.2 + w3c-xmlserializer: 5.0.0 + webidl-conversions: 7.0.0 + whatwg-encoding: 3.1.1 + whatwg-mimetype: 4.0.0 + whatwg-url: 14.2.0 + ws: 8.19.0 + xml-name-validator: 5.0.0 + transitivePeerDependencies: + - bufferutil + - supports-color + - utf-8-validate + + jsesc@3.1.0: {} + json-buffer@3.0.1: {} json-schema-traverse@0.4.1: {} json-stable-stringify-without-jsonify@1.0.1: {} - json5@1.0.2: - dependencies: - minimist: 1.2.8 - - jsx-ast-utils@3.3.5: - dependencies: - array-includes: 3.1.9 - array.prototype.flat: 1.3.3 - object.assign: 4.1.7 - object.values: 1.2.1 + json5@2.2.3: {} keyv@4.5.4: dependencies: json-buffer: 3.0.1 - language-subtag-registry@0.3.23: {} - - language-tags@1.0.9: - dependencies: - language-subtag-registry: 0.3.23 - levn@0.4.1: dependencies: prelude-ls: 1.2.1 @@ -4551,12 +4371,24 @@ snapshots: dependencies: js-tokens: 4.0.0 - lru-cache@11.2.5: {} + loupe@3.2.1: {} + + lru-cache@10.4.3: {} + + lru-cache@5.1.1: + dependencies: + yallist: 3.1.1 lucide-react@0.411.0(react@19.1.0): dependencies: react: 19.1.0 + lz-string@1.5.0: {} + + magic-string@0.30.21: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.5 + math-intrinsics@1.1.0: {} merge2@1.4.1: {} @@ -4566,22 +4398,18 @@ snapshots: braces: 3.0.3 picomatch: 2.3.1 - minimatch@10.1.1: + mime-db@1.52.0: {} + + mime-types@2.1.35: dependencies: - '@isaacs/brace-expansion': 5.0.0 + mime-db: 1.52.0 + + min-indent@1.0.1: {} minimatch@3.1.2: dependencies: brace-expansion: 1.1.12 - minimatch@9.0.5: - dependencies: - brace-expansion: 2.0.2 - - minimist@1.2.8: {} - - minipass@7.1.2: {} - ms@2.1.3: {} mz@2.7.0: @@ -4592,84 +4420,18 @@ snapshots: nanoid@3.3.11: {} - napi-postinstall@0.3.4: {} - natural-compare@1.4.0: {} - next-themes@0.4.6(react-dom@19.1.0(react@19.1.0))(react@19.1.0): - dependencies: - react: 19.1.0 - react-dom: 19.1.0(react@19.1.0) - - next@15.4.10(react-dom@19.1.0(react@19.1.0))(react@19.1.0): - dependencies: - '@next/env': 15.4.10 - '@swc/helpers': 0.5.15 - caniuse-lite: 1.0.30001766 - postcss: 8.4.31 - react: 19.1.0 - react-dom: 19.1.0(react@19.1.0) - styled-jsx: 5.1.6(react@19.1.0) - optionalDependencies: - '@next/swc-darwin-arm64': 15.4.8 - '@next/swc-darwin-x64': 15.4.8 - '@next/swc-linux-arm64-gnu': 15.4.8 - '@next/swc-linux-arm64-musl': 15.4.8 - '@next/swc-linux-x64-gnu': 15.4.8 - '@next/swc-linux-x64-musl': 15.4.8 - '@next/swc-win32-arm64-msvc': 15.4.8 - '@next/swc-win32-x64-msvc': 15.4.8 - sharp: 0.34.5 - transitivePeerDependencies: - - '@babel/core' - - babel-plugin-macros + node-releases@2.0.27: {} normalize-path@3.0.0: {} + nwsapi@2.2.23: {} + object-assign@4.1.1: {} object-hash@3.0.0: {} - object-inspect@1.13.4: {} - - object-keys@1.1.1: {} - - object.assign@4.1.7: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-object-atoms: 1.1.1 - has-symbols: 1.1.0 - object-keys: 1.1.1 - - object.entries@1.1.9: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-object-atoms: 1.1.1 - - object.fromentries@2.0.8: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-object-atoms: 1.1.1 - - object.groupby@1.0.3: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - - object.values@1.2.1: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-object-atoms: 1.1.1 - optionator@0.9.4: dependencies: deep-is: 0.1.4 @@ -4679,12 +4441,6 @@ snapshots: type-check: 0.4.0 word-wrap: 1.2.5 - own-keys@1.0.1: - dependencies: - get-intrinsic: 1.3.0 - object-keys: 1.1.1 - safe-push-apply: 1.0.0 - p-limit@3.1.0: dependencies: yocto-queue: 0.1.0 @@ -4693,22 +4449,23 @@ snapshots: dependencies: p-limit: 3.1.0 - package-json-from-dist@1.0.1: {} - parent-module@1.0.1: dependencies: callsites: 3.1.0 + parse5@7.3.0: + dependencies: + entities: 6.0.1 + path-exists@4.0.0: {} path-key@3.1.1: {} path-parse@1.0.7: {} - path-scurry@2.0.1: - dependencies: - lru-cache: 11.2.5 - minipass: 7.1.2 + pathe@1.1.2: {} + + pathval@2.0.1: {} picocolors@1.1.1: {} @@ -4720,8 +4477,6 @@ snapshots: pirates@4.0.7: {} - possible-typed-array-names@1.1.0: {} - postcss-import@15.1.0(postcss@8.5.6): dependencies: postcss: 8.5.6 @@ -4754,12 +4509,6 @@ snapshots: postcss-value-parser@4.2.0: {} - postcss@8.4.31: - dependencies: - nanoid: 3.3.11 - picocolors: 1.1.1 - source-map-js: 1.2.1 - postcss@8.5.6: dependencies: nanoid: 3.3.11 @@ -4770,6 +4519,12 @@ snapshots: prettier@3.3.3: {} + pretty-format@27.5.1: + dependencies: + ansi-regex: 5.0.1 + ansi-styles: 5.2.0 + react-is: 17.0.2 + prop-types@15.8.1: dependencies: loose-envify: 1.4.0 @@ -4794,8 +4549,12 @@ snapshots: react-is@16.13.1: {} + react-is@17.0.2: {} + react-is@18.3.1: {} + react-refresh@0.17.0: {} + react-remove-scroll-bar@2.3.8(@types/react@19.1.0)(react@19.1.0): dependencies: react: 19.1.0 @@ -4815,6 +4574,18 @@ snapshots: optionalDependencies: '@types/react': 19.1.0 + react-router-dom@6.30.3(react-dom@19.1.0(react@19.1.0))(react@19.1.0): + dependencies: + '@remix-run/router': 1.23.2 + react: 19.1.0 + react-dom: 19.1.0(react@19.1.0) + react-router: 6.30.3(react@19.1.0) + + react-router@6.30.3(react@19.1.0): + dependencies: + '@remix-run/router': 1.23.2 + react: 19.1.0 + react-smooth@4.0.4(react-dom@19.1.0(react@19.1.0))(react@19.1.0): dependencies: fast-equals: 5.4.0 @@ -4867,162 +4638,77 @@ snapshots: tiny-invariant: 1.3.3 victory-vendor: 36.9.2 - reflect.getprototypeof@1.0.10: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - get-intrinsic: 1.3.0 - get-proto: 1.0.1 - which-builtin-type: 1.2.1 - - regexp.prototype.flags@1.5.4: + redent@3.0.0: dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-errors: 1.3.0 - get-proto: 1.0.1 - gopd: 1.2.0 - set-function-name: 2.0.2 + indent-string: 4.0.0 + strip-indent: 3.0.0 resolve-from@4.0.0: {} - resolve-pkg-maps@1.0.0: {} - resolve@1.22.11: dependencies: is-core-module: 2.16.1 path-parse: 1.0.7 supports-preserve-symlinks-flag: 1.0.0 - resolve@2.0.0-next.5: + reusify@1.1.0: {} + + rollup@4.59.0: dependencies: - is-core-module: 2.16.1 - path-parse: 1.0.7 - supports-preserve-symlinks-flag: 1.0.0 + '@types/estree': 1.0.8 + optionalDependencies: + '@rollup/rollup-android-arm-eabi': 4.59.0 + '@rollup/rollup-android-arm64': 4.59.0 + '@rollup/rollup-darwin-arm64': 4.59.0 + '@rollup/rollup-darwin-x64': 4.59.0 + '@rollup/rollup-freebsd-arm64': 4.59.0 + '@rollup/rollup-freebsd-x64': 4.59.0 + '@rollup/rollup-linux-arm-gnueabihf': 4.59.0 + '@rollup/rollup-linux-arm-musleabihf': 4.59.0 + '@rollup/rollup-linux-arm64-gnu': 4.59.0 + '@rollup/rollup-linux-arm64-musl': 4.59.0 + '@rollup/rollup-linux-loong64-gnu': 4.59.0 + '@rollup/rollup-linux-loong64-musl': 4.59.0 + '@rollup/rollup-linux-ppc64-gnu': 4.59.0 + '@rollup/rollup-linux-ppc64-musl': 4.59.0 + '@rollup/rollup-linux-riscv64-gnu': 4.59.0 + '@rollup/rollup-linux-riscv64-musl': 4.59.0 + '@rollup/rollup-linux-s390x-gnu': 4.59.0 + '@rollup/rollup-linux-x64-gnu': 4.59.0 + '@rollup/rollup-linux-x64-musl': 4.59.0 + '@rollup/rollup-openbsd-x64': 4.59.0 + '@rollup/rollup-openharmony-arm64': 4.59.0 + '@rollup/rollup-win32-arm64-msvc': 4.59.0 + '@rollup/rollup-win32-ia32-msvc': 4.59.0 + '@rollup/rollup-win32-x64-gnu': 4.59.0 + '@rollup/rollup-win32-x64-msvc': 4.59.0 + fsevents: 2.3.3 - reusify@1.1.0: {} + rrweb-cssom@0.7.1: {} + + rrweb-cssom@0.8.0: {} run-parallel@1.2.0: dependencies: queue-microtask: 1.2.3 - safe-array-concat@1.1.3: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - get-intrinsic: 1.3.0 - has-symbols: 1.1.0 - isarray: 2.0.5 - - safe-push-apply@1.0.0: - dependencies: - es-errors: 1.3.0 - isarray: 2.0.5 + safer-buffer@2.1.2: {} - safe-regex-test@1.1.0: + saxes@6.0.0: dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - is-regex: 1.2.1 + xmlchars: 2.2.0 scheduler@0.26.0: {} semver@6.3.1: {} - semver@7.7.3: {} - - set-function-length@1.2.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - function-bind: 1.1.2 - get-intrinsic: 1.3.0 - gopd: 1.2.0 - has-property-descriptors: 1.0.2 - - set-function-name@2.0.2: - dependencies: - define-data-property: 1.1.4 - es-errors: 1.3.0 - functions-have-names: 1.2.3 - has-property-descriptors: 1.0.2 - - set-proto@1.0.0: - dependencies: - dunder-proto: 1.0.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - - sharp@0.34.5: - dependencies: - '@img/colour': 1.0.0 - detect-libc: 2.1.2 - semver: 7.7.3 - optionalDependencies: - '@img/sharp-darwin-arm64': 0.34.5 - '@img/sharp-darwin-x64': 0.34.5 - '@img/sharp-libvips-darwin-arm64': 1.2.4 - '@img/sharp-libvips-darwin-x64': 1.2.4 - '@img/sharp-libvips-linux-arm': 1.2.4 - '@img/sharp-libvips-linux-arm64': 1.2.4 - '@img/sharp-libvips-linux-ppc64': 1.2.4 - '@img/sharp-libvips-linux-riscv64': 1.2.4 - '@img/sharp-libvips-linux-s390x': 1.2.4 - '@img/sharp-libvips-linux-x64': 1.2.4 - '@img/sharp-libvips-linuxmusl-arm64': 1.2.4 - '@img/sharp-libvips-linuxmusl-x64': 1.2.4 - '@img/sharp-linux-arm': 0.34.5 - '@img/sharp-linux-arm64': 0.34.5 - '@img/sharp-linux-ppc64': 0.34.5 - '@img/sharp-linux-riscv64': 0.34.5 - '@img/sharp-linux-s390x': 0.34.5 - '@img/sharp-linux-x64': 0.34.5 - '@img/sharp-linuxmusl-arm64': 0.34.5 - '@img/sharp-linuxmusl-x64': 0.34.5 - '@img/sharp-wasm32': 0.34.5 - '@img/sharp-win32-arm64': 0.34.5 - '@img/sharp-win32-ia32': 0.34.5 - '@img/sharp-win32-x64': 0.34.5 - optional: true - shebang-command@2.0.0: dependencies: shebang-regex: 3.0.0 shebang-regex@3.0.0: {} - side-channel-list@1.0.0: - dependencies: - es-errors: 1.3.0 - object-inspect: 1.13.4 - - side-channel-map@1.0.1: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - get-intrinsic: 1.3.0 - object-inspect: 1.13.4 - - side-channel-weakmap@1.0.2: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - get-intrinsic: 1.3.0 - object-inspect: 1.13.4 - side-channel-map: 1.0.1 - - side-channel@1.1.0: - dependencies: - es-errors: 1.3.0 - object-inspect: 1.13.4 - side-channel-list: 1.0.0 - side-channel-map: 1.0.1 - side-channel-weakmap: 1.0.2 - - signal-exit@4.1.0: {} + siginfo@2.0.0: {} sonner@2.0.7(react-dom@19.1.0(react@19.1.0))(react@19.1.0): dependencies: @@ -5031,92 +4717,16 @@ snapshots: source-map-js@1.2.1: {} - stable-hash@0.0.5: {} - - stop-iteration-iterator@1.1.0: - dependencies: - es-errors: 1.3.0 - internal-slot: 1.1.0 - - string-width@4.2.3: - dependencies: - emoji-regex: 8.0.0 - is-fullwidth-code-point: 3.0.0 - strip-ansi: 6.0.1 - - string-width@5.1.2: - dependencies: - eastasianwidth: 0.2.0 - emoji-regex: 9.2.2 - strip-ansi: 7.1.2 - - string.prototype.includes@2.0.1: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-abstract: 1.24.1 - - string.prototype.matchall@4.0.12: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-errors: 1.3.0 - es-object-atoms: 1.1.1 - get-intrinsic: 1.3.0 - gopd: 1.2.0 - has-symbols: 1.1.0 - internal-slot: 1.1.0 - regexp.prototype.flags: 1.5.4 - set-function-name: 2.0.2 - side-channel: 1.1.0 - - string.prototype.repeat@1.0.0: - dependencies: - define-properties: 1.2.1 - es-abstract: 1.24.1 - - string.prototype.trim@1.2.10: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-data-property: 1.1.4 - define-properties: 1.2.1 - es-abstract: 1.24.1 - es-object-atoms: 1.1.1 - has-property-descriptors: 1.0.2 - - string.prototype.trimend@1.0.9: - dependencies: - call-bind: 1.0.8 - call-bound: 1.0.4 - define-properties: 1.2.1 - es-object-atoms: 1.1.1 - - string.prototype.trimstart@1.0.8: - dependencies: - call-bind: 1.0.8 - define-properties: 1.2.1 - es-object-atoms: 1.1.1 + stackback@0.0.2: {} - strip-ansi@6.0.1: - dependencies: - ansi-regex: 5.0.1 + std-env@3.10.0: {} - strip-ansi@7.1.2: + strip-indent@3.0.0: dependencies: - ansi-regex: 6.2.2 - - strip-bom@3.0.0: {} + min-indent: 1.0.1 strip-json-comments@3.1.1: {} - styled-jsx@5.1.6(react@19.1.0): - dependencies: - client-only: 0.0.1 - react: 19.1.0 - sucrase@3.35.1: dependencies: '@jridgewell/gen-mapping': 0.3.13 @@ -5139,6 +4749,8 @@ snapshots: react: 19.1.0 use-sync-external-store: 1.6.0(react@19.1.0) + symbol-tree@3.2.4: {} + tailwind-merge@2.6.1: {} tailwindcss-animate@1.0.7(tailwindcss@3.4.19(yaml@2.8.0)): @@ -5183,29 +4795,42 @@ snapshots: tiny-invariant@1.3.3: {} + tinybench@2.9.0: {} + + tinyexec@0.3.2: {} + tinyglobby@0.2.15: dependencies: fdir: 6.5.0(picomatch@4.0.3) picomatch: 4.0.3 + tinypool@1.1.1: {} + + tinyrainbow@1.2.0: {} + + tinyspy@3.0.2: {} + + tldts-core@6.1.86: {} + + tldts@6.1.86: + dependencies: + tldts-core: 6.1.86 + to-regex-range@5.0.1: dependencies: is-number: 7.0.0 toggle-selection@1.0.6: {} - ts-api-utils@2.4.0(typescript@5.9.3): + tough-cookie@5.1.2: dependencies: - typescript: 5.9.3 - - ts-interface-checker@0.1.13: {} + tldts: 6.1.86 - tsconfig-paths@3.15.0: + tr46@5.1.1: dependencies: - '@types/json5': 0.0.29 - json5: 1.0.2 - minimist: 1.2.8 - strip-bom: 3.0.0 + punycode: 2.3.1 + + ts-interface-checker@0.1.13: {} tslib@2.8.1: {} @@ -5213,73 +4838,15 @@ snapshots: dependencies: prelude-ls: 1.2.1 - typed-array-buffer@1.0.3: - dependencies: - call-bound: 1.0.4 - es-errors: 1.3.0 - is-typed-array: 1.1.15 - - typed-array-byte-length@1.0.3: - dependencies: - call-bind: 1.0.8 - for-each: 0.3.5 - gopd: 1.2.0 - has-proto: 1.2.0 - is-typed-array: 1.1.15 - - typed-array-byte-offset@1.0.4: - dependencies: - available-typed-arrays: 1.0.7 - call-bind: 1.0.8 - for-each: 0.3.5 - gopd: 1.2.0 - has-proto: 1.2.0 - is-typed-array: 1.1.15 - reflect.getprototypeof: 1.0.10 - - typed-array-length@1.0.7: - dependencies: - call-bind: 1.0.8 - for-each: 0.3.5 - gopd: 1.2.0 - is-typed-array: 1.1.15 - possible-typed-array-names: 1.1.0 - reflect.getprototypeof: 1.0.10 - typescript@5.9.3: {} - unbox-primitive@1.1.0: - dependencies: - call-bound: 1.0.4 - has-bigints: 1.1.0 - has-symbols: 1.1.0 - which-boxed-primitive: 1.1.1 - undici-types@6.21.0: {} - unrs-resolver@1.11.1: + update-browserslist-db@1.2.3(browserslist@4.28.1): dependencies: - napi-postinstall: 0.3.4 - optionalDependencies: - '@unrs/resolver-binding-android-arm-eabi': 1.11.1 - '@unrs/resolver-binding-android-arm64': 1.11.1 - '@unrs/resolver-binding-darwin-arm64': 1.11.1 - '@unrs/resolver-binding-darwin-x64': 1.11.1 - '@unrs/resolver-binding-freebsd-x64': 1.11.1 - '@unrs/resolver-binding-linux-arm-gnueabihf': 1.11.1 - '@unrs/resolver-binding-linux-arm-musleabihf': 1.11.1 - '@unrs/resolver-binding-linux-arm64-gnu': 1.11.1 - '@unrs/resolver-binding-linux-arm64-musl': 1.11.1 - '@unrs/resolver-binding-linux-ppc64-gnu': 1.11.1 - '@unrs/resolver-binding-linux-riscv64-gnu': 1.11.1 - '@unrs/resolver-binding-linux-riscv64-musl': 1.11.1 - '@unrs/resolver-binding-linux-s390x-gnu': 1.11.1 - '@unrs/resolver-binding-linux-x64-gnu': 1.11.1 - '@unrs/resolver-binding-linux-x64-musl': 1.11.1 - '@unrs/resolver-binding-wasm32-wasi': 1.11.1 - '@unrs/resolver-binding-win32-arm64-msvc': 1.11.1 - '@unrs/resolver-binding-win32-ia32-msvc': 1.11.1 - '@unrs/resolver-binding-win32-x64-msvc': 1.11.1 + browserslist: 4.28.1 + escalade: 3.2.0 + picocolors: 1.1.1 uri-js@4.4.1: dependencies: @@ -5323,64 +4890,118 @@ snapshots: d3-time: 3.1.0 d3-timer: 3.0.1 - which-boxed-primitive@1.1.1: + vite-node@2.1.9(@types/node@20.19.30): dependencies: - is-bigint: 1.1.0 - is-boolean-object: 1.2.2 - is-number-object: 1.1.1 - is-string: 1.1.1 - is-symbol: 1.1.1 + cac: 6.7.14 + debug: 4.4.3 + es-module-lexer: 1.7.0 + pathe: 1.1.2 + vite: 5.4.21(@types/node@20.19.30) + transitivePeerDependencies: + - '@types/node' + - less + - lightningcss + - sass + - sass-embedded + - stylus + - sugarss + - supports-color + - terser - which-builtin-type@1.2.1: + vite@5.4.21(@types/node@20.19.30): dependencies: - call-bound: 1.0.4 - function.prototype.name: 1.1.8 - has-tostringtag: 1.0.2 - is-async-function: 2.1.1 - is-date-object: 1.1.0 - is-finalizationregistry: 1.1.1 - is-generator-function: 1.1.2 - is-regex: 1.2.1 - is-weakref: 1.1.1 - isarray: 2.0.5 - which-boxed-primitive: 1.1.1 - which-collection: 1.0.2 - which-typed-array: 1.1.20 - - which-collection@1.0.2: - dependencies: - is-map: 2.0.3 - is-set: 2.0.3 - is-weakmap: 2.0.2 - is-weakset: 2.0.4 - - which-typed-array@1.1.20: - dependencies: - available-typed-arrays: 1.0.7 - call-bind: 1.0.8 - call-bound: 1.0.4 - for-each: 0.3.5 - get-proto: 1.0.1 - gopd: 1.2.0 - has-tostringtag: 1.0.2 + esbuild: 0.21.5 + postcss: 8.5.6 + rollup: 4.59.0 + optionalDependencies: + '@types/node': 20.19.30 + fsevents: 2.3.3 + + vite@6.4.1(@types/node@20.19.30)(jiti@1.21.7)(yaml@2.8.0): + dependencies: + esbuild: 0.25.12 + fdir: 6.5.0(picomatch@4.0.3) + picomatch: 4.0.3 + postcss: 8.5.6 + rollup: 4.59.0 + tinyglobby: 0.2.15 + optionalDependencies: + '@types/node': 20.19.30 + fsevents: 2.3.3 + jiti: 1.21.7 + yaml: 2.8.0 + + vitest@2.1.9(@types/node@20.19.30)(jsdom@25.0.1): + dependencies: + '@vitest/expect': 2.1.9 + '@vitest/mocker': 2.1.9(vite@5.4.21(@types/node@20.19.30)) + '@vitest/pretty-format': 2.1.9 + '@vitest/runner': 2.1.9 + '@vitest/snapshot': 2.1.9 + '@vitest/spy': 2.1.9 + '@vitest/utils': 2.1.9 + chai: 5.3.3 + debug: 4.4.3 + expect-type: 1.3.0 + magic-string: 0.30.21 + pathe: 1.1.2 + std-env: 3.10.0 + tinybench: 2.9.0 + tinyexec: 0.3.2 + tinypool: 1.1.1 + tinyrainbow: 1.2.0 + vite: 5.4.21(@types/node@20.19.30) + vite-node: 2.1.9(@types/node@20.19.30) + why-is-node-running: 2.3.0 + optionalDependencies: + '@types/node': 20.19.30 + jsdom: 25.0.1 + transitivePeerDependencies: + - less + - lightningcss + - msw + - sass + - sass-embedded + - stylus + - sugarss + - supports-color + - terser + + w3c-xmlserializer@5.0.0: + dependencies: + xml-name-validator: 5.0.0 + + webidl-conversions@7.0.0: {} + + whatwg-encoding@3.1.1: + dependencies: + iconv-lite: 0.6.3 + + whatwg-mimetype@4.0.0: {} + + whatwg-url@14.2.0: + dependencies: + tr46: 5.1.1 + webidl-conversions: 7.0.0 which@2.0.2: dependencies: isexe: 2.0.0 + why-is-node-running@2.3.0: + dependencies: + siginfo: 2.0.0 + stackback: 0.0.2 + word-wrap@1.2.5: {} - wrap-ansi@7.0.0: - dependencies: - ansi-styles: 4.3.0 - string-width: 4.2.3 - strip-ansi: 6.0.1 + ws@8.19.0: {} - wrap-ansi@8.1.0: - dependencies: - ansi-styles: 6.2.3 - string-width: 5.1.2 - strip-ansi: 7.1.2 + xml-name-validator@5.0.0: {} + + xmlchars@2.2.0: {} + + yallist@3.1.1: {} yaml@2.8.0: optional: true diff --git a/webapp/src/api/client.ts b/webapp/src/api/client.ts new file mode 100644 index 000000000..68caa5f39 --- /dev/null +++ b/webapp/src/api/client.ts @@ -0,0 +1,68 @@ +import { ZodSchema } from "zod"; +import { ApiError, ValidationError } from "./errors"; + +const API_BASE = import.meta.env.VITE_API_URL; + +export async function fetchApi( + endpoint: string, + schema: ZodSchema, + options?: RequestInit, +): Promise { + const response = await fetch(`${API_BASE}${endpoint}`, { + ...options, + credentials: "include", + headers: { + "Content-Type": "application/json", + ...(options?.headers || {}), + }, + }); + + if (!response.ok) { + let detail = `${response.status} ${response.statusText}`; + try { + const body = await response.json(); + detail = body.detail || detail; + } catch { + // ignore JSON parse errors + } + throw new ApiError(detail, response.status, endpoint); + } + + if (response.status === 204) return undefined as T; + + const data = await response.json(); + const parsed = schema.safeParse(data); + if (!parsed.success) { + throw new ValidationError( + `Invalid response from ${endpoint}`, + parsed.error, + endpoint, + ); + } + return parsed.data; +} + +export async function fetchApiVoid( + endpoint: string, + options?: RequestInit, +): Promise { + const response = await fetch(`${API_BASE}${endpoint}`, { + ...options, + credentials: "include", + headers: { + "Content-Type": "application/json", + ...(options?.headers || {}), + }, + }); + + if (!response.ok) { + let detail = `${response.status} ${response.statusText}`; + try { + const body = await response.json(); + detail = body.detail || detail; + } catch { + // ignore JSON parse errors + } + throw new ApiError(detail, response.status, endpoint); + } +} diff --git a/webapp/src/api/errors.ts b/webapp/src/api/errors.ts new file mode 100644 index 000000000..68b738bf1 --- /dev/null +++ b/webapp/src/api/errors.ts @@ -0,0 +1,41 @@ +import { toast } from "sonner"; +import { ZodError } from "zod"; + +export class ApiError extends Error { + constructor( + message: string, + public status: number, + public endpoint: string, + ) { + super(message); + this.name = "ApiError"; + } +} + +export class ValidationError extends Error { + constructor( + message: string, + public zodError: ZodError, + public endpoint: string, + ) { + super(message); + this.name = "ValidationError"; + } +} + +export function handleError(error: unknown): void { + if (error instanceof ApiError) { + if (error.status === 401) { + window.location.href = `${import.meta.env.VITE_API_URL}/auth/login?redirect=${import.meta.env.VITE_BASE_URL}/home?auth=true`; + return; + } + toast.error(error.message); + } else if (error instanceof ValidationError) { + console.error(`[ValidationError] ${error.endpoint}:`, error.zodError.issues); + toast.error("Received unexpected data from the server"); + } else if (error instanceof Error) { + toast.error(error.message); + } else { + toast.error("An unexpected error occurred"); + } +} diff --git a/webapp/src/api/experiments.ts b/webapp/src/api/experiments.ts new file mode 100644 index 000000000..b11bab696 --- /dev/null +++ b/webapp/src/api/experiments.ts @@ -0,0 +1,67 @@ +import { fetchApi } from "./client"; +import { + Experiment, + ExperimentSchema, + ExperimentReport, + ExperimentReportSchema, +} from "./schemas"; +import { DateRange } from "react-day-picker"; + +export async function createExperiment( + experiment: Experiment, +): Promise { + return await fetchApi("/experiments", ExperimentSchema, { + method: "POST", + body: JSON.stringify(experiment), + }); +} + +export async function getExperiments(projectId: string): Promise { + try { + const result = await fetchApi( + `/projects/${projectId}/experiments`, + ExperimentSchema.array(), + ); + return result.map((experiment) => ({ + id: experiment.id, + name: experiment.name, + description: experiment.description, + project_id: experiment.project_id, + timestamp: experiment.timestamp, + })); + } catch { + return []; + } +} + +export async function getProjectEmissionsByExperiment( + projectId: string, + dateRange: DateRange, +): Promise { + let url = `/projects/${projectId}/experiments/sums`; + + if (dateRange?.from || dateRange?.to) { + const params = new URLSearchParams(); + if (dateRange.from) { + params.append("start_date", dateRange.from.toISOString()); + } + if (dateRange.to) { + params.append("end_date", dateRange.to.toISOString()); + } + url += `?${params.toString()}`; + } + + try { + const result = await fetchApi(url, ExperimentReportSchema.array()); + return result.map((experimentReport) => ({ + experiment_id: experimentReport.experiment_id, + description: experimentReport.description, + name: experimentReport.name, + emissions: experimentReport.emissions, + energy_consumed: experimentReport.energy_consumed, + duration: experimentReport.duration, + })); + } catch { + return []; + } +} diff --git a/webapp/src/api/organizations.ts b/webapp/src/api/organizations.ts new file mode 100644 index 000000000..2b92218cf --- /dev/null +++ b/webapp/src/api/organizations.ts @@ -0,0 +1,43 @@ +import { fetchApi } from "./client"; +import { + Organization, + OrganizationSchema, + OrganizationReport, + OrganizationReportSchema, +} from "./schemas"; +import { DateRange } from "react-day-picker"; + +export async function getOrganizationEmissionsByProject( + organizationId: string, + dateRange: DateRange | undefined, +): Promise { + let endpoint = `/organizations/${organizationId}/sums`; + + if (dateRange?.from && dateRange?.to) { + endpoint += `?start_date=${dateRange.from.toISOString()}&end_date=${dateRange.to.toISOString()}`; + } + + try { + return await fetchApi(endpoint, OrganizationReportSchema); + } catch { + return { name: "", emissions: 0, energy_consumed: 0, duration: 0 }; + } +} + +export async function getOrganizations(): Promise { + try { + return await fetchApi("/organizations", OrganizationSchema.array()); + } catch { + return []; + } +} + +export async function createOrganization(organization: { + name: string; + description: string; +}): Promise { + return await fetchApi("/organizations", OrganizationSchema, { + method: "POST", + body: JSON.stringify(organization), + }); +} diff --git a/webapp/src/api/projectTokens.ts b/webapp/src/api/projectTokens.ts new file mode 100644 index 000000000..f34cc9b31 --- /dev/null +++ b/webapp/src/api/projectTokens.ts @@ -0,0 +1,39 @@ +import { fetchApi, fetchApiVoid } from "./client"; +import { IProjectToken, ProjectTokenSchema } from "./schemas"; + +export async function getProjectTokens( + projectId: string, +): Promise { + try { + return await fetchApi( + `/projects/${projectId}/api-tokens`, + ProjectTokenSchema.array(), + ); + } catch { + return []; + } +} + +export async function createProjectToken( + projectId: string, + tokenName: string, + access?: number, +): Promise { + return await fetchApi( + `/projects/${projectId}/api-tokens`, + ProjectTokenSchema, + { + method: "POST", + body: JSON.stringify({ name: tokenName, access }), + }, + ); +} + +export async function deleteProjectToken( + projectId: string, + tokenId: string, +): Promise { + await fetchApiVoid(`/projects/${projectId}/api-tokens/${tokenId}`, { + method: "DELETE", + }); +} diff --git a/webapp/src/api/projects.ts b/webapp/src/api/projects.ts new file mode 100644 index 000000000..45b129bfe --- /dev/null +++ b/webapp/src/api/projects.ts @@ -0,0 +1,58 @@ +import { fetchApi, fetchApiVoid } from "./client"; +import { + Project, + ProjectSchema, + ProjectInputs, +} from "./schemas"; + +export async function createProject( + organizationId: string, + project: { name: string; description: string }, +): Promise { + return await fetchApi("/projects", ProjectSchema, { + method: "POST", + body: JSON.stringify({ + ...project, + organization_id: organizationId, + }), + }); +} + +export async function updateProject( + projectId: string, + project: ProjectInputs, +): Promise { + return await fetchApi(`/projects/${projectId}`, ProjectSchema, { + method: "PATCH", + body: JSON.stringify(project), + }); +} + +export async function getProjects( + organizationId: string, +): Promise { + try { + return await fetchApi( + `/projects?organization=${organizationId}`, + ProjectSchema.array(), + ); + } catch { + return []; + } +} + +export async function getOneProject( + projectId: string, +): Promise { + try { + return await fetchApi(`/projects/${projectId}`, ProjectSchema); + } catch { + return null; + } +} + +export async function deleteProject(projectId: string): Promise { + await fetchApiVoid(`/projects/${projectId}`, { + method: "DELETE", + }); +} diff --git a/webapp/src/api/runs.ts b/webapp/src/api/runs.ts new file mode 100644 index 000000000..00837f690 --- /dev/null +++ b/webapp/src/api/runs.ts @@ -0,0 +1,120 @@ +import { z } from "zod"; +import { fetchApi } from "./client"; +import { + Emission, + EmissionSchema, + EmissionsTimeSeries, + RunMetadata, + RunMetadataSchema, + RunReport, + RunReportSchema, +} from "./schemas"; + +export async function getRunMetadata( + runId: string, +): Promise { + try { + return await fetchApi(`/runs/${runId}`, RunMetadataSchema); + } catch { + return null; + } +} + +export async function getRunEmissionsByExperiment( + experimentId: string, + startDate: string, + endDate: string, +): Promise { + if (!experimentId || experimentId === "") { + return []; + } + + try { + const result = await fetchApi( + `/experiments/${experimentId}/runs/sums?start_date=${startDate}&end_date=${endDate}`, + z.array(z.object({ + run_id: z.string(), + emissions: z.number(), + timestamp: z.string(), + energy_consumed: z.number(), + duration: z.number(), + })), + ); + + return result.map((runReport) => ({ + runId: runReport.run_id, + emissions: runReport.emissions, + timestamp: runReport.timestamp, + energy_consumed: runReport.energy_consumed, + duration: runReport.duration, + })); + } catch { + return []; + } +} + +export async function getEmissionsTimeSeries( + runId: string, +): Promise { + try { + const [runMetadataData, emissionsData] = await Promise.all([ + fetchApi(`/runs/${runId}`, RunMetadataSchema), + fetchApi( + `/runs/${runId}/emissions`, + z.object({ + items: z.array( + z.object({ + run_id: z.string(), + timestamp: z.string(), + emissions_sum: z.number(), + emissions_rate: z.number(), + cpu_power: z.number(), + gpu_power: z.number(), + ram_power: z.number(), + cpu_energy: z.number(), + gpu_energy: z.number(), + ram_energy: z.number(), + energy_consumed: z.number(), + }), + ), + }), + ), + ]); + + const metadata: RunMetadata = { + timestamp: runMetadataData.timestamp, + experiment_id: runMetadataData.experiment_id, + os: runMetadataData.os, + python_version: runMetadataData.python_version, + codecarbon_version: runMetadataData.codecarbon_version, + cpu_count: runMetadataData.cpu_count, + cpu_model: runMetadataData.cpu_model, + gpu_count: runMetadataData.gpu_count, + gpu_model: runMetadataData.gpu_model, + longitude: runMetadataData.longitude, + latitude: runMetadataData.latitude, + region: runMetadataData.region, + provider: runMetadataData.provider, + ram_total_size: runMetadataData.ram_total_size, + tracking_mode: runMetadataData.tracking_mode, + }; + + const emissions: Emission[] = emissionsData.items.map((item) => ({ + emission_id: item.run_id, + timestamp: item.timestamp, + emissions_sum: item.emissions_sum, + emissions_rate: item.emissions_rate, + cpu_power: item.cpu_power, + gpu_power: item.gpu_power, + ram_power: item.ram_power, + cpu_energy: item.cpu_energy, + gpu_energy: item.gpu_energy, + ram_energy: item.ram_energy, + energy_consumed: item.energy_consumed, + })); + + return { runId, emissions, metadata }; + } catch { + return { runId, emissions: [], metadata: null }; + } +} diff --git a/webapp/src/api/schemas.ts b/webapp/src/api/schemas.ts new file mode 100644 index 000000000..9b7dc460d --- /dev/null +++ b/webapp/src/api/schemas.ts @@ -0,0 +1,182 @@ +import { z } from "zod"; + +export const OrganizationSchema = z.object({ + id: z.string(), + name: z.string(), + description: z.string(), +}); +export type Organization = z.infer; + +export const UserSchema = z.object({ + id: z.string(), + email: z.string(), + name: z.string(), + organizations: z.array(z.string()), + is_active: z.boolean(), +}); +export type User = z.infer; + +export const ProjectSchema = z.object({ + id: z.string(), + name: z.string(), + description: z.string(), + public: z.boolean(), + organizationId: z.string(), + experiments: z.array(z.string()), +}); +export type Project = z.infer; + +export const ProjectInputsSchema = z.object({ + name: z.string(), + description: z.string(), + public: z.boolean(), +}); +export type ProjectInputs = z.infer; + +export const ProjectTokenSchema = z.object({ + id: z.string(), + project_id: z.string(), + last_used: z.string().nullable(), + name: z.string(), + token: z.string(), + access: z.number(), +}); +export type IProjectToken = z.infer; + +export const ExperimentSchema = z.object({ + id: z.string().optional(), + timestamp: z.string().optional(), + name: z.string(), + description: z.string(), + on_cloud: z.boolean().optional(), + project_id: z.string(), + country_name: z.string().optional(), + country_iso_code: z.string().optional(), + region: z.string().optional(), + cloud_provider: z.string().optional(), + cloud_region: z.string().optional(), +}); +export type Experiment = z.infer; + +export const ExperimentReportSchema = z.object({ + experiment_id: z.string(), + name: z.string(), + emissions: z.number(), + energy_consumed: z.number(), + duration: z.number(), + description: z.string().optional(), +}); +export type ExperimentReport = z.infer; + +export const RunReportSchema = z.object({ + runId: z.string(), + emissions: z.number(), + timestamp: z.string(), + energy_consumed: z.number(), + duration: z.number(), +}); +export type RunReport = z.infer; + +export const EmissionSchema = z.object({ + emission_id: z.string(), + timestamp: z.string(), + emissions_sum: z.number(), + emissions_rate: z.number(), + cpu_power: z.number(), + gpu_power: z.number(), + ram_power: z.number(), + cpu_energy: z.number(), + gpu_energy: z.number(), + ram_energy: z.number(), + energy_consumed: z.number(), +}); +export type Emission = z.infer; + +export const RunMetadataSchema = z.object({ + timestamp: z.string(), + experiment_id: z.string(), + os: z.string(), + python_version: z.string(), + codecarbon_version: z.string(), + cpu_count: z.number(), + cpu_model: z.string(), + gpu_count: z.number(), + gpu_model: z.string(), + longitude: z.number(), + latitude: z.number(), + region: z.string(), + provider: z.string(), + ram_total_size: z.number(), + tracking_mode: z.string(), +}); +export type RunMetadata = z.infer; + +export const OrganizationReportSchema = z.object({ + name: z.string(), + emissions: z.number(), + energy_consumed: z.number(), + duration: z.number(), +}); +export type OrganizationReport = z.infer; + +export const EmissionsTimeSeriesSchema = z.object({ + runId: z.string(), + emissions: z.array(EmissionSchema), + metadata: RunMetadataSchema.nullable(), +}); +export type EmissionsTimeSeries = z.infer; + +// Dashboard prop types (not API responses, but shared across components) +export interface RadialChartData { + energy: { label: string; value: number }; + emissions: { label: string; value: number }; + duration: { label: string; value: number }; +} + +export interface ConvertedValues { + citizen: string; + transportation: string; + tvTime: string; +} + +export interface ProjectDashboardProps { + project: Project; + date: import("react-day-picker").DateRange; + onDateChange: (newDate: import("react-day-picker").DateRange | undefined) => void; + radialChartData: RadialChartData; + convertedValues: ConvertedValues; + experimentsReportData: ExperimentReport[]; + projectExperiments: Experiment[]; + runData: { + experimentId: string; + startDate: string; + endDate: string; + }; + selectedExperimentId: string; + selectedRunId: string; + onExperimentClick: (experimentId: string) => void; + onRunClick: (runId: string) => void; + onSettingsClick: () => void; + onRefresh: () => void; + isLoading?: boolean; +} + +export interface PublicProjectDashboardProps { + project: Project; + date: import("react-day-picker").DateRange; + onDateChange: (newDate: import("react-day-picker").DateRange | undefined) => void; + radialChartData: RadialChartData; + convertedValues: ConvertedValues; + experimentsReportData: ExperimentReport[]; + projectExperiments: Experiment[]; + runData: { + experimentId: string; + startDate: string; + endDate: string; + }; + selectedExperimentId: string; + selectedRunId: string; + onExperimentClick: (experimentId: string) => void; + onRunClick: (runId: string) => void; + isLoading?: boolean; +} diff --git a/webapp/src/api/swr.ts b/webapp/src/api/swr.ts new file mode 100644 index 000000000..fe5c98e9f --- /dev/null +++ b/webapp/src/api/swr.ts @@ -0,0 +1,22 @@ +import { ZodSchema } from "zod"; +import { fetchApi } from "./client"; +import { handleError } from "./errors"; + +const API_BASE = import.meta.env.VITE_API_URL; + +export const fetcher = async (url: string) => { + const res = await fetch(`${API_BASE}${url}`, { credentials: "include" }); + if (!res.ok) throw new Error(`Failed to fetch: ${res.statusText}`); + return res.json(); +}; + +export function createValidatedFetcher(schema: ZodSchema) { + return (url: string) => fetchApi(url, schema); +} + +export const swrConfig = { + onError: handleError, + dedupingInterval: 5000, + focusThrottleInterval: 30000, + revalidateOnFocus: false, +}; diff --git a/webapp/src/app/(dashboard)/[organizationId]/loading.tsx b/webapp/src/app/(dashboard)/[organizationId]/loading.tsx deleted file mode 100644 index 4087e424c..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/loading.tsx +++ /dev/null @@ -1 +0,0 @@ -export { default } from "@/components/loader"; // Adjust the relative path accordingly diff --git a/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx b/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx deleted file mode 100644 index f50a7527f..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/members/members-list.tsx +++ /dev/null @@ -1,148 +0,0 @@ -"use client"; - -import CustomRow from "@/components/custom-row"; -import { Button } from "@/components/ui/button"; -import { Card } from "@/components/ui/card"; -import { Input } from "@/components/ui/input"; -import { Table, TableBody } from "@/components/ui/table"; -import { User } from "@/types/user"; -import { useRouter } from "next/navigation"; -import { useState } from "react"; -import { z } from "zod"; -import { toast } from "sonner"; -import { fetchApi } from "@/utils/api"; - -export default function MembersList({ - users, - organizationId, -}: { - users: User[]; - organizationId: string; -}) { - const router = useRouter(); - const [isDialogOpen, setDialogOpen] = useState(false); - const [error, setError] = useState(null); - const [isLoading, setIsLoading] = useState(false); - - const form = { email: undefined }; - - const emailSchema = z.object({ - email: z.string().email("Please enter a valid email address"), - }); - - async function addUser() { - try { - setIsLoading(true); - const email = ( - document.getElementById("emailInput") as HTMLInputElement - ).value; - - // Validate email - emailSchema.parse({ email }); - setError(null); - - const body = JSON.stringify({ email }); - - await toast - .promise( - fetchApi(`/organizations/${organizationId}/add-user`, { - method: "POST", - body: body, - }).then(async (result) => { - if (!result) { - throw new Error("Failed to add user"); - } - return result; - }), - { - loading: `Adding user ${email}...`, - success: `User ${email} added successfully`, - error: (err) => `${err.message}`, - }, - ) - .unwrap(); - - // On success - router.refresh(); - setDialogOpen(false); - } catch (err) { - if (err instanceof z.ZodError) { - setError(err.errors[0].message); - } else { - setError( - err instanceof Error ? err.message : "An error occurred", - ); - } - } finally { - setIsLoading(false); - } - } - - return ( -
-
-
-

Members

- -
-
- {isDialogOpen && ( -
-

Add member

- - {error && ( -

{error}

- )} -
- - -
-
- )} - - - - {users - .sort((a, b) => - a.name - .toLowerCase() - .localeCompare(b.name.toLowerCase()), - ) - .map((user, index) => ( - - ))} - -
-
-
-
- ); -} diff --git a/webapp/src/app/(dashboard)/[organizationId]/members/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/members/page.tsx deleted file mode 100644 index d55f5e3c6..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/members/page.tsx +++ /dev/null @@ -1,43 +0,0 @@ -import { User } from "@/types/user"; -import MembersList from "./members-list"; -import BreadcrumbHeader from "@/components/breadcrumb"; -import { Organization } from "@/types/organization"; -import { fetchApi } from "@/utils/api"; - -export default async function MembersPage({ - params, -}: { - params: Promise<{ organizationId: string }>; -}) { - const { organizationId } = await params; - - const users: User[] | null = await fetchApi( - `/organizations/${organizationId}/users`, - { - cache: "no-store", - }, - ); - - const organization = await fetchApi( - `/organizations/${organizationId}`, - ); - - if (!users) { - return
Error loading users
; - } - - return ( - <> - - - - ); -} diff --git a/webapp/src/app/(dashboard)/[organizationId]/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/page.tsx deleted file mode 100644 index 1de1f3318..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/page.tsx +++ /dev/null @@ -1,187 +0,0 @@ -"use client"; - -import Image from "next/image"; -import dynamic from "next/dynamic"; -import { use, useEffect, useState } from "react"; - -import ErrorMessage from "@/components/error-message"; -import Loader from "@/components/loader"; -import { Card, CardContent } from "@/components/ui/card"; -import { Skeleton } from "@/components/ui/skeleton"; - -// Lazy-load chart to keep recharts off the critical path -const RadialChart = dynamic(() => import("@/components/radial-chart"), { - loading: () => ( - - - - - - ), - ssr: false, -}); -import { - getEquivalentCarKm, - getEquivalentCitizenPercentage, - getEquivalentTvTime, -} from "@/helpers/constants"; -import { - REFRESH_INTERVAL_ONE_MINUTE, - THIRTY_DAYS_MS, - SECONDS_PER_DAY, -} from "@/helpers/time-constants"; -import { fetcher } from "@/helpers/swr"; -import { getOrganizationEmissionsByProject } from "@/server-functions/organizations"; -import { Organization } from "@/types/organization"; -import { OrganizationReport } from "@/types/organization-report"; -import { DateRange } from "react-day-picker"; -import useSWR from "swr"; - -export default function OrganizationPage({ - params, -}: { - params: Promise<{ organizationId: string }>; -}) { - const { organizationId } = use(params); - const { - data: organization, - isLoading, - error, - } = useSWR(`/organizations/${organizationId}`, fetcher, { - refreshInterval: REFRESH_INTERVAL_ONE_MINUTE, - }); - - const today = new Date(); - const [date, setDate] = useState({ - from: new Date(today.getTime() - THIRTY_DAYS_MS), - to: today, - }); - const [organizationReport, setOrganizationReport] = useState< - OrganizationReport | undefined - >({ name: "", duration: 0, emissions: 0, energy_consumed: 0 }); - - useEffect(() => { - async function fetchOrganizationReport() { - try { - const organizationReport = - await getOrganizationEmissionsByProject( - organizationId, - date, - ); - if (organizationReport) { - setOrganizationReport(organizationReport); - } - } catch (error) { - console.error("Failed to fetch organization report:", error); - // Keep the default empty report with zeros - } - } - - fetchOrganizationReport(); - }, [organizationId, date]); - - if (isLoading) { - return ; - } - - if (error) { - return ; - } - - const RadialChartData = { - energy: { - label: "kWh", - value: organizationReport?.energy_consumed - ? parseFloat(organizationReport.energy_consumed.toFixed(2)) - : 0, - }, - emissions: { - label: "kg eq CO2", - value: organizationReport?.emissions - ? parseFloat(organizationReport.emissions.toFixed(2)) - : 0, - }, - duration: { - label: "days", - value: organizationReport?.duration - ? parseFloat( - (organizationReport.duration / SECONDS_PER_DAY).toFixed( - 2, - ), - ) - : 0, - }, - }; - - const citizen_converted_value = getEquivalentCitizenPercentage( - RadialChartData.emissions.value, - ).toFixed(2); - const transportation_converted_value = getEquivalentCarKm( - RadialChartData.emissions.value, - ).toFixed(2); - const tv_time_converted_value = getEquivalentTvTime( - RadialChartData.energy.value, - ).toFixed(2); - - return ( -
- {!organization && } - {organization && ( -
-
-
- Household consumption icon -

- {citizen_converted_value} % -

-

- Of a U.S citizen weekly energy emissions -

-
-
- Transportation icon -

- {transportation_converted_value} -

-

- Kilometers ridden -

-
-
- TV icon -

- {tv_time_converted_value} days -

-

- Of watching TV -

-
-
-
- - - -
-
- )} -
- ); -} diff --git a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/page.tsx deleted file mode 100644 index 73303e1dc..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/page.tsx +++ /dev/null @@ -1,250 +0,0 @@ -"use client"; - -import BreadcrumbHeader from "@/components/breadcrumb"; -import ProjectDashboard from "@/components/project-dashboard"; -import { - getEquivalentCarKm, - getEquivalentCitizenPercentage, - getEquivalentTvTime, -} from "@/helpers/constants"; -import { getDefaultDateRange } from "@/helpers/date-utils"; -import { - getExperiments, - getProjectEmissionsByExperiment, -} from "@/server-functions/experiments"; -import { getOneProject } from "@/server-functions/projects"; -import { Experiment } from "@/types/experiment"; -import { ExperimentReport } from "@/types/experiment-report"; -import { Project } from "@/types/project"; -import { use, useCallback, useEffect, useState } from "react"; -import { DateRange } from "react-day-picker"; - -export default function ProjectPage({ - params, -}: Readonly<{ - params: Promise<{ - projectId: string; - organizationId: string; - }>; -}>) { - const { projectId, organizationId } = use(params); - const [isLoading, setIsLoading] = useState(true); - - const [project, setProject] = useState({ - name: "", - description: "", - } as Project); - - // This function now just refreshes the project data instead of navigating - const handleSettingsClick = async () => { - try { - const updatedProject = await getOneProject(projectId); - if (updatedProject) { - setProject(updatedProject); - } - } catch (error) { - console.error("Error refreshing project data:", error); - } - }; - - const default_date = getDefaultDateRange(); - const [date, setDate] = useState(default_date); - - const [radialChartData, setRadialChartData] = useState({ - energy: { label: "kWh", value: 0 }, - emissions: { label: "kg eq CO2", value: 0 }, - duration: { label: "days", value: 0 }, - }); - // The experiments of the current project. We need this because experimentReport only contains the experiments that have been run - const [projectExperiments, setProjectExperiments] = useState( - [], - ); - // The reports (if any) of the experiments - const [experimentsReportData, setExperimentsReportData] = useState< - ExperimentReport[] - >([]); - - const [runData, setRunData] = useState({ - experimentId: "", - startDate: default_date.from.toISOString(), - endDate: default_date.to.toISOString(), - }); - - const [convertedValues, setConvertedValues] = useState({ - citizen: "0", - transportation: "0", - tvTime: "0", - }); - - const [selectedExperimentId, setSelectedExperimentId] = - useState(""); - const [selectedRunId, setSelectedRunId] = useState(""); - - const refreshExperimentList = useCallback(async () => { - // Logic to refresh experiments if needed - const experiments: Experiment[] = await getExperiments(projectId); - setProjectExperiments(experiments); - }, [projectId]); - - /** Use effect functions */ - useEffect(() => { - const fetchProjectDetails = async () => { - try { - const project: Project | null = await getOneProject(projectId); - if (!project) { - return; - } - setProject(project); - } catch (error) { - console.error("Error fetching project description:", error); - } - }; - - fetchProjectDetails(); - refreshExperimentList(); - }, [projectId, refreshExperimentList]); - // Fetch the experiment report of the current project - useEffect(() => { - async function fetchData() { - setIsLoading(true); - try { - const report = await getProjectEmissionsByExperiment( - projectId, - date, - ); - - const newRadialChartData = { - energy: { - label: "kWh", - value: parseFloat( - report - .reduce( - (n, { energy_consumed }) => - n + energy_consumed, - 0, - ) - .toFixed(2), - ), - }, - emissions: { - label: "kg eq CO2", - value: parseFloat( - report - .reduce((n, { emissions }) => n + emissions, 0) - .toFixed(2), - ), - }, - duration: { - label: "days", - value: parseFloat( - report - .reduce( - (n, { duration }) => n + duration / 86400, - 0, - ) - .toFixed(2), - ), - }, - }; - setRadialChartData(newRadialChartData); - - setExperimentsReportData(report); - - setRunData({ - experimentId: report[0]?.experiment_id ?? "", - startDate: date?.from?.toISOString() ?? "", - endDate: date?.to?.toISOString() ?? "", - }); - - setSelectedExperimentId(report[0]?.experiment_id ?? ""); - - setConvertedValues({ - citizen: getEquivalentCitizenPercentage( - newRadialChartData.emissions.value, - ).toFixed(2), - transportation: getEquivalentCarKm( - newRadialChartData.emissions.value, - ).toFixed(2), - tvTime: getEquivalentTvTime( - newRadialChartData.energy.value, - ).toFixed(2), - }); - } catch (error) { - console.error("Error fetching project data:", error); - } finally { - setIsLoading(false); - } - } - - if (projectId) { - fetchData(); - } - }, [projectId, date]); - - const handleExperimentClick = useCallback( - (experimentId: string) => { - if (experimentId === selectedExperimentId) { - setSelectedExperimentId(""); - setSelectedRunId(""); - return; - } - setSelectedExperimentId(experimentId); - setSelectedRunId(""); - }, - [selectedExperimentId], - ); - - const handleRunClick = useCallback( - (runId: string) => { - if (runId === selectedRunId) { - setSelectedRunId(""); - return; - } - setSelectedRunId(runId); - }, - [selectedRunId], - ); - - return ( -
- -
- - setDate(newDates || getDefaultDateRange()) - } - radialChartData={radialChartData} - convertedValues={convertedValues} - experimentsReportData={experimentsReportData} - runData={runData} - selectedExperimentId={selectedExperimentId} - selectedRunId={selectedRunId} - projectExperiments={projectExperiments} - onExperimentClick={handleExperimentClick} - onRunClick={handleRunClick} - onSettingsClick={handleSettingsClick} - isLoading={isLoading} - /> -
-
- ); -} diff --git a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx deleted file mode 100644 index 4df47eec5..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/projects/[projectId]/settings/page.tsx +++ /dev/null @@ -1,121 +0,0 @@ -import { Button } from "@/components/ui/button"; -import { Input } from "@/components/ui/input"; -import { Label } from "@/components/ui/label"; -import { Switch } from "@/components/ui/switch"; -import { getOneProject, updateProject } from "@/server-functions/projects"; -import { Project } from "@/types/project"; -import { revalidatePath } from "next/cache"; -import { ProjectTokensTable } from "../../../../../../components/projectTokens/projectTokenTable"; -import ShareProjectButton from "@/components/share-project-button"; - -// Server Action for updating project -async function updateProjectAction(projectId: string, formData: FormData) { - "use server"; - - const name = formData.get("name") as string; - const description = formData.get("description") as string; - const isPublic = formData.has("isPublic"); - - await updateProject(projectId, { - name, - description, - public: isPublic, - }); - - revalidatePath(`/projects/${projectId}/settings`); -} - -export default async function ProjectSettingsPage({ - params, -}: { - params: Promise<{ projectId: string }>; -}) { - const { projectId } = await params; - const project: Project | null = await getOneProject(projectId); - - if (!project) { - return
Project not found
; - } - - return ( -
-
-
-
-

Settings

-

Project Settings

-
-
- -
-
-
-
-
-

General

-
-
-
-
-
- - -
-
- - -
-
- - -

- (enables public sharing link) -

-
-
-
- -
-
-
-
-

API tokens

-
- -
-
-
-
- ); -} diff --git a/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx b/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx deleted file mode 100644 index eb3ae2a42..000000000 --- a/webapp/src/app/(dashboard)/[organizationId]/projects/page.tsx +++ /dev/null @@ -1,156 +0,0 @@ -"use client"; - -import BreadcrumbHeader from "@/components/breadcrumb"; -import CreateProjectModal from "@/components/createProjectModal"; -import CustomRow from "@/components/custom-row"; -import DeleteProjectModal from "@/components/delete-project-modal"; -import ErrorMessage from "@/components/error-message"; -import Loader from "@/components/loader"; -import { Button } from "@/components/ui/button"; -import { Card } from "@/components/ui/card"; -import { Table, TableBody } from "@/components/ui/table"; -import { fetcher } from "@/helpers/swr"; -import { REFRESH_INTERVAL_ONE_MINUTE } from "@/helpers/time-constants"; -import { useModal } from "@/hooks/useModal"; -import { getProjects, deleteProject } from "@/server-functions/projects"; -import { Project } from "@/types/project"; -import { use, useEffect, useState } from "react"; -import useSWR from "swr"; -import { toast } from "sonner"; - -export default function ProjectsPage({ - params, -}: { - params: Promise<{ organizationId: string }>; -}) { - const { organizationId } = use(params); - const createModal = useModal(); - const deleteModal = useModal(); - const [projectList, setProjectList] = useState([]); - const [projectToDelete, setProjectToDelete] = useState( - null, - ); - - const handleClick = async () => { - createModal.open(); - }; - - const refreshProjectList = async () => { - // Fetch the updated list of projects from the server - const projectList = await getProjects(organizationId); - setProjectList(projectList || []); - }; - - const handleDeleteClick = (project: Project) => { - setProjectToDelete(project); - deleteModal.open(); - }; - - const handleDeleteConfirm = async (projectId: string) => { - try { - await deleteProject(projectId); - toast.success("Project deleted successfully"); - await refreshProjectList(); - } catch (error) { - console.error("Error deleting project:", error); - toast.error("Failed to delete project"); - } - }; - - // Fetch the updated list of projects from the server - const { - data: projects, - error, - isLoading, - } = useSWR(`/projects?organization=${organizationId}`, fetcher, { - refreshInterval: REFRESH_INTERVAL_ONE_MINUTE, - }); - - useEffect(() => { - if (projects) { - setProjectList(projects); - } - }, [projects]); - if (isLoading) { - return ; - } - - if (error) { - return ; - } - - return ( -
- -
-
-

Projects

- - -
- - - - {projectList && - projectList - .sort((a, b) => - a.name - .toLowerCase() - .localeCompare( - b.name.toLowerCase(), - ), - ) - .map((project) => ( - - handleDeleteClick(project) - } - /> - ))} - -
-
- {projectToDelete && ( - - )} -
-
- ); -} diff --git a/webapp/src/app/(dashboard)/home/page.tsx b/webapp/src/app/(dashboard)/home/page.tsx deleted file mode 100644 index 80723a898..000000000 --- a/webapp/src/app/(dashboard)/home/page.tsx +++ /dev/null @@ -1,91 +0,0 @@ -"use client"; - -import { - Card, - CardDescription, - CardHeader, - CardTitle, -} from "@/components/ui/card"; -import Loader from "@/components/loader"; -import { Organization } from "@/types/organization"; -import { fetcher } from "@/helpers/swr"; -import { useRouter } from "next/navigation"; -import { useEffect, useState } from "react"; -import useSWR from "swr"; - -export default function HomePage() { - const router = useRouter(); - const [redirecting, setRedirecting] = useState(true); - - // Fetch organizations to find the default - const { data: organizations, error } = useSWR( - "/organizations", - fetcher, - { - revalidateOnFocus: false, - }, - ); - - useEffect(() => { - // Check if we have organizations data - if (organizations && organizations.length > 0) { - // Find default org ID - using the first organization - const defaultOrgId = organizations[0].id; - - // Save to localStorage - try { - localStorage.setItem("organizationId", defaultOrgId); - localStorage.setItem( - "organizationName", - organizations[0].name || "", - ); - } catch (error) { - console.error("Error writing to localStorage:", error); - } - - // Navigate to the organization page without a page reload - router.push(`/${defaultOrgId}`); - } else if ((organizations && organizations.length === 0) || error) { - setRedirecting(false); - } - }, [organizations, router, error]); - - // Show a loader while we fetch the data and redirect - if (redirecting) { - return ; - } - - // Fallback content if there are no organizations - return ( -
- - - Get Started - - You can do that by installing the command line tool and - running: - - codecarbon login
- codecarbon config
- codecarbon monitor -
- You'll then need to get the project id from the - config file before generating the token. -
- You can then write the token in the config file and - start monitoring.
-
- For more information, please refer to the documentation: -
- - https://mlco2.github.io/codecarbon/usage.html - -
-
-
-
- ); -} diff --git a/webapp/src/app/(dashboard)/layout.tsx b/webapp/src/app/(dashboard)/layout.tsx deleted file mode 100644 index 5b6d00996..000000000 --- a/webapp/src/app/(dashboard)/layout.tsx +++ /dev/null @@ -1,85 +0,0 @@ -"use client"; - -import NavBar from "@/components/navbar"; -import dynamic from "next/dynamic"; -import Image from "next/image"; -import Link from "next/link"; -import { Organization } from "@/types/organization"; -import { fetcher } from "@/helpers/swr"; -import { useEffect, useState } from "react"; -import useSWR from "swr"; -import Loader from "@/components/loader"; - -const MobileHeader = dynamic(() => import("@/components/mobile-header"), { - ssr: false, -}); - -export default function MainLayout({ - children, -}: Readonly<{ - children: React.ReactNode; -}>) { - const [initialLoad, setInitialLoad] = useState(true); - - // Fetch organizations for the navbar - const { data: orgs, error } = useSWR( - "/organizations", - fetcher, - { - revalidateOnFocus: false, - }, - ); - - useEffect(() => { - // Set initial load to false after first data fetch - if (orgs || error) { - // Wait a small delay to ensure smooth transition - const timer = setTimeout(() => { - setInitialLoad(false); - }, 100); - - return () => clearTimeout(timer); - } - }, [orgs, error]); - - return ( -
- {/* Side bar that shows only on screens larger than 768px */} -
-
-
- - Logo - -
- -
-
- - {/* Main content */} -
-
- -
- {initialLoad ? ( -
- -
- ) : ( - children - )} -
-
-
-
- ); -} diff --git a/webapp/src/app/(dashboard)/loading.tsx b/webapp/src/app/(dashboard)/loading.tsx deleted file mode 100644 index 4087e424c..000000000 --- a/webapp/src/app/(dashboard)/loading.tsx +++ /dev/null @@ -1 +0,0 @@ -export { default } from "@/components/loader"; // Adjust the relative path accordingly diff --git a/webapp/src/app/(dashboard)/profile/page.tsx b/webapp/src/app/(dashboard)/profile/page.tsx deleted file mode 100644 index df0fca79e..000000000 --- a/webapp/src/app/(dashboard)/profile/page.tsx +++ /dev/null @@ -1,114 +0,0 @@ -import ErrorMessage from "@/components/error-message"; -import { Button } from "@/components/ui/button"; -import { Input } from "@/components/ui/input"; -import { Label } from "@/components/ui/label"; -import { User } from "@/types/user"; -import { fetchApiServer } from "@/helpers/api-server"; - -async function getUser(): Promise { - // TODO: implement without fief - const userId = null; - if (!userId) { - return null; - } - - return await fetchApiServer(`/users/${userId}`); -} - -export default async function ProfilePage() { - const user = await getUser(); - - if (!user) { - return ; - } - - return ( -
-
-
-
-
-

{user?.name}

-

- {user?.email} -

-
-
-
-
-
-

- Personal Information -

-
-
- - -
-
- - -
-
-
-
-

- Change Password -

-
-
- - -
-
- - -
-
- - -
-
-
-
-
- -
-
-
- ); -} diff --git a/webapp/src/app/api/current-user/route.ts b/webapp/src/app/api/current-user/route.ts deleted file mode 100644 index 242126d8e..000000000 --- a/webapp/src/app/api/current-user/route.ts +++ /dev/null @@ -1,8 +0,0 @@ -// import { NextResponse } from "next/server"; -// import { fiefAuth } from "../../../helpers/fief"; - -// @TODO Implement a proper Fief implementation in Next 14 -export async function GET() { - // const currentUser = await fiefAuth.currentUser(); - return Response.json({}); -} diff --git a/webapp/src/app/favicon.ico b/webapp/src/app/favicon.ico deleted file mode 100644 index 2629fdc0da69fd8a2672892a2a7e27b0a03a0112..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15406 zcmeHNd32S<6@P$KTM-v3+EeX`C>Cw&!KxsufrKO^kd-7vvJnW8B!mDVED2#PvPCrP zTR=e|fP^&=LH4l7CRzoR9R#u1*7n%-k6-W1yqV1VmiNB*qU|3w=e#rX-MMq`?=Can zWbWs3HF7m^-FKgh+91~}ceq^bTrO8oka%u$zsprgw6<;C<4?L=haYsg+7J&-p&5cs zjpbi*mIT;1|4XKL7DCA-8^>drZ!!md#c3 zHM*E%9%EQN`Rkv#EQLweZNRQ0p|G4{+ zKRyg`l{pwVJ_31Zk74YPc1Ve6DtyRdtJxrJ&WyVk%jOP2|B)Th{@=$?u)Y*477WL_ zwKiYk=bU6@r3YbpSqz3|w8o6_v7ATdoQ$*H zCPn=S5ew51QIW`Xi7_n?IcBL}&_%2W3mYT*BWJ}RHt%UIhKw%qK=80Leu%TWvk39**cG(utA@v^$lyz zm%Ucwp0;bUjd9_&*o-SXE$lLmv1cRAsLRJQ$Lv6cvdzt_p55zOe-8?oHum|z!~JT3 z#~P>5Q8~8&@7Ha?>?tYOuxbj5a-PBTalJ94EEZKO$}lFs9ab(LMdyeIaO}`V?5%wZ zQ;Ne7UECgB&aOn4k1COn{wS*6oJjAX#;B>TKt7!@7?yfF-SNCw$_`gkpF-MoT1(=#wI@nQV_gT2_it{iXDIVAXK1rlSL zp?6s@dTbbh!v{BE`TSwH`1u)($o{oDGng=FK1}c3jBMYGvf;sWR*K+rjvd;B1?2;I z|J1P^U}u<*PVeRVKl$(w4(+YN{8?G({_bQ%&5TB9buoIrk%*JTyLH2CIv-8+&~KIN zI;|k{Ni3U_kNnhDTxVYDV{|sVgZeH^C<>-?U;^?|gU~PPF3g&kK(UENO4j4(dTJ4R zRwSX!mKu!H?u!-C{ApC?67AM zHN6+tom<+2a;7ha5BL=yn=qmqUjJPRW)eoO*Ocor2iz^mv(8H~o|Vu7%-`(H_}b(o z#0-B9?f-QY?dUApefJbBt;ofi<>PR0_uD8d?22P`TTwnK5mQT|ENvI(yn5ZjH}Yry!|Twf1!5kVo&`% z-=%KZYP$Hi`B|+SI!DL7@=LsQav_p>-DO_WhWYwATG^%LoT90DW=t|KLpKzm+n!g^ z@#Z!p^br%R<9@!fCJ?VY=wnRux8TrzMu+>A5knyP5c9N}dxb>)K zU%0e!HAi73xjmbr>*>XaSdxk0BNa%Bx|3s({$uY9HIK{k#P2yKY!&AWPd+J6-xnk; zF}4}s=M&@Z5&MRQPxA2Aj;`70!S9f++aUTj*p2cG?-^Z=Hyn!3!X7y$Tb_p&eBR`< zF?rVW#pit7!Y4Mk^mB%l{CeBu+OUvq^tH#c1-}gzuHSM@m!bD1CU-7d>b2)*`b@sJ z)UT~+Hb?2Pk!>kMuZP9p@E`8j(ug9_;qmkDvoAWYE?lr=mm0vhkHb?$E!{ByF(&8P-3$D`(Ahl+UD3KS>F4X7q~3>~F|gXFZFxQN$y=NZIy~n2Hzs5443m{? z=Qx!vpKHvZN9Wi3c6K}Av74*Y{gq?jSN7^M^}Zdyf$k(W-gP;TkrQ41@(65p80_eV z98iAqm)|VMyRQuV6brxj6;2_iA#*ML>@zo<`zK&*ItIJG@`H0c%3fV2$x%2Bbj7c9 z>9Gqyuce#@%Jd6!19d6=_!Ul~tG?abhR_Xq zo#$RdVxh-H^o5POOzQiEO_!s~_Y!wIzxkzC$86T?-Jd$Ym%d+kyvVV$m*_ro?7x6P z_{zCIePZ@vufK8euFDQ{V6o7#nf2P~(&>7iF!&oghx|Exe~w9XSI+KU*0y5F$Ljip^o;?l`xj(9!SC4nMc#?hf)2<|eh8_(euA=LUFg>tB}9{V#Yr)D(;Z_E zey7atT9duiTs;jZk8Vdc*_fBs8h^NTh3?yT;`6h| z`F(s}sugsvorsP%wxaW=tMTHO+v%RVH%8~afR8_{BfO32zBCRuzc^2R3-QFS`<(jb zxYy?r^8vf#U%Q;{1?gV2Amd47#5CtvuU|PwcuwG(Ki|ftwbQ|TzH8eeeEP}z_|w<7 z$hMyRett+s8>(qsLNWPa(d&b8;jP4K|VT1 z-;lH>`*+}cj>!pmQh?z^L%3T z_iC2l*uiS7SvHn%p2EbU?qK-8{q`H|+qDYUuAIe=P4f^vrX#wXU&HG$Y+l5IRCKz$ z0b@!dasAqP?Af^jZ!aCo>pBuY*RSP;lV?^ZS}N%{b97pw}CY;7{iq&&gFDv1(f0b$j4F6yM@-?=sn~f_MKg7X3Ytbj^L5?@LZZ^V~ zXQSg6o6%!!AxfyG^2ggZaCrX)%KH|$bp8zaU(|`5`rK$&3f=3@o6#R9j%?$1_t{j} zl02*imqzPTJyYF-(gx}WsWWE83v|!D1iedN61dzsDaY+v zU9aG=-*fC__@d|b3&OT{-k^27m}+sfX54#4OmPQZ|6_H>&KIf>u_zr2EAp_tdLGqG z$7A?FQGd*%njWjm)DXR#@F4YlZ zMs+~@e;ih|tWH-qqEE_0I9|62r)W>zwQULQuh(d;htT_>BW~TejP)xg^B>!%efi8B zm-_uv&J{d*y!5`{pG5W8FK?U&d)Kjf?7hX>Zu_Py?5q~o;dU)zirR7fm2(S;ht;6x zplm$7m6RN8;W&^l=06TieLW38U`7&z#U%xR0;UtPt>oM*Yt{Pf3p{)Ft9 z;EDgM9f&P>M&W1Q?y)_K*}(k&#q|rc=k%gII}~jHmUx`_S@H(jZmybz^Pir?nUlL| z&pk-{Nh_*($DsYU2k_!Id(h$QcM-a=nBG}Uv47W_#Cw9?cSm`>JUj6rIxoCHd*%mJ z(?7z0hxb=2{0fs@K3n?OoF_t#HN7ACvXp$t-Z?obk5K*kx9;_t!OuF;qh3T(Wpaz7LRR6(@R~@R)I#bByFTiIKFpU z?A4dNuA%AhVoPHDz36;y6(V1YN8I425V~m;)q(So9NE+%eoYUHy%${qbHnXb@giMu zgP#W9+o_Jt$P>4cspqdxTRdg>qfhcCMc++1vyQ)8;|H`N|BpfV zib1?CE_uvtU_KN!1~yk2$@dsIN`CAZ_a<+KYIWBv(Y&=!eGdy@78v+jh6Ba`R^USIgXYxEpyd?$BACu zKTi0a=p`QKm<>6jCXMS4do6u`1-;&?Ce*(kf Uul(OV diff --git a/webapp/src/app/layout.tsx b/webapp/src/app/layout.tsx deleted file mode 100644 index 217fb44de..000000000 --- a/webapp/src/app/layout.tsx +++ /dev/null @@ -1,33 +0,0 @@ -"use client"; -import { IBM_Plex_Mono } from "next/font/google"; -import "./globals.css"; -import { SWRProvider } from "../helpers/swr"; -import { ThemeProvider } from "next-themes"; -import { Toaster } from "@/components/ui/sonner"; - -const font = IBM_Plex_Mono({ weight: "400", subsets: ["latin"] }); - -export default function RootLayout({ - children, -}: Readonly<{ - children: React.ReactNode; -}>) { - return ( - - {/* suppressHydrationWarning is a Next theme recommendation */} - - - - {children} - - - - - - ); -} diff --git a/webapp/src/app/loading.tsx b/webapp/src/app/loading.tsx deleted file mode 100644 index 4087e424c..000000000 --- a/webapp/src/app/loading.tsx +++ /dev/null @@ -1 +0,0 @@ -export { default } from "@/components/loader"; // Adjust the relative path accordingly diff --git a/webapp/src/app/page.tsx b/webapp/src/app/page.tsx deleted file mode 100644 index 947dc38ba..000000000 --- a/webapp/src/app/page.tsx +++ /dev/null @@ -1,51 +0,0 @@ -import { Button } from "@/components/ui/button"; -import { LogIn } from "lucide-react"; - -export default async function Home() { - return ( - <> -
-
-

- Welcome to Code Carbon! -

-

- Get started by signing in with your preferred service. -

- -
-
- - - ); -} diff --git a/webapp/src/app/public/projects/[projectId]/page.tsx b/webapp/src/app/public/projects/[projectId]/page.tsx deleted file mode 100644 index 93a6f42f5..000000000 --- a/webapp/src/app/public/projects/[projectId]/page.tsx +++ /dev/null @@ -1,278 +0,0 @@ -"use client"; - -import { useState, useEffect, useCallback, use } from "react"; -import { useRouter } from "next/navigation"; -import { DateRange } from "react-day-picker"; -import { decryptProjectId } from "@/utils/crypto"; -import { ExperimentReport } from "@/types/experiment-report"; -import PublicProjectDashboard from "@/components/public-project-dashboard"; -import { - getEquivalentCarKm, - getEquivalentCitizenPercentage, - getEquivalentTvTime, -} from "@/helpers/constants"; -import { fetchApi } from "@/utils/api"; -import { Project } from "@/types/project"; -import ErrorMessage from "@/components/error-message"; -import Loader from "@/components/loader"; -import { Alert, AlertDescription, AlertTitle } from "@/components/ui/alert"; -import { AlertCircle } from "lucide-react"; -import { getDefaultDateRange } from "@/helpers/date-utils"; -import { Experiment } from "@/types/experiment"; -import { getExperiments } from "@/server-functions/experiments"; - -export default function PublicProjectPage({ - params, -}: { - params: Promise<{ projectId: string }>; -}) { - const { projectId: encryptedId } = use(params); - const router = useRouter(); - - const [isLoading, setIsLoading] = useState(true); - const [error, setError] = useState(null); - const [projectId, setProjectId] = useState(null); - const [project, setProject] = useState(null); - - // Dashboard state - const default_date = getDefaultDateRange(); - const [date, setDate] = useState(default_date); - // The experiments of the current project. We need this because experimentReport only contains the experiments that have been run - const [projectExperiments, setProjectExperiments] = useState( - [], - ); - // The reports (if any) of the experiments - const [experimentsReportData, setExperimentsReportData] = useState< - ExperimentReport[] - >([]); - - const [radialChartData, setRadialChartData] = useState({ - energy: { label: "kWh", value: 0 }, - emissions: { label: "kg eq CO2", value: 0 }, - duration: { label: "days", value: 0 }, - }); - - const [runData, setRunData] = useState({ - experimentId: "", - startDate: default_date.from.toISOString(), - endDate: default_date.to.toISOString(), - }); - const [convertedValues, setConvertedValues] = useState({ - citizen: "0", - transportation: "0", - tvTime: "0", - }); - const [selectedExperimentId, setSelectedExperimentId] = - useState(""); - const [selectedRunId, setSelectedRunId] = useState(""); - - const refreshExperimentList = useCallback(async () => { - if (!projectId) return; - // Logic to refresh experiments if needed - const experiments: Experiment[] = await getExperiments(projectId); - setProjectExperiments(experiments); - }, [projectId]); - - // Decrypt the project ID - useEffect(() => { - const decrypt = async () => { - try { - setIsLoading(true); - const decryptedId = await decryptProjectId(encryptedId); - setProjectId(decryptedId); - } catch (error) { - console.error("Failed to decrypt project ID:", error); - setError( - "Invalid project link or the project no longer exists.", - ); - } - }; - - decrypt(); - }, [encryptedId]); - - // Fetch project data - useEffect(() => { - const fetchProjectData = async () => { - if (!projectId) return; - - try { - // Use regular endpoint - the backend already handles public projects without auth - const projectData = await fetchApi( - `/projects/${projectId}`, - ); - - if (!projectData || !projectData.public) { - setError( - "This project is not available for public viewing.", - ); - return; - } - - setProject(projectData); - } catch (error) { - console.error("Error fetching project:", error); - setError("Failed to load project data."); - } finally { - setIsLoading(false); - } - }; - - if (projectId && !project) { - fetchProjectData(); - refreshExperimentList(); - } - }, [projectId, project, refreshExperimentList]); - - // Fetch experiments and emissions data - useEffect(() => { - async function fetchData() { - if (!projectId) return; - - setIsLoading(true); - try { - const report = await fetchApi( - `/projects/${projectId}/experiments/sums?start_date=${date?.from?.toISOString()}&end_date=${date?.to?.toISOString()}`, - ); - - if (!report) { - return; - } - - setExperimentsReportData(report); - - const newRadialChartData = { - energy: { - label: "kWh", - value: parseFloat( - report - .reduce( - (n, { energy_consumed }) => - n + energy_consumed, - 0, - ) - .toFixed(2), - ), - }, - emissions: { - label: "kg eq CO2", - value: parseFloat( - report - .reduce((n, { emissions }) => n + emissions, 0) - .toFixed(2), - ), - }, - duration: { - label: "days", - value: parseFloat( - report - .reduce( - (n, { duration }) => n + duration / 86400, - 0, - ) - .toFixed(2), - ), - }, - }; - - setRadialChartData(newRadialChartData); - - if (report.length > 0) { - setRunData({ - experimentId: report[0]?.experiment_id ?? "", - startDate: date?.from?.toISOString() ?? "", - endDate: date?.to?.toISOString() ?? "", - }); - - setSelectedExperimentId(report[0]?.experiment_id ?? ""); - } - - setConvertedValues({ - citizen: getEquivalentCitizenPercentage( - newRadialChartData.emissions.value, - ).toFixed(2), - transportation: getEquivalentCarKm( - newRadialChartData.emissions.value, - ).toFixed(2), - tvTime: getEquivalentTvTime( - newRadialChartData.energy.value, - ).toFixed(2), - }); - } catch (error) { - console.error("Error fetching data:", error); - } finally { - setIsLoading(false); - } - } - - if (projectId && project) { - fetchData(); - } - }, [projectId, project, date]); - - const handleExperimentClick = useCallback((experimentId: string) => { - setSelectedExperimentId(experimentId); - setRunData((prevData) => ({ - ...prevData, - experimentId: experimentId, - })); - setSelectedRunId(""); // Reset the run ID - }, []); - - const handleRunClick = useCallback((runId: string) => { - setSelectedRunId(runId); - }, []); - - // Show full page loader only during initial load - if (isLoading && !project) { - return ; - } - - if (error) { - return ( -
- - - Error - {error} - -
- -
-
- ); - } - - if (!project) { - return ; - } - - return ( -
-
- - setDate(newDates || getDefaultDateRange()) - } - radialChartData={radialChartData} - convertedValues={convertedValues} - experimentsReportData={experimentsReportData} - projectExperiments={projectExperiments} - runData={runData} - selectedExperimentId={selectedExperimentId} - selectedRunId={selectedRunId} - onExperimentClick={handleExperimentClick} - onRunClick={handleRunClick} - isLoading={isLoading} - /> -
-
- ); -} diff --git a/webapp/src/components/auth-guard.tsx b/webapp/src/components/auth-guard.tsx new file mode 100644 index 000000000..3da71e8a3 --- /dev/null +++ b/webapp/src/components/auth-guard.tsx @@ -0,0 +1,13 @@ +import { ReactNode } from "react"; + +function hasSessionCookie(): boolean { + return document.cookie.split(";").some((c) => c.trim().startsWith("user_session=")); +} + +export default function AuthGuard({ children }: { children: ReactNode }) { + if (!hasSessionCookie()) { + window.location.href = `${import.meta.env.VITE_API_URL}/auth/login?redirect=${import.meta.env.VITE_BASE_URL}/home?auth=true`; + return null; + } + return <>{children}; +} diff --git a/webapp/src/components/chart-skeleton.tsx b/webapp/src/components/chart-skeleton.tsx index 67f03774d..fde0bebf8 100644 --- a/webapp/src/components/chart-skeleton.tsx +++ b/webapp/src/components/chart-skeleton.tsx @@ -19,7 +19,7 @@ export default function ChartSkeleton({ - + ); diff --git a/webapp/src/components/createExperimentModal.tsx b/webapp/src/components/createExperimentModal.tsx index 4f789bce3..9d724c829 100644 --- a/webapp/src/components/createExperimentModal.tsx +++ b/webapp/src/components/createExperimentModal.tsx @@ -1,10 +1,9 @@ -"use client"; - -import { useEffect, useState } from "react"; -import { createExperiment } from "@/server-functions/experiments"; +import { useEffect, useRef, useState } from "react"; +import { createExperiment } from "@/api/experiments"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; -import { Experiment } from "@/types/experiment"; +import { Label } from "@/components/ui/label"; +import { Experiment } from "@/api/schemas"; import { Separator } from "./ui/separator"; import { ClipboardCheck, ClipboardCopy, Loader2 } from "lucide-react"; import { toast } from "sonner"; @@ -28,6 +27,7 @@ export default function CreateExperimentModal({ onExperimentCreated: () => void; }) { const [isCopied, setIsCopied] = useState(false); + const copyTimerRef = useRef | null>(null); const [isSaving, setIsSaving] = useState(false); const [isCreated, setIsCreated] = useState(false); const [experimentData, setExperimentData] = useState({ @@ -48,6 +48,10 @@ export default function CreateExperimentModal({ } }, [projectId, experimentData]); + useEffect(() => { + return () => { if (copyTimerRef.current) clearTimeout(copyTimerRef.current); }; + }, []); + const resetForm = () => { setExperimentData({ name: "", @@ -95,7 +99,7 @@ export default function CreateExperimentModal({ .then(() => { setIsCopied(true); toast.success("Experiment ID copied to clipboard"); - setTimeout(() => setIsCopied(false), 2000); + copyTimerRef.current = setTimeout(() => setIsCopied(false), 2000); }) .catch((err) => { console.error("Failed to copy experiment id:", err); @@ -114,7 +118,9 @@ export default function CreateExperimentModal({
+ @@ -127,7 +133,9 @@ export default function CreateExperimentModal({ />
+ @@ -170,6 +178,7 @@ export default function CreateExperimentModal({ + + +

Refresh data

+
+ + @@ -164,6 +195,7 @@ export default function ProjectDashboard({ className="p-1 rounded-full" variant="outline" size="icon" + aria-label="Project settings" onClick={settingsModal.open} > @@ -209,7 +241,7 @@ export function ProjectVisibilityBadge({ isPublic }: { isPublic: boolean }) { return isPublic ? ( Public @@ -217,7 +249,7 @@ export function ProjectVisibilityBadge({ isPublic }: { isPublic: boolean }) { ) : ( Private diff --git a/webapp/src/components/project-settings-modal.tsx b/webapp/src/components/project-settings-modal.tsx index 7a8f3695d..0befd0ba9 100644 --- a/webapp/src/components/project-settings-modal.tsx +++ b/webapp/src/components/project-settings-modal.tsx @@ -1,7 +1,5 @@ -"use client"; - import { useState, useEffect } from "react"; -import { Project } from "@/types/project"; +import { Project } from "@/api/schemas"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; import { Label } from "@/components/ui/label"; @@ -16,10 +14,9 @@ import { } from "@/components/ui/dialog"; import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"; import { ProjectTokensTable } from "./projectTokens/projectTokenTable"; -import { updateProject } from "@/server-functions/projects"; +import { updateProject } from "@/api/projects"; import { toast } from "sonner"; import { Loader2 } from "lucide-react"; -import { useRouter } from "next/navigation"; interface ProjectSettingsModalProps { open: boolean; @@ -34,7 +31,6 @@ export default function ProjectSettingsModal({ project, onProjectUpdated, }: ProjectSettingsModalProps) { - const router = useRouter(); const [name, setName] = useState(project.name || ""); const [description, setDescription] = useState(project.description || ""); const [isPublic, setIsPublic] = useState(project.public || false); @@ -64,7 +60,6 @@ export default function ProjectSettingsModal({ } finally { setIsSaving(false); onOpenChange(false); - router.refresh(); } }; diff --git a/webapp/src/components/projectTokens/custom-row-token.tsx b/webapp/src/components/projectTokens/custom-row-token.tsx index 609e6a8e1..8d4be2db0 100644 --- a/webapp/src/components/projectTokens/custom-row-token.tsx +++ b/webapp/src/components/projectTokens/custom-row-token.tsx @@ -1,8 +1,6 @@ -"use client"; - -import { IProjectToken } from "@/types/project"; +import { IProjectToken } from "@/api/schemas"; import CustomRow from "../custom-row"; -import { deleteProjectToken } from "@/server-functions/projectTokens"; +import { deleteProjectToken } from "@/api/projectTokens"; import { useState } from "react"; import { toast } from "sonner"; diff --git a/webapp/src/components/projectTokens/projectTokenTable.tsx b/webapp/src/components/projectTokens/projectTokenTable.tsx index 2f4503e49..52a6d4f0b 100644 --- a/webapp/src/components/projectTokens/projectTokenTable.tsx +++ b/webapp/src/components/projectTokens/projectTokenTable.tsx @@ -1,15 +1,12 @@ -"use client"; - import { Card } from "@/components/ui/card"; import { Table, TableBody } from "@/components/ui/table"; -import { IProjectToken } from "@/types/project"; +import { IProjectToken } from "@/api/schemas"; import { getProjectTokens, createProjectToken, -} from "@/server-functions/projectTokens"; +} from "@/api/projectTokens"; import CustomRowToken from "@/components/projectTokens/custom-row-token"; -import { useState, useEffect } from "react"; -import { useRouter } from "next/navigation"; +import { useState, useEffect, useRef } from "react"; import { Loader2, ClipboardCopy, ClipboardCheck } from "lucide-react"; import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; @@ -23,8 +20,10 @@ export const ProjectTokensTable = ({ projectId }: { projectId: string }) => { const [tokenName, setTokenName] = useState(""); const [createdToken, setCreatedToken] = useState(null); const [isCopied, setIsCopied] = useState(false); - const router = useRouter(); - + const copyTimerRef = useRef | null>(null); + useEffect(() => { + return () => { if (copyTimerRef.current) clearTimeout(copyTimerRef.current); }; + }, []); useEffect(() => { const fetchTokens = async () => { // Fetch the updated list of tokens from the server @@ -38,7 +37,6 @@ export const ProjectTokensTable = ({ projectId }: { projectId: string }) => { const refreshTokens = () => { setTokens(null); // This will trigger a refetch in the useEffect - router.refresh(); // Refresh the current route }; const handleCreateToken = async () => { @@ -74,7 +72,7 @@ export const ProjectTokensTable = ({ projectId }: { projectId: string }) => { if (success) { setIsCopied(true); toast.success("Token copied to clipboard"); - setTimeout(() => setIsCopied(false), 2000); // Revert back after 2 seconds + copyTimerRef.current = setTimeout(() => setIsCopied(false), 2000); } else { throw new Error("Copy operation failed"); } @@ -113,7 +111,8 @@ export const ProjectTokensTable = ({ projectId }: { projectId: string }) => { - - - setTheme("light")}> - Light - - setTheme("dark")}> - Dark - - setTheme("system")}> - System - - - - ); -} diff --git a/webapp/src/components/ui/sonner.tsx b/webapp/src/components/ui/sonner.tsx index f8dffa4fb..676be2f68 100644 --- a/webapp/src/components/ui/sonner.tsx +++ b/webapp/src/components/ui/sonner.tsx @@ -1,27 +1,22 @@ -"use client"; - -import { useTheme } from "next-themes"; import { Toaster as Sonner } from "sonner"; type ToasterProps = React.ComponentProps; const Toaster = ({ ...props }: ToasterProps) => { - const { theme = "system" } = useTheme(); - return ( ( - endpoint: string, - options?: RequestInit, -): Promise { - const response = await fetch(`${API_BASE}${endpoint}`, { - ...options, - credentials: "include", - headers: { - "Content-Type": "application/json", - ...(options?.headers || {}), - }, - }); - - if (!response.ok) { - let errorMessage = `API error: ${response.status} ${response.statusText}`; - try { - const errorData = await response.json(); - errorMessage = errorData.detail || errorMessage; - } catch (e) { - // Ignore JSON parsing errors - } - console.error(errorMessage); - return null; - } - - // Handle 204 No Content responses (e.g., DELETE operations) - if (response.status === 204) { - return null; - } - - return response.json(); -} diff --git a/webapp/src/helpers/api-server.ts b/webapp/src/helpers/api-server.ts deleted file mode 100644 index d40e43851..000000000 --- a/webapp/src/helpers/api-server.ts +++ /dev/null @@ -1,71 +0,0 @@ -"use server"; - -import { cookies } from "next/headers"; -import { SESSION_COOKIE_NAME } from "./auth"; - -const API_BASE = process.env.NEXT_PUBLIC_API_URL; - -/** - * Fetch API with authentication for server-side requests - * Uses cookie-based auth for server components/actions - */ -export async function fetchApiServer( - endpoint: string, - options?: RequestInit, -): Promise { - try { - // Get session cookie for server-side auth - const cookieStore = await cookies(); - const sessionCookie = cookieStore.get(SESSION_COOKIE_NAME); - - if (!sessionCookie?.value) { - throw new Error("No authentication session found"); - } - - const response = await fetch(`${API_BASE}${endpoint}`, { - ...options, - headers: { - "Content-Type": "application/json", - Cookie: `${SESSION_COOKIE_NAME}=${sessionCookie.value}`, - ...(options?.headers || {}), - }, - }); - - if (!response.ok) { - let errorMessage = `API error: ${response.status} ${response.statusText}`; - try { - const errorData = await response.json(); - errorMessage = errorData.detail || errorMessage; - } catch (e) { - // Ignore JSON parsing errors - } - console.error(errorMessage); - return null; - } - - // Handle 204 No Content responses (e.g., DELETE operations) - if (response.status === 204) { - return null; - } - - // Parse JSON response - try { - return await response.json(); - } catch (e) { - // If JSON parsing fails (e.g., empty response body), return null - console.warn( - `Empty or invalid JSON response from ${endpoint}, returning null`, - ); - return null; - } - } catch (error) { - // Log server-side error with more details - console.error("API server request failed:", { - endpoint, - error: error instanceof Error ? error.message : String(error), - }); - - // Return null to let callers handle defaults appropriately - return null; - } -} diff --git a/webapp/src/helpers/auth.ts b/webapp/src/helpers/auth.ts deleted file mode 100644 index 69bdd06c7..000000000 --- a/webapp/src/helpers/auth.ts +++ /dev/null @@ -1 +0,0 @@ -export const SESSION_COOKIE_NAME = "user_session"; diff --git a/webapp/src/helpers/dashboard-calculations.ts b/webapp/src/helpers/dashboard-calculations.ts index 8b4b67e27..2668f4823 100644 --- a/webapp/src/helpers/dashboard-calculations.ts +++ b/webapp/src/helpers/dashboard-calculations.ts @@ -1,4 +1,4 @@ -import { ExperimentReport } from "@/types/experiment-report"; +import { ExperimentReport } from "@/api/schemas"; import { getEquivalentCarKm, getEquivalentCitizenPercentage, diff --git a/webapp/src/helpers/swr.tsx b/webapp/src/helpers/swr.tsx deleted file mode 100644 index 0a16fb956..000000000 --- a/webapp/src/helpers/swr.tsx +++ /dev/null @@ -1,14 +0,0 @@ -import { SWRConfig } from "swr"; - -export const SWRProvider = ({ children }: { children: React.ReactNode }) => { - return {children}; -}; - -export const fetcher = async (url: string) => { - const res = await fetch(`${process.env.NEXT_PUBLIC_API_URL}${url}`); - if (!res.ok) { - console.error("Failed to fetch data", res.statusText); - throw new Error("Failed to fetch data"); - } - return res.json(); -}; diff --git a/webapp/src/helpers/time-constants.ts b/webapp/src/helpers/time-constants.ts index 85a8c479b..dd284ea27 100644 --- a/webapp/src/helpers/time-constants.ts +++ b/webapp/src/helpers/time-constants.ts @@ -8,11 +8,9 @@ const MINUTES_PER_HOUR = 60; const HOURS_PER_DAY = 24; // Millisecond durations -const ONE_MINUTE_MS = 1000 * SECONDS_PER_MINUTE; -const ONE_DAY_MS = ONE_MINUTE_MS * MINUTES_PER_HOUR * HOURS_PER_DAY; +const ONE_DAY_MS = 1000 * SECONDS_PER_MINUTE * MINUTES_PER_HOUR * HOURS_PER_DAY; // Exported constants used across the app export const THIRTY_DAYS_MS = 30 * ONE_DAY_MS; export const SECONDS_PER_DAY = SECONDS_PER_MINUTE * MINUTES_PER_HOUR * HOURS_PER_DAY; -export const REFRESH_INTERVAL_ONE_MINUTE = ONE_MINUTE_MS; diff --git a/webapp/src/hooks/useProjectDashboard.ts b/webapp/src/hooks/useProjectDashboard.ts index 3f242b65f..2ea2f8608 100644 --- a/webapp/src/hooks/useProjectDashboard.ts +++ b/webapp/src/hooks/useProjectDashboard.ts @@ -1,7 +1,6 @@ import { useCallback, useEffect, useState } from "react"; import { DateRange } from "react-day-picker"; -import { Experiment } from "@/types/experiment"; -import { ExperimentReport } from "@/types/experiment-report"; +import { Experiment, ExperimentReport } from "@/api/schemas"; import { calculateConvertedValues, calculateRadialChartData, @@ -10,7 +9,7 @@ import { getDefaultRadialChartData, RadialChartData, } from "@/helpers/dashboard-calculations"; -import { getExperiments } from "@/server-functions/experiments"; +import { getExperiments } from "@/api/experiments"; export type RunData = { experimentId: string; diff --git a/webapp/src/layouts/DashboardLayout.tsx b/webapp/src/layouts/DashboardLayout.tsx new file mode 100644 index 000000000..c1eee6308 --- /dev/null +++ b/webapp/src/layouts/DashboardLayout.tsx @@ -0,0 +1,74 @@ +import NavBar from "@/components/navbar"; +import { lazy, Suspense, useEffect, useState } from "react"; +import { Link, Outlet } from "react-router-dom"; +import { Organization } from "@/api/schemas"; +import { fetcher } from "@/api/swr"; +import useSWR from "swr"; +import Loader from "@/components/loader"; + +const MobileHeader = lazy(() => import("@/components/mobile-header")); + +export default function DashboardLayout() { + const [initialLoad, setInitialLoad] = useState(true); + + const { data: orgs, error } = useSWR( + "/organizations", + fetcher, + { revalidateOnFocus: false }, + ); + + useEffect(() => { + if (orgs || error) { + const timer = setTimeout(() => { + setInitialLoad(false); + }, 100); + return () => clearTimeout(timer); + } + }, [orgs, error]); + + return ( +
+ + Skip to main content + + + +
+
+ + + +
+ {initialLoad ? ( +
+ +
+ ) : ( + + )} +
+
+
+
+ ); +} diff --git a/webapp/src/main.tsx b/webapp/src/main.tsx new file mode 100644 index 000000000..2c5e43176 --- /dev/null +++ b/webapp/src/main.tsx @@ -0,0 +1,17 @@ +import { StrictMode } from "react"; +import { createRoot } from "react-dom/client"; +import { RouterProvider } from "react-router-dom"; +import { SWRConfig } from "swr"; +import { Toaster } from "@/components/ui/sonner"; +import { router } from "./router"; +import { swrConfig } from "./api/swr"; +import "./globals.css"; + +createRoot(document.getElementById("root")!).render( + + + + + + , +); diff --git a/webapp/src/middleware.ts b/webapp/src/middleware.ts deleted file mode 100644 index 4251a2463..000000000 --- a/webapp/src/middleware.ts +++ /dev/null @@ -1,14 +0,0 @@ -import type { NextRequest } from "next/server"; - -export async function middleware(request: NextRequest) { - const pathname = request.nextUrl.pathname; - - // Skip auth for public routes - if ( - pathname === "/" || - pathname.startsWith("/public/") || - pathname.startsWith("/api/public/") - ) { - return; - } -} diff --git a/webapp/src/pages/HomePage.tsx b/webapp/src/pages/HomePage.tsx new file mode 100644 index 000000000..ce3c00c33 --- /dev/null +++ b/webapp/src/pages/HomePage.tsx @@ -0,0 +1,74 @@ +import { + Card, + CardDescription, + CardHeader, + CardTitle, +} from "@/components/ui/card"; +import Loader from "@/components/loader"; +import { Organization } from "@/api/schemas"; +import { fetcher } from "@/api/swr"; +import { useNavigate } from "react-router-dom"; +import { useEffect, useState } from "react"; +import useSWR from "swr"; + +export default function HomePage() { + const navigate = useNavigate(); + const [redirecting, setRedirecting] = useState(true); + + const { data: organizations, error } = useSWR( + "/organizations", + fetcher, + { revalidateOnFocus: false }, + ); + + useEffect(() => { + if (organizations && organizations.length > 0) { + const defaultOrgId = organizations[0].id; + try { + localStorage.setItem("organizationId", defaultOrgId); + localStorage.setItem("organizationName", organizations[0].name || ""); + } catch (error) { + console.error("Error writing to localStorage:", error); + } + navigate(`/${defaultOrgId}`); + } else if ((organizations && organizations.length === 0) || error) { + setRedirecting(false); + } + }, [organizations, navigate, error]); + + if (redirecting) { + return ; + } + + return ( +
+ + + Get Started + + You can do that by installing the command line tool and running: + + codecarbon login
+ codecarbon config
+ codecarbon monitor +
+ You'll then need to get the project id from the config file + before generating the token. +
+ You can then write the token in the config file and start + monitoring.
+
+ For more information, please refer to the documentation: +
+ + https://mlco2.github.io/codecarbon/usage.html + +
+
+
+
+ ); +} diff --git a/webapp/src/pages/LandingPage.tsx b/webapp/src/pages/LandingPage.tsx new file mode 100644 index 000000000..ef19d7a9d --- /dev/null +++ b/webapp/src/pages/LandingPage.tsx @@ -0,0 +1,48 @@ +import { Button } from "@/components/ui/button"; +import { LogIn } from "lucide-react"; + +export default function LandingPage() { + return ( + <> +
+
+

+ Welcome to Code Carbon! +

+

+ Get started by signing in with your preferred service. +

+ +
+
+ + + ); +} diff --git a/webapp/src/pages/MembersPage.tsx b/webapp/src/pages/MembersPage.tsx new file mode 100644 index 000000000..78468ea1e --- /dev/null +++ b/webapp/src/pages/MembersPage.tsx @@ -0,0 +1,173 @@ +import BreadcrumbHeader from "@/components/breadcrumb"; +import CustomRow from "@/components/custom-row"; +import { Button } from "@/components/ui/button"; +import { Card } from "@/components/ui/card"; +import { Input } from "@/components/ui/input"; +import { Label } from "@/components/ui/label"; +import { Table, TableBody } from "@/components/ui/table"; +import { Organization, User } from "@/api/schemas"; +import { fetcher } from "@/api/swr"; +import { useParams } from "react-router-dom"; +import { useState } from "react"; +import { z } from "zod"; +import { toast } from "sonner"; +import useSWR, { mutate } from "swr"; +import Loader from "@/components/loader"; +import ErrorMessage from "@/components/error-message"; + +export default function MembersPage() { + const { organizationId } = useParams<{ organizationId: string }>(); + const [isDialogOpen, setDialogOpen] = useState(false); + const [error, setError] = useState(null); + const [isLoading, setIsLoading] = useState(false); + + const { data: users, error: fetchError, isLoading: usersLoading } = useSWR( + `/organizations/${organizationId}/users`, + fetcher, + ); + + const { data: organization } = useSWR( + `/organizations/${organizationId}`, + fetcher, + ); + + const form = { email: undefined as string | undefined }; + + const emailSchema = z.object({ + email: z.string().email("Please enter a valid email address"), + }); + + async function addUser() { + try { + setIsLoading(true); + const email = ( + document.getElementById("emailInput") as HTMLInputElement + ).value; + + emailSchema.parse({ email }); + setError(null); + + const body = JSON.stringify({ email }); + const API_BASE = import.meta.env.VITE_API_URL; + + await toast + .promise( + fetch(`${API_BASE}/organizations/${organizationId}/add-user`, { + method: "POST", + credentials: "include", + headers: { "Content-Type": "application/json" }, + body, + }).then(async (res) => { + if (!res.ok) throw new Error("Failed to add user"); + return res.json(); + }), + { + loading: `Adding user ${email}...`, + success: `User ${email} added successfully`, + error: (err) => `${err.message}`, + }, + ) + .unwrap(); + + mutate(`/organizations/${organizationId}/users`); + setDialogOpen(false); + } catch (err) { + if (err instanceof z.ZodError) { + setError(err.errors[0].message); + } else { + setError( + err instanceof Error ? err.message : "An error occurred", + ); + } + } finally { + setIsLoading(false); + } + } + + if (usersLoading) { + return ; + } + + if (fetchError || !users) { + return ; + } + + return ( + <> + +
+
+
+

Members

+ +
+ {isDialogOpen && ( +
+

Add member

+ + + {error && ( + + )} +
+ + +
+
+ )} + + + + {users + .sort((a, b) => + a.name.toLowerCase().localeCompare(b.name.toLowerCase()), + ) + .map((user) => ( + + ))} + +
+
+
+
+ + ); +} diff --git a/webapp/src/pages/OrgDashboardPage.tsx b/webapp/src/pages/OrgDashboardPage.tsx new file mode 100644 index 000000000..4a723d098 --- /dev/null +++ b/webapp/src/pages/OrgDashboardPage.tsx @@ -0,0 +1,172 @@ +import { lazy, Suspense, useEffect, useState } from "react"; +import { useParams } from "react-router-dom"; + +import ErrorMessage from "@/components/error-message"; +import Loader from "@/components/loader"; +import { Card, CardContent } from "@/components/ui/card"; +import { Skeleton } from "@/components/ui/skeleton"; + +const RadialChart = lazy(() => import("@/components/radial-chart")); + +import { + getEquivalentCarKm, + getEquivalentCitizenPercentage, + getEquivalentTvTime, +} from "@/helpers/constants"; +import { + THIRTY_DAYS_MS, + SECONDS_PER_DAY, +} from "@/helpers/time-constants"; +import { fetcher } from "@/api/swr"; +import { getOrganizationEmissionsByProject } from "@/api/organizations"; +import { Organization, OrganizationReport } from "@/api/schemas"; +import { DateRange } from "react-day-picker"; +import useSWR from "swr"; + +export default function OrgDashboardPage() { + const { organizationId } = useParams<{ organizationId: string }>(); + const { + data: organization, + isLoading, + error, + } = useSWR(`/organizations/${organizationId}`, fetcher); + + const today = new Date(); + const [date, setDate] = useState({ + from: new Date(today.getTime() - THIRTY_DAYS_MS), + to: today, + }); + const [organizationReport, setOrganizationReport] = useState< + OrganizationReport | undefined + >({ name: "", duration: 0, emissions: 0, energy_consumed: 0 }); + + useEffect(() => { + async function fetchOrganizationReport() { + try { + const report = await getOrganizationEmissionsByProject( + organizationId!, + date, + ); + if (report) { + setOrganizationReport(report); + } + } catch (error) { + console.error("Failed to fetch organization report:", error); + } + } + fetchOrganizationReport(); + }, [organizationId, date]); + + if (isLoading) { + return ; + } + + if (error) { + return ; + } + + const RadialChartData = { + energy: { + label: "kWh", + value: organizationReport?.energy_consumed + ? parseFloat(organizationReport.energy_consumed.toFixed(2)) + : 0, + }, + emissions: { + label: "kg eq CO2", + value: organizationReport?.emissions + ? parseFloat(organizationReport.emissions.toFixed(2)) + : 0, + }, + duration: { + label: "days", + value: organizationReport?.duration + ? parseFloat( + (organizationReport.duration / SECONDS_PER_DAY).toFixed(2), + ) + : 0, + }, + }; + + const citizen_converted_value = getEquivalentCitizenPercentage( + RadialChartData.emissions.value, + ).toFixed(2); + const transportation_converted_value = getEquivalentCarKm( + RadialChartData.emissions.value, + ).toFixed(2); + const tv_time_converted_value = getEquivalentTvTime( + RadialChartData.energy.value, + ).toFixed(2); + + const radialFallback = ( + + + + + + ); + + return ( +
+ {!organization && } + {organization && ( +
+
+
+ Household consumption icon +

+ {citizen_converted_value} % +

+

+ Of a U.S citizen weekly energy emissions +

+
+
+ Transportation icon +

+ {transportation_converted_value} +

+

Kilometers ridden

+
+
+ TV icon +

+ {tv_time_converted_value} days +

+

Of watching TV

+
+
+
+ + + + + + + + + +
+
+ )} +
+ ); +} diff --git a/webapp/src/pages/PrivacyPage.tsx b/webapp/src/pages/PrivacyPage.tsx new file mode 100644 index 000000000..c94e1c71f --- /dev/null +++ b/webapp/src/pages/PrivacyPage.tsx @@ -0,0 +1,64 @@ +export default function PrivacyPage() { + return ( +
+

Privacy Policy

+ +
+

Data Collection

+

+ CodeCarbon collects computing emissions data that you explicitly submit + through our tracking tools. This includes energy consumption metrics, + hardware information, and geographic region data used to calculate + carbon emissions. +

+
+ +
+

Usage

+

+ Your data is used solely to calculate and display carbon emission + estimates from your computing activities. We do not sell or share your + data with third parties for marketing purposes. +

+
+ +
+

Data Retention

+

+ Your emissions data is retained for as long as your account is active. + You may request deletion of your data at any time by contacting us. +

+
+ +
+

Third Parties

+

+ We use authentication providers to manage user accounts. No emissions + data is shared with these providers. +

+
+ +
+

Your Rights

+

+ You have the right to access, modify, or delete your personal data. + Contact us at the email below to exercise these rights. +

+
+ +
+

Contact

+

+ For privacy-related inquiries, please open an issue on our{" "} + + GitHub repository + + . +

+
+
+ ); +} diff --git a/webapp/src/pages/ProjectDashboardPage.tsx b/webapp/src/pages/ProjectDashboardPage.tsx new file mode 100644 index 000000000..0be90ce47 --- /dev/null +++ b/webapp/src/pages/ProjectDashboardPage.tsx @@ -0,0 +1,237 @@ +import BreadcrumbHeader from "@/components/breadcrumb"; +import ProjectDashboard from "@/components/project-dashboard"; +import { + getEquivalentCarKm, + getEquivalentCitizenPercentage, + getEquivalentTvTime, +} from "@/helpers/constants"; +import { getDefaultDateRange } from "@/helpers/date-utils"; +import { + getExperiments, + getProjectEmissionsByExperiment, +} from "@/api/experiments"; +import { getOneProject } from "@/api/projects"; +import { Experiment, ExperimentReport, Project } from "@/api/schemas"; +import { useCallback, useEffect, useState } from "react"; +import { useParams } from "react-router-dom"; +import { DateRange } from "react-day-picker"; + +export default function ProjectDashboardPage() { + const { projectId, organizationId } = useParams<{ + projectId: string; + organizationId: string; + }>(); + const [isLoading, setIsLoading] = useState(true); + + const [project, setProject] = useState({ + name: "", + description: "", + } as Project); + + const default_date = getDefaultDateRange(); + const [date, setDate] = useState(default_date); + + const [radialChartData, setRadialChartData] = useState({ + energy: { label: "kWh", value: 0 }, + emissions: { label: "kg eq CO2", value: 0 }, + duration: { label: "days", value: 0 }, + }); + const [projectExperiments, setProjectExperiments] = useState( + [], + ); + const [experimentsReportData, setExperimentsReportData] = useState< + ExperimentReport[] + >([]); + + const [runData, setRunData] = useState({ + experimentId: "", + startDate: default_date.from.toISOString(), + endDate: default_date.to.toISOString(), + }); + + const [convertedValues, setConvertedValues] = useState({ + citizen: "0", + transportation: "0", + tvTime: "0", + }); + + const [selectedExperimentId, setSelectedExperimentId] = + useState(""); + const [selectedRunId, setSelectedRunId] = useState(""); + + const refreshExperimentList = useCallback(async () => { + const experiments: Experiment[] = await getExperiments(projectId!); + setProjectExperiments(experiments); + }, [projectId]); + + const fetchEmissionsReport = useCallback( + async (dateRange: DateRange) => { + setIsLoading(true); + try { + const report = await getProjectEmissionsByExperiment( + projectId!, + dateRange, + ); + + const newRadialChartData = { + energy: { + label: "kWh", + value: parseFloat( + report + .reduce((n, { energy_consumed }) => n + energy_consumed, 0) + .toFixed(2), + ), + }, + emissions: { + label: "kg eq CO2", + value: parseFloat( + report + .reduce((n, { emissions }) => n + emissions, 0) + .toFixed(2), + ), + }, + duration: { + label: "days", + value: parseFloat( + report + .reduce((n, { duration }) => n + duration / 86400, 0) + .toFixed(2), + ), + }, + }; + setRadialChartData(newRadialChartData); + setExperimentsReportData(report); + setRunData({ + experimentId: report[0]?.experiment_id ?? "", + startDate: dateRange?.from?.toISOString() ?? "", + endDate: dateRange?.to?.toISOString() ?? "", + }); + setSelectedExperimentId(report[0]?.experiment_id ?? ""); + setConvertedValues({ + citizen: getEquivalentCitizenPercentage( + newRadialChartData.emissions.value, + ).toFixed(2), + transportation: getEquivalentCarKm( + newRadialChartData.emissions.value, + ).toFixed(2), + tvTime: getEquivalentTvTime( + newRadialChartData.energy.value, + ).toFixed(2), + }); + } catch (error) { + console.error("Error fetching project data:", error); + } finally { + setIsLoading(false); + } + }, + [projectId], + ); + + const handleSettingsClick = async () => { + try { + const updatedProject = await getOneProject(projectId!); + if (updatedProject) { + setProject(updatedProject); + } + } catch (error) { + console.error("Error refreshing project data:", error); + } + }; + + const handleRefresh = useCallback(async () => { + const projectPromise = getOneProject(projectId!).then((p) => { + if (p) setProject(p); + }); + const experimentsPromise = refreshExperimentList(); + const reportPromise = fetchEmissionsReport(date); + await Promise.all([projectPromise, experimentsPromise, reportPromise]); + }, [projectId, date, refreshExperimentList, fetchEmissionsReport]); + + useEffect(() => { + const fetchProjectDetails = async () => { + try { + const p = await getOneProject(projectId!); + if (!p) return; + setProject(p); + } catch (error) { + console.error("Error fetching project description:", error); + } + }; + + fetchProjectDetails(); + refreshExperimentList(); + }, [projectId, refreshExperimentList]); + + useEffect(() => { + if (projectId) { + fetchEmissionsReport(date); + } + }, [projectId, date, fetchEmissionsReport]); + + const handleExperimentClick = useCallback( + (experimentId: string) => { + if (experimentId === selectedExperimentId) { + setSelectedExperimentId(""); + setSelectedRunId(""); + return; + } + setSelectedExperimentId(experimentId); + setSelectedRunId(""); + }, + [selectedExperimentId], + ); + + const handleRunClick = useCallback( + (runId: string) => { + if (runId === selectedRunId) { + setSelectedRunId(""); + return; + } + setSelectedRunId(runId); + }, + [selectedRunId], + ); + + return ( +
+ +
+ + setDate(newDates || getDefaultDateRange()) + } + radialChartData={radialChartData} + convertedValues={convertedValues} + experimentsReportData={experimentsReportData} + runData={runData} + selectedExperimentId={selectedExperimentId} + selectedRunId={selectedRunId} + projectExperiments={projectExperiments} + onExperimentClick={handleExperimentClick} + onRunClick={handleRunClick} + onSettingsClick={handleSettingsClick} + onRefresh={handleRefresh} + isLoading={isLoading} + /> +
+
+ ); +} diff --git a/webapp/src/pages/ProjectSettingsPage.tsx b/webapp/src/pages/ProjectSettingsPage.tsx new file mode 100644 index 000000000..9092d3740 --- /dev/null +++ b/webapp/src/pages/ProjectSettingsPage.tsx @@ -0,0 +1,128 @@ +import { Button } from "@/components/ui/button"; +import { Input } from "@/components/ui/input"; +import { Label } from "@/components/ui/label"; +import { Switch } from "@/components/ui/switch"; +import { getOneProject, updateProject } from "@/api/projects"; +import { Project } from "@/api/schemas"; +import { ProjectTokensTable } from "@/components/projectTokens/projectTokenTable"; +import ShareProjectButton from "@/components/share-project-button"; +import { useParams } from "react-router-dom"; +import { useEffect, useState } from "react"; +import { toast } from "sonner"; +import Loader from "@/components/loader"; +import { handleError } from "@/api/errors"; + +export default function ProjectSettingsPage() { + const { projectId } = useParams<{ projectId: string }>(); + const [project, setProject] = useState(null); + const [isLoading, setIsLoading] = useState(true); + + useEffect(() => { + async function fetchProject() { + try { + const data = await getOneProject(projectId!); + setProject(data); + } catch (error) { + console.error("Error fetching project:", error); + } finally { + setIsLoading(false); + } + } + fetchProject(); + }, [projectId]); + + const handleSubmit = async (e: React.FormEvent) => { + e.preventDefault(); + try { + const formData = new FormData(e.currentTarget); + const updated = await updateProject(projectId!, { + name: formData.get("name") as string, + description: formData.get("description") as string, + public: formData.has("isPublic"), + }); + setProject(updated); + toast.success("Project updated"); + } catch (err) { + handleError(err); + } + }; + + if (isLoading) { + return ; + } + + if (!project) { + return
Project not found
; + } + + return ( +
+
+
+
+

Settings

+

Project Settings

+
+
+ +
+
+
+
+
+

General

+
+
+
+
+
+ + +
+
+ + +
+
+ + +

+ (enables public sharing link) +

+
+
+
+ +
+
+
+
+

API tokens

+
+ +
+
+
+
+ ); +} diff --git a/webapp/src/pages/ProjectsPage.tsx b/webapp/src/pages/ProjectsPage.tsx new file mode 100644 index 000000000..c8aefb6ba --- /dev/null +++ b/webapp/src/pages/ProjectsPage.tsx @@ -0,0 +1,139 @@ +import BreadcrumbHeader from "@/components/breadcrumb"; +import CreateProjectModal from "@/components/createProjectModal"; +import CustomRow from "@/components/custom-row"; +import DeleteProjectModal from "@/components/delete-project-modal"; +import ErrorMessage from "@/components/error-message"; +import Loader from "@/components/loader"; +import { Button } from "@/components/ui/button"; +import { Card } from "@/components/ui/card"; +import { Table, TableBody } from "@/components/ui/table"; +import { fetcher } from "@/api/swr"; +import { useModal } from "@/hooks/useModal"; +import { getProjects, deleteProject } from "@/api/projects"; +import { Project } from "@/api/schemas"; +import { useEffect, useState } from "react"; +import { useParams } from "react-router-dom"; +import useSWR from "swr"; +import { toast } from "sonner"; + +export default function ProjectsPage() { + const { organizationId } = useParams<{ organizationId: string }>(); + const createModal = useModal(); + const deleteModal = useModal(); + const [projectList, setProjectList] = useState([]); + const [projectToDelete, setProjectToDelete] = useState(null); + + const handleClick = async () => { + createModal.open(); + }; + + const refreshProjectList = async () => { + const projects = await getProjects(organizationId!); + setProjectList(projects || []); + }; + + const handleDeleteClick = (project: Project) => { + setProjectToDelete(project); + deleteModal.open(); + }; + + const handleDeleteConfirm = async (projectId: string) => { + try { + await deleteProject(projectId); + toast.success("Project deleted successfully"); + await refreshProjectList(); + } catch (error) { + console.error("Error deleting project:", error); + toast.error("Failed to delete project"); + } + }; + + const { + data: projects, + error, + isLoading, + } = useSWR( + `/projects?organization=${organizationId}`, + fetcher, + {}, + ); + + useEffect(() => { + if (projects) { + setProjectList(projects); + } + }, [projects]); + + if (isLoading) { + return ; + } + + if (error) { + return ; + } + + return ( +
+ +
+
+

Projects

+ + +
+ + + + {projectList && + projectList + .sort((a, b) => + a.name.toLowerCase().localeCompare(b.name.toLowerCase()), + ) + .map((project) => ( + handleDeleteClick(project)} + /> + ))} + +
+
+ {projectToDelete && ( + + )} +
+
+ ); +} diff --git a/webapp/src/pages/PublicProjectPage.tsx b/webapp/src/pages/PublicProjectPage.tsx new file mode 100644 index 000000000..c5603da22 --- /dev/null +++ b/webapp/src/pages/PublicProjectPage.tsx @@ -0,0 +1,212 @@ +import { useState, useEffect, useCallback } from "react"; +import { useNavigate, useParams } from "react-router-dom"; +import { DateRange } from "react-day-picker"; +import { ExperimentReport, Project, Experiment } from "@/api/schemas"; +import PublicProjectDashboard from "@/components/public-project-dashboard"; +import { + getEquivalentCarKm, + getEquivalentCitizenPercentage, + getEquivalentTvTime, +} from "@/helpers/constants"; +import { fetchApi } from "@/api/client"; +import { ProjectSchema, ExperimentReportSchema, ExperimentSchema } from "@/api/schemas"; +import ErrorMessage from "@/components/error-message"; +import Loader from "@/components/loader"; +import { Alert, AlertDescription, AlertTitle } from "@/components/ui/alert"; +import { AlertCircle } from "lucide-react"; +import { getDefaultDateRange } from "@/helpers/date-utils"; + +export default function PublicProjectPage() { + const { projectId: encryptedId } = useParams<{ projectId: string }>(); + const navigate = useNavigate(); + + const [isLoading, setIsLoading] = useState(true); + const [error, setError] = useState(null); + const [projectId, setProjectId] = useState(null); + const [project, setProject] = useState(null); + + const default_date = getDefaultDateRange(); + const [date, setDate] = useState(default_date); + const [projectExperiments, setProjectExperiments] = useState([]); + const [experimentsReportData, setExperimentsReportData] = useState([]); + + const [radialChartData, setRadialChartData] = useState({ + energy: { label: "kWh", value: 0 }, + emissions: { label: "kg eq CO2", value: 0 }, + duration: { label: "days", value: 0 }, + }); + + const [runData, setRunData] = useState({ + experimentId: "", + startDate: default_date.from.toISOString(), + endDate: default_date.to.toISOString(), + }); + const [convertedValues, setConvertedValues] = useState({ + citizen: "0", + transportation: "0", + tvTime: "0", + }); + const [selectedExperimentId, setSelectedExperimentId] = useState(""); + const [selectedRunId, setSelectedRunId] = useState(""); + + const refreshExperimentList = useCallback(async () => { + if (!projectId) return; + try { + const experiments = await fetchApi( + `/projects/${projectId}/experiments`, + ExperimentSchema.array(), + ); + setProjectExperiments(experiments); + } catch { + setProjectExperiments([]); + } + }, [projectId]); + + // Decrypt the project ID via the backend + useEffect(() => { + const decrypt = async () => { + try { + setIsLoading(true); + const result = await fetchApi( + `/projects/public/${encryptedId}`, + ProjectSchema, + ); + setProjectId(result.id); + setProject(result); + } catch { + setError("Invalid project link or the project no longer exists."); + } finally { + setIsLoading(false); + } + }; + decrypt(); + }, [encryptedId]); + + useEffect(() => { + if (projectId && project) { + refreshExperimentList(); + } + }, [projectId, project, refreshExperimentList]); + + useEffect(() => { + async function fetchData() { + if (!projectId || !project) return; + + setIsLoading(true); + try { + const report = await fetchApi( + `/projects/${projectId}/experiments/sums?start_date=${date?.from?.toISOString()}&end_date=${date?.to?.toISOString()}`, + ExperimentReportSchema.array(), + ); + + setExperimentsReportData(report); + + const newRadialChartData = { + energy: { + label: "kWh", + value: parseFloat( + report.reduce((n, { energy_consumed }) => n + energy_consumed, 0).toFixed(2), + ), + }, + emissions: { + label: "kg eq CO2", + value: parseFloat( + report.reduce((n, { emissions }) => n + emissions, 0).toFixed(2), + ), + }, + duration: { + label: "days", + value: parseFloat( + report.reduce((n, { duration }) => n + duration / 86400, 0).toFixed(2), + ), + }, + }; + + setRadialChartData(newRadialChartData); + + if (report.length > 0) { + setRunData({ + experimentId: report[0]?.experiment_id ?? "", + startDate: date?.from?.toISOString() ?? "", + endDate: date?.to?.toISOString() ?? "", + }); + setSelectedExperimentId(report[0]?.experiment_id ?? ""); + } + + setConvertedValues({ + citizen: getEquivalentCitizenPercentage(newRadialChartData.emissions.value).toFixed(2), + transportation: getEquivalentCarKm(newRadialChartData.emissions.value).toFixed(2), + tvTime: getEquivalentTvTime(newRadialChartData.energy.value).toFixed(2), + }); + } catch (error) { + console.error("Error fetching data:", error); + } finally { + setIsLoading(false); + } + } + + if (projectId && project) { + fetchData(); + } + }, [projectId, project, date]); + + const handleExperimentClick = useCallback((experimentId: string) => { + setSelectedExperimentId(experimentId); + setRunData((prevData) => ({ ...prevData, experimentId })); + setSelectedRunId(""); + }, []); + + const handleRunClick = useCallback((runId: string) => { + setSelectedRunId(runId); + }, []); + + if (isLoading && !project) { + return ; + } + + if (error) { + return ( +
+ + + Error + {error} + +
+ +
+
+ ); + } + + if (!project) { + return ; + } + + return ( +
+
+ setDate(newDates || getDefaultDateRange())} + radialChartData={radialChartData} + convertedValues={convertedValues} + experimentsReportData={experimentsReportData} + projectExperiments={projectExperiments} + runData={runData} + selectedExperimentId={selectedExperimentId} + selectedRunId={selectedRunId} + onExperimentClick={handleExperimentClick} + onRunClick={handleRunClick} + isLoading={isLoading} + /> +
+
+ ); +} diff --git a/webapp/src/router.tsx b/webapp/src/router.tsx new file mode 100644 index 000000000..8c5d9628b --- /dev/null +++ b/webapp/src/router.tsx @@ -0,0 +1,103 @@ +import { createBrowserRouter } from "react-router-dom"; +import { lazy, Suspense } from "react"; +import DashboardLayout from "./layouts/DashboardLayout"; +import AuthGuard from "./components/auth-guard"; +import Loader from "./components/loader"; + +const LandingPage = lazy(() => import("./pages/LandingPage")); +const PrivacyPage = lazy(() => import("./pages/PrivacyPage")); +const PublicProjectPage = lazy(() => import("./pages/PublicProjectPage")); +const HomePage = lazy(() => import("./pages/HomePage")); +const OrgDashboardPage = lazy(() => import("./pages/OrgDashboardPage")); +const ProjectsPage = lazy(() => import("./pages/ProjectsPage")); +const ProjectDashboardPage = lazy(() => import("./pages/ProjectDashboardPage")); +const ProjectSettingsPage = lazy(() => import("./pages/ProjectSettingsPage")); +const MembersPage = lazy(() => import("./pages/MembersPage")); + +function SuspenseWrapper({ children }: { children: React.ReactNode }) { + return }>{children}; +} + +export const router = createBrowserRouter([ + { + path: "/", + element: ( + + + + ), + }, + { + path: "/privacy", + element: ( + + + + ), + }, + { + path: "/public/projects/:projectId", + element: ( + + + + ), + }, + { + element: ( + + + + ), + children: [ + { + path: "/home", + element: ( + + + + ), + }, + { + path: "/:organizationId", + element: ( + + + + ), + }, + { + path: "/:organizationId/projects", + element: ( + + + + ), + }, + { + path: "/:organizationId/projects/:projectId", + element: ( + + + + ), + }, + { + path: "/:organizationId/projects/:projectId/settings", + element: ( + + + + ), + }, + { + path: "/:organizationId/members", + element: ( + + + + ), + }, + ], + }, +]); diff --git a/webapp/src/server-functions/ERROR_HANDLING.md b/webapp/src/server-functions/ERROR_HANDLING.md deleted file mode 100644 index 7928252a9..000000000 --- a/webapp/src/server-functions/ERROR_HANDLING.md +++ /dev/null @@ -1,151 +0,0 @@ -# Error Handling Standards for Server Functions - -## Overview - -This document defines the standard error handling patterns for all server-side API functions in the codebase. - -## Core Principles - -1. **User feedback comes from the UI layer** - Server functions focus on data retrieval/mutation -2. **Consistent patterns** - Similar operations should handle errors the same way -3. **Graceful degradation** - Read operations should fail gracefully with empty data -4. **Clear failure signals** - Write operations should throw errors for the UI to catch - ---- - -## Standard Patterns - -### Pattern A: Read Operations (GET) - -**Use for:** Fetching data, list operations, queries - -```typescript -export async function getData(id: string): Promise { - const result = await fetchApi(`/endpoint/${id}`); - - // Return empty array/null on failure - UI will show "no data" state - if (!result) { - return []; // or null for single items - } - - return result; -} -``` - -**Why:** - -- Users can still use the app even if one data source fails -- UI naturally shows "no data" or "empty" states -- Errors are already logged by `fetchApi` - -### Pattern B: Write Operations (POST/PUT/PATCH/DELETE) - -**Use for:** Creating, updating, deleting data - -```typescript -export async function createData(data: Data): Promise { - const result = await fetchApi("/endpoint", { - method: "POST", - body: JSON.stringify(data), - }); - - // Throw error - UI will catch and show toast/error message - if (!result) { - throw new Error("Failed to create data"); - } - - return result; -} -``` - -**Why:** - -- Write operations are user-initiated actions that need feedback -- UI layer can catch the error and show appropriate toast/modal -- Clear signal that the operation failed - -### Pattern C: Critical Read Operations - -**Use for:** Data required for the page to function (rare) - -```typescript -export async function getCriticalData(id: string): Promise { - const result = await fetchApi(`/critical/${id}`); - - if (!result) { - throw new Error("Failed to load required data"); - } - - return result; -} -``` - -**Why:** - -- Some data is essential for the page to work -- UI can show error boundary or redirect - ---- - -## Migration Guide - -### ❌ Avoid Try-Catch in Server Functions - -```typescript -// DON'T DO THIS - fetchApi already handles errors -try { - const result = await fetchApi(endpoint); - return result || []; -} catch (error) { - console.error(error); - return []; -} -``` - -```typescript -// DO THIS - Let fetchApi handle errors, check result -const result = await fetchApi(endpoint); -return result || []; -``` - -### UI Layer Responsibilities - -The UI components should handle errors from write operations: - -```typescript -// In React component -const handleCreate = async () => { - try { - await createData(formData); - toast.success("Created successfully"); - } catch (error) { - toast.error(error.message || "Failed to create"); - } -}; -``` - ---- - -## Examples by Function Type - -| Function Type | Pattern | Return on Error | Example | -| -------------------- | ------- | --------------- | ------------------ | -| `getProjects()` | A | `[]` | List of projects | -| `getOneProject()` | A | `null` | Single project | -| `createProject()` | B | `throw` | Create new project | -| `updateProject()` | B | `throw` | Update project | -| `deleteProject()` | B | `void` (throws) | Delete project | -| `getOrganizations()` | A | `[]` | List of orgs | -| `getUserProfile()` | C | `throw` | Required for auth | - ---- - -## Decision Tree - -``` -Is this a READ operation? -├─ Yes: Is the data critical for the page? -│ ├─ Yes: Use Pattern C (throw) -│ └─ No: Use Pattern A (return empty) -└─ No (WRITE operation): Use Pattern B (throw) -``` diff --git a/webapp/src/server-functions/experiments.ts b/webapp/src/server-functions/experiments.ts deleted file mode 100644 index 986313be2..000000000 --- a/webapp/src/server-functions/experiments.ts +++ /dev/null @@ -1,71 +0,0 @@ -import { Experiment } from "@/types/experiment"; -import { ExperimentReport } from "@/types/experiment-report"; -import { fetchApi } from "@/utils/api"; -import { DateRange } from "react-day-picker"; - -export async function createExperiment( - experiment: Experiment, -): Promise { - const result = await fetchApi("/experiments", { - method: "POST", - body: JSON.stringify(experiment), - }); - - if (!result) { - throw new Error("Failed to create experiment"); - } - - return result; -} -export async function getExperiments(projectId: string): Promise { - const result = await fetchApi( - `/projects/${projectId}/experiments`, - ); - - if (!result) { - return []; - } - - return result.map((experiment: Experiment) => { - return { - id: experiment.id, - name: experiment.name, - description: experiment.description, - project_id: experiment.project_id, - created_at: experiment.timestamp, - }; - }); -} -export async function getProjectEmissionsByExperiment( - projectId: string, - dateRange: DateRange, -): Promise { - let url = `/projects/${projectId}/experiments/sums`; - - if (dateRange?.from || dateRange?.to) { - const params = new URLSearchParams(); - if (dateRange.from) { - params.append("start_date", dateRange.from.toISOString()); - } - if (dateRange.to) { - params.append("end_date", dateRange.to.toISOString()); - } - url += `?${params.toString()}`; - } - - const result = await fetchApi(url, {}); - if (!result) { - return []; - } - - return result.map((experimentReport: ExperimentReport) => { - return { - experiment_id: experimentReport.experiment_id, - description: experimentReport.description, - name: experimentReport.name, - emissions: experimentReport.emissions, - energy_consumed: experimentReport.energy_consumed, - duration: experimentReport.duration, - }; - }); -} diff --git a/webapp/src/server-functions/organizations.ts b/webapp/src/server-functions/organizations.ts deleted file mode 100644 index bca8c7b13..000000000 --- a/webapp/src/server-functions/organizations.ts +++ /dev/null @@ -1,68 +0,0 @@ -import { Organization } from "@/types/organization"; -import { OrganizationReport } from "@/types/organization-report"; -import { DateRange } from "react-day-picker"; -import { fetchApiServer } from "@/helpers/api-server"; - -export async function getOrganizationEmissionsByProject( - organizationId: string, - dateRange: DateRange | undefined, -): Promise { - let endpoint = `/organizations/${organizationId}/sums`; - - if (dateRange?.from && dateRange?.to) { - endpoint += `?start_date=${dateRange.from.toISOString()}&end_date=${dateRange.to.toISOString()}`; - } - - const result = await fetchApiServer(endpoint); - - // Handle case when no emissions data is found - if (!result) { - return { - name: "", - emissions: 0, - energy_consumed: 0, - duration: 0, - }; - } - - return result; -} - -export async function getDefaultOrgId(): Promise { - const orgs = await fetchApiServer("/organizations"); - - // Return null on failure (Pattern A - Read operation) - if (!orgs || orgs.length === 0) { - return null; - } - - return orgs[0].id; -} - -export async function getOrganizations(): Promise { - const orgs = await fetchApiServer("/organizations"); - - // Return empty array on failure (Pattern A - Read operation) - if (!orgs) { - return []; - } - - return orgs; -} - -export const createOrganization = async (organization: { - name: string; - description: string; -}): Promise => { - const result = await fetchApiServer("/organizations", { - method: "POST", - body: JSON.stringify(organization), - }); - - // Throw on failure (Pattern B - Write operation) - if (!result) { - throw new Error("Failed to create organization"); - } - - return result; -}; diff --git a/webapp/src/server-functions/projectTokens.ts b/webapp/src/server-functions/projectTokens.ts deleted file mode 100644 index e692aeb04..000000000 --- a/webapp/src/server-functions/projectTokens.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { IProjectToken } from "@/types/project"; -import { fetchApi } from "@/utils/api"; - -/** - * Retrieves the list of tokens for a given project - */ -export async function getProjectTokens( - projectId: string, -): Promise { - const data = await fetchApi( - `/projects/${projectId}/api-tokens`, - ); - - // Return empty array on failure (Pattern A - Read operation) - if (!data) { - return []; - } - - return data; -} - -export async function createProjectToken( - projectId: string, - tokenName: string, - access?: Number, -): Promise { - const result = await fetchApi( - `/projects/${projectId}/api-tokens`, - { - method: "POST", - body: JSON.stringify({ name: tokenName, access }), - }, - ); - - if (!result) { - throw new Error("Failed to create project token"); - } - - return result; -} -export async function deleteProjectToken( - projectId: string, - tokenId: string, -): Promise { - await fetchApi(`/projects/${projectId}/api-tokens/${tokenId}`, { - method: "DELETE", - }); -} diff --git a/webapp/src/server-functions/projects.ts b/webapp/src/server-functions/projects.ts deleted file mode 100644 index 056d0ac9b..000000000 --- a/webapp/src/server-functions/projects.ts +++ /dev/null @@ -1,64 +0,0 @@ -import { Project, ProjectInputs } from "@/types/project"; -import { fetchApiServer } from "@/helpers/api-server"; - -export const createProject = async ( - organizationId: string, - project: { name: string; description: string }, -): Promise => { - const result = await fetchApiServer("/projects", { - method: "POST", - body: JSON.stringify({ - ...project, - organization_id: organizationId, - }), - }); - - // Throw on failure (Pattern B - Write operation) - if (!result) { - throw new Error("Failed to create project"); - } - - return result; -}; - -export const updateProject = async ( - projectId: string, - project: ProjectInputs, -): Promise => { - const result = await fetchApiServer(`/projects/${projectId}`, { - method: "PATCH", - body: JSON.stringify(project), - }); - - // Throw on failure (Pattern B - Write operation) - if (!result) { - throw new Error("Failed to update project"); - } - - return result; -}; - -export const getProjects = async ( - organizationId: string, -): Promise => { - const projects = await fetchApiServer( - `/projects?organization=${organizationId}`, - ); - if (!projects) { - return []; - } - return projects; -}; - -export const getOneProject = async ( - projectId: string, -): Promise => { - const project = await fetchApiServer(`/projects/${projectId}`); - return project; -}; - -export const deleteProject = async (projectId: string): Promise => { - await fetchApiServer(`/projects/${projectId}`, { - method: "DELETE", - }); -}; diff --git a/webapp/src/server-functions/runs.ts b/webapp/src/server-functions/runs.ts deleted file mode 100644 index 0ce1bf725..000000000 --- a/webapp/src/server-functions/runs.ts +++ /dev/null @@ -1,96 +0,0 @@ -import { Emission } from "@/types/emission"; -import { EmissionsTimeSeries } from "@/types/emissions-time-series"; -import { RunMetadata } from "@/types/run-metadata"; -import { fetchApi } from "@/utils/api"; -import { RunReport } from "@/types/run-report"; - -export async function getRunMetadata( - runId: string, -): Promise { - const result = await fetchApi(`/runs/${runId}`); - return result; -} - -export async function getRunEmissionsByExperiment( - experimentId: string, - startDate: string, - endDate: string, -): Promise { - if (!experimentId || experimentId == "") { - return []; - } - - const result = await fetchApi( - `/experiments/${experimentId}/runs/sums?start_date=${startDate}&end_date=${endDate}`, - ); - - if (!result) { - return []; - } - - return result.map((runReport: any) => { - return { - runId: runReport.run_id, - emissions: runReport.emissions, - timestamp: runReport.timestamp, - energy_consumed: runReport.energy_consumed, - duration: runReport.duration, - }; - }); -} - -export async function getEmissionsTimeSeries( - runId: string, -): Promise { - const [runMetadataData, emissionsData] = await Promise.all([ - fetchApi(`/runs/${runId}`), - fetchApi<{ items: Emission[] }>(`/runs/${runId}/emissions`), - ]); - - // Return empty data on failure (Pattern A - Read operation) - if (!runMetadataData || !emissionsData) { - return { - runId, - emissions: [], - metadata: null, - }; - } - - const metadata: RunMetadata = { - timestamp: runMetadataData.timestamp, - experiment_id: runMetadataData.experiment_id, - os: runMetadataData.os, - python_version: runMetadataData.python_version, - codecarbon_version: runMetadataData.codecarbon_version, - cpu_count: runMetadataData.cpu_count, - cpu_model: runMetadataData.cpu_model, - gpu_count: runMetadataData.gpu_count, - gpu_model: runMetadataData.gpu_model, - longitude: runMetadataData.longitude, - latitude: runMetadataData.latitude, - region: runMetadataData.region, - provider: runMetadataData.provider, - ram_total_size: runMetadataData.ram_total_size, - tracking_mode: runMetadataData.tracking_mode, - }; - - const emissions: Emission[] = emissionsData.items.map((item: any) => ({ - emission_id: item.run_id, - timestamp: item.timestamp, - emissions_sum: item.emissions_sum, - emissions_rate: item.emissions_rate, - cpu_power: item.cpu_power, - gpu_power: item.gpu_power, - ram_power: item.ram_power, - cpu_energy: item.cpu_energy, - gpu_energy: item.gpu_energy, - ram_energy: item.ram_energy, - energy_consumed: item.energy_consumed, - })); - - return { - runId, - emissions, - metadata, - }; -} diff --git a/webapp/src/types/emission.ts b/webapp/src/types/emission.ts deleted file mode 100644 index 4af44e71c..000000000 --- a/webapp/src/types/emission.ts +++ /dev/null @@ -1,13 +0,0 @@ -export interface Emission { - emission_id: string; - timestamp: string; - emissions_sum: number; - emissions_rate: number; - cpu_power: number; - gpu_power: number; - ram_power: number; - cpu_energy: number; - gpu_energy: number; - ram_energy: number; - energy_consumed: number; -} diff --git a/webapp/src/types/emissions-time-series.ts b/webapp/src/types/emissions-time-series.ts deleted file mode 100644 index 88522b37c..000000000 --- a/webapp/src/types/emissions-time-series.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { Emission } from "@/types/emission"; -import { RunMetadata } from "@/types/run-metadata"; - -export interface EmissionsTimeSeries { - runId: string; - emissions: Emission[]; - metadata: RunMetadata | null; -} diff --git a/webapp/src/types/experiment-report.ts b/webapp/src/types/experiment-report.ts deleted file mode 100644 index 006ba438c..000000000 --- a/webapp/src/types/experiment-report.ts +++ /dev/null @@ -1,8 +0,0 @@ -export interface ExperimentReport { - experiment_id: string; - name: string; - emissions: number; - energy_consumed: number; - duration: number; - description?: string; -} diff --git a/webapp/src/types/experiment.ts b/webapp/src/types/experiment.ts deleted file mode 100644 index 74f9b2553..000000000 --- a/webapp/src/types/experiment.ts +++ /dev/null @@ -1,13 +0,0 @@ -export interface Experiment { - timestamp?: string; - name: string; - description: string; - on_cloud?: boolean; - project_id: string; - country_name?: string; - country_iso_code?: string; - region?: string; - cloud_provider?: string; - cloud_region?: string; - id?: string; -} diff --git a/webapp/src/types/organization-report.ts b/webapp/src/types/organization-report.ts deleted file mode 100644 index 545cfa1a2..000000000 --- a/webapp/src/types/organization-report.ts +++ /dev/null @@ -1,6 +0,0 @@ -export interface OrganizationReport { - name: string; - emissions: number; - energy_consumed: number; - duration: number; -} diff --git a/webapp/src/types/organization.ts b/webapp/src/types/organization.ts deleted file mode 100644 index 70fd3f519..000000000 --- a/webapp/src/types/organization.ts +++ /dev/null @@ -1,5 +0,0 @@ -export interface Organization { - id: string; - name: string; - description: string; -} diff --git a/webapp/src/types/project-dashboard.ts b/webapp/src/types/project-dashboard.ts deleted file mode 100644 index 5dff10190..000000000 --- a/webapp/src/types/project-dashboard.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { DateRange } from "react-day-picker"; -import { Project } from "./project"; -import { ExperimentReport } from "./experiment-report"; -import { Experiment } from "./experiment"; - -export interface RadialChartData { - energy: { label: string; value: number }; - emissions: { label: string; value: number }; - duration: { label: string; value: number }; -} - -export interface ConvertedValues { - citizen: string; - transportation: string; - tvTime: string; -} - -export interface ProjectDashboardProps { - project: Project; - date: DateRange; - onDateChange: (newDate: DateRange | undefined) => void; - radialChartData: RadialChartData; - convertedValues: ConvertedValues; - experimentsReportData: ExperimentReport[]; - projectExperiments: Experiment[]; - runData: { - experimentId: string; - startDate: string; - endDate: string; - }; - selectedExperimentId: string; - selectedRunId: string; - onExperimentClick: (experimentId: string) => void; - onRunClick: (runId: string) => void; - onSettingsClick: () => void; - isLoading?: boolean; -} diff --git a/webapp/src/types/project.ts b/webapp/src/types/project.ts deleted file mode 100644 index 3ead6bd02..000000000 --- a/webapp/src/types/project.ts +++ /dev/null @@ -1,23 +0,0 @@ -export interface Project { - id: string; - name: string; - description: string; - public: boolean; - organizationId: string; - experiments: string[]; -} - -export interface ProjectInputs { - name: string; - description: string; - public: boolean; -} - -export interface IProjectToken { - id: string; - project_id: string; - last_used: string | null; - name: string; - token: string; - access: number; -} diff --git a/webapp/src/types/public-project-dashboard.ts b/webapp/src/types/public-project-dashboard.ts deleted file mode 100644 index 749e7aa9b..000000000 --- a/webapp/src/types/public-project-dashboard.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { DateRange } from "react-day-picker"; -import { Project } from "./project"; -import { RadialChartData, ConvertedValues } from "./project-dashboard"; -import { ExperimentReport } from "./experiment-report"; -import { Experiment } from "./experiment"; - -export interface PublicProjectDashboardProps { - project: Project; - date: DateRange; - onDateChange: (newDate: DateRange | undefined) => void; - radialChartData: RadialChartData; - convertedValues: ConvertedValues; - experimentsReportData: ExperimentReport[]; - projectExperiments: Experiment[]; - runData: { - experimentId: string; - startDate: string; - endDate: string; - }; - selectedExperimentId: string; - selectedRunId: string; - onExperimentClick: (experimentId: string) => void; - onRunClick: (runId: string) => void; - isLoading?: boolean; -} diff --git a/webapp/src/types/run-metadata.ts b/webapp/src/types/run-metadata.ts deleted file mode 100644 index 997fe2507..000000000 --- a/webapp/src/types/run-metadata.ts +++ /dev/null @@ -1,17 +0,0 @@ -export interface RunMetadata { - timestamp: string; - experiment_id: string; - os: string; - python_version: string; - codecarbon_version: string; - cpu_count: number; - cpu_model: string; - gpu_count: number; - gpu_model: string; - longitude: number; - latitude: number; - region: string; - provider: string; - ram_total_size: number; - tracking_mode: string; -} diff --git a/webapp/src/types/run-report.ts b/webapp/src/types/run-report.ts deleted file mode 100644 index cafd152ef..000000000 --- a/webapp/src/types/run-report.ts +++ /dev/null @@ -1,7 +0,0 @@ -export interface RunReport { - runId: string; - emissions: number; - timestamp: string; - energy_consumed: number; - duration: number; -} diff --git a/webapp/src/types/user.ts b/webapp/src/types/user.ts deleted file mode 100644 index 8e26c9449..000000000 --- a/webapp/src/types/user.ts +++ /dev/null @@ -1,7 +0,0 @@ -export interface User { - id: string; - email: string; - name: string; - organizations: string[]; - is_active: boolean; -} diff --git a/webapp/src/utils/api.ts b/webapp/src/utils/api.ts deleted file mode 100644 index 2b0cf11d5..000000000 --- a/webapp/src/utils/api.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { fetchApiClient } from "@/helpers/api-client"; -import { fetchApiServer } from "@/helpers/api-server"; - -/** - * API utility functions that help determine whether to use client or server API - * and provide consistent error handling - */ - -// Universal API function that works in both client and server components -export async function fetchApi( - endpoint: string, - options?: RequestInit, -): Promise { - // Both fetchApiClient and fetchApiServer handle errors internally - // and return null on failure, so no try/catch needed here - if (typeof window !== "undefined") { - return await fetchApiClient(endpoint, options); - } - return await fetchApiServer(endpoint, options); -} diff --git a/webapp/src/utils/crypto.ts b/webapp/src/utils/crypto.ts deleted file mode 100644 index 8a34c17cc..000000000 --- a/webapp/src/utils/crypto.ts +++ /dev/null @@ -1,68 +0,0 @@ -"use server"; - -import crypto from "crypto"; - -// Environment variable to use as encryption key (should be 32 bytes for AES-256) -const SECRET_KEY = process.env.PROJECT_ENCRYPTION_KEY as string; - -/** - * Encrypts a project ID to create a short, consistent secure sharing link - * @param projectId The original project ID to encrypt - * @returns A short encrypted string safe to use in URLs - */ -export async function encryptProjectId(projectId: string): Promise { - // Create a deterministic IV by hashing the project ID with our secret key - // This ensures the same project ID always generates the same encrypted output - // while still being secure due to the secret key - const hmac = crypto.createHmac("sha256", SECRET_KEY); - hmac.update(projectId); - // Use first 16 bytes of the HMAC output as our IV - const iv = Buffer.from(hmac.digest().subarray(0, 16)); - - // Create a cipher using AES-256-CBC with our deterministic IV - const cipher = crypto.createCipheriv( - "aes-256-cbc", - Buffer.from(SECRET_KEY.substring(0, 32).padEnd(32, "0")), - iv, - ); - - // Encrypt the project ID - let encrypted = cipher.update(projectId, "utf8", "base64"); - encrypted += cipher.final("base64"); - - // Combine IV and encrypted data and convert to URL-safe base64 - const combined = Buffer.concat([iv, Buffer.from(encrypted, "base64")]); - return combined.toString("base64url"); -} - -/** - * Decrypts an encrypted project ID from a sharing link - * @param encryptedData The encrypted project ID from the URL - * @returns The original project ID - */ -export async function decryptProjectId(encryptedData: string): Promise { - try { - // Convert from base64url to buffer - const encryptedBuffer = Buffer.from(encryptedData, "base64url"); - - // Extract IV (first 16 bytes) and actual encrypted data - const iv = encryptedBuffer.subarray(0, 16); - const encryptedText = encryptedBuffer.subarray(16).toString("base64"); - - // Create decipher - const decipher = crypto.createDecipheriv( - "aes-256-cbc", - Buffer.from(SECRET_KEY.substring(0, 32).padEnd(32, "0")), - iv, - ); - - // Decrypt the data - let decrypted = decipher.update(encryptedText, "base64", "utf8"); - decrypted += decipher.final("utf8"); - - return decrypted; - } catch (error) { - console.error("Failed to decrypt project ID:", error); - throw new Error("Invalid or corrupted project link"); - } -} diff --git a/webapp/src/utils/export.ts b/webapp/src/utils/export.ts index ee3298907..176e9a7e8 100644 --- a/webapp/src/utils/export.ts +++ b/webapp/src/utils/export.ts @@ -1,9 +1,11 @@ -import { getRunMetadata } from "@/server-functions/runs"; -import { Emission } from "@/types/emission"; -import { EmissionsTimeSeries } from "@/types/emissions-time-series"; -import { ExperimentReport } from "@/types/experiment-report"; -import { RunMetadata } from "@/types/run-metadata"; -import { RunReport } from "@/types/run-report"; +import { getRunMetadata } from "@/api/runs"; +import { + Emission, + EmissionsTimeSeries, + ExperimentReport, + RunMetadata, + RunReport, +} from "@/api/schemas"; // Enhanced run report with metadata and emissions interface EnhancedRunReport extends Omit { diff --git a/webapp/tailwind.config.ts b/webapp/tailwind.config.ts index a23faa5fa..31cf1a474 100644 --- a/webapp/tailwind.config.ts +++ b/webapp/tailwind.config.ts @@ -2,12 +2,7 @@ import type { Config } from "tailwindcss"; const config = { darkMode: ["class"], - content: [ - "./pages/**/*.{ts,tsx}", - "./components/**/*.{ts,tsx}", - "./app/**/*.{ts,tsx}", - "./src/**/*.{ts,tsx}", - ], + content: ["./index.html", "./src/**/*.{ts,tsx}"], prefix: "", theme: { container: { diff --git a/webapp/tsconfig.json b/webapp/tsconfig.json index 9c6c6d5e8..8f3d2fcca 100644 --- a/webapp/tsconfig.json +++ b/webapp/tsconfig.json @@ -1,27 +1,19 @@ { - "compilerOptions": { - "lib": ["dom", "dom.iterable", "esnext"], - "allowJs": true, - "skipLibCheck": true, - "strict": true, - "noEmit": true, - "esModuleInterop": true, - "module": "esnext", - "moduleResolution": "bundler", - "resolveJsonModule": true, - "isolatedModules": true, - "jsx": "preserve", - "incremental": true, - "plugins": [ - { - "name": "next" - } - ], - "paths": { - "@/*": ["./src/*"] - }, - "target": "ES2017" - }, - "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"], - "exclude": ["node_modules"] + "compilerOptions": { + "target": "ES2020", + "lib": ["ES2020", "DOM", "DOM.Iterable"], + "module": "ESNext", + "moduleResolution": "bundler", + "strict": true, + "jsx": "react-jsx", + "skipLibCheck": true, + "paths": { "@/*": ["./src/*"] }, + "baseUrl": ".", + "noEmit": true, + "esModuleInterop": true, + "isolatedModules": true, + "resolveJsonModule": true, + "types": ["vite/client"] + }, + "include": ["src"] } diff --git a/webapp/vite.config.ts b/webapp/vite.config.ts new file mode 100644 index 000000000..47405fc7f --- /dev/null +++ b/webapp/vite.config.ts @@ -0,0 +1,11 @@ +import { defineConfig } from "vite"; +import react from "@vitejs/plugin-react"; +import path from "path"; + +export default defineConfig({ + plugins: [react()], + resolve: { + alias: { "@": path.resolve(__dirname, "./src") }, + }, + server: { port: 3000 }, +});