diff --git a/apps/public-web/ARCHITECTURE.md b/apps/public-web/ARCHITECTURE.md index e17715d..31473bd 100644 --- a/apps/public-web/ARCHITECTURE.md +++ b/apps/public-web/ARCHITECTURE.md @@ -1,40 +1,48 @@ -VOYAGE – Blog Architecture Documentation -======================================== +VOYAGE – Blog & Admin Architecture Documentation +=============================================== Purpose ------- -This document describes the architecture of the VOYAGE public blog, -including folder structure, routing, data flow, and rendering logic. +This document describes the architecture of the VOYAGE public blog +and its internal admin system. -A new developer should be able to understand how blog posts are loaded, -render routes work, and where to extend the system after reading this file. +It covers folder structure, routing, data flow, authentication, +and the admin UI shell. + +A new developer should be able to understand: +- how blog posts are loaded and rendered +- how routing works (public + admin) +- how authentication is handled +- where and how to extend the system High-Level Overview ------------------- -The blog is implemented using the **Next.js App Router** with: +The VOYAGE public site is implemented using the **Next.js App Router** +inside a monorepo, with authentication delegated to a **Spring Boot backend**. -- File-based routing -- Dynamic route segments -- MDX-based content -- A shared public layout +Key technologies: +- Next.js (App Router, Server Components) +- Spring Boot (session-based auth) +- Tailwind CSS (UI styling) +- File-based content (TXT / MD / MDX) -Key principles: -- Blog content lives outside the app router (content/posts) -- Routing is derived from folder structure -- Rendering happens in async server components -- Shared UI is colocated in components/ +Core principles: +- Public content is always accessible +- Admin tools are protected via backend session +- No duplicate authentication logic +- Clear separation between public site and admin system Monorepo Context ---------------- This repository is a monorepo containing multiple applications. -Relevant app for the blog: -- apps/public-web → Public website & blog +Relevant apps: +- apps/public-web → Public website, blog, and admin UI +- apps/workspace-api → Spring Boot backend (auth, API, DB) Other apps (not covered here): -- workspace-api - workspace-ui @@ -45,32 +53,36 @@ apps/public-web/ │ ├── app/ │ ├── (site)/ → Public site route group -│ │ ├── layout.tsx → Shared layout (TopBar, globals) +│ │ ├── layout.tsx → Shared public layout (TopBar, globals) │ │ ├── page.tsx → Homepage │ │ ├── about/ │ │ │ └── page.tsx → Static About page │ │ ├── blog/ -│ │ │ ├── layout.tsx → Blog-specific layout (optional) +│ │ │ ├── layout.tsx → Blog-specific layout │ │ │ ├── page.tsx → Blog index (list of posts) │ │ │ └── [slug]/ │ │ │ └── page.tsx → Dynamic blog post page │ │ └── admin/ -│ │ └── page.tsx → Admin-only page (protected) +│ │ ├── layout.tsx → Admin auth guard (server-side) +│ │ ├── page.tsx → Admin dashboard (sidebar + header) +│ │ └── posts/ +│ │ └── page.tsx → Admin posts manager │ │ -│ ├── global.css → Global styles -│ └── layout.tsx → Root layout +│ ├── global.css → Tailwind + global styles +│ └── layout.tsx → Root layout (imports global.css) │ ├── components/ │ └── shell/ -│ └── TopBar.tsx → Shared navigation bar +│ └── TopBar.tsx → Shared public navigation bar │ ├── visuals/ │ └── ImageSphereSketch.tsx → Creative / visual components │ ├── content/ │ └── posts/ -│ ├── YYYY-MM-DD-title.mdx → Blog post source files -│ └── ... +│ ├── YY-MM-DD-title.txt → Blog post source files (current) +│ ├── YYYY-MM-DD-title.md → (optional / supported) +│ └── YYYY-MM-DD-title.mdx → (optional / supported) │ ├── lib/ │ └── posts.ts → Blog data loading & parsing logic @@ -87,125 +99,102 @@ Routing Logic ------------- Routing is defined entirely by the folder structure. -Static routes: +Public routes: - / → app/(site)/page.tsx - /about → app/(site)/about/page.tsx - /blog → app/(site)/blog/page.tsx - -Dynamic routes: - /blog/[slug] → app/(site)/blog/[slug]/page.tsx -Admin route: -- /admin → app/(site)/admin/page.tsx (protected) +Admin routes: +- /admin → Admin dashboard (protected) +- /admin/posts → Admin post manager (protected) -The `[slug]` directory defines a dynamic route parameter -that is passed to the page component as `params.slug`. +Dynamic route parameters: +- `[slug]` is derived from the filename in `content/posts` -Data Flow (How a Blog Post Is Rendered) ---------------------------------------- +Blog Data Flow (Rendering a Post) +--------------------------------- 1. User navigates to: /blog/some-post-slug -2. Next.js resolves the route: +2. Next.js resolves: app/(site)/blog/[slug]/page.tsx -3. The page component receives: +3. Page receives: params.slug 4. The page calls: getPostBySlug(slug) from lib/posts.ts 5. lib/posts.ts: - - Reads the corresponding MDX file from content/posts - - Parses frontmatter metadata (title, date, etc.) + - Reads a file from content/posts + - Supports .txt, .md, .mdx + - Strips date prefix from filename - Returns structured post data -6. The page component renders: - - Shared layout (TopBar) +6. Page renders: + - Shared public layout - Post metadata - - MDX content as React components + - Parsed content 7. If the slug does not exist: - notFound() is triggered -Core Files Explained -------------------- +Admin System Overview +--------------------- -app/(site)/blog/[slug]/page.tsx --------------------------------- -- Async Server Component -- Receives dynamic params (slug) -- Loads post data via lib/posts -- Handles invalid slugs with notFound() -- Renders full blog post view +The admin system is an **internal tool**, not a CMS yet. -app/(site)/blog/page.tsx ------------------------- -- Blog index page -- Loads all posts via lib/posts -- Renders list / preview of posts +Current capabilities: +- Admin dashboard UI +- Post listing / detection +- Backend session verification +- Logout handling -lib/posts.ts -------------- -- Central data access layer for blog content -- Handles file system access and MDX parsing -- Keeps routing and rendering logic clean - -content/posts/*.mdx -------------------- -- Source of truth for blog content -- File name defines the slug -- Frontmatter stores metadata -- Body is rendered as MDX - -components/shell/TopBar.tsx ---------------------------- -- Shared navigation component -- Used across all public pages -- Ensures consistent layout and navigation - -public/blog/visuals/ --------------------- -- Static images for blog posts -- Served directly by Next.js -- Referenced in MDX or page components +Admin UI principles: +- Clean, minimal, internal-tool aesthetic +- Sidebar + large system header +- No duplicated navigation actions +- Tailwind-based styling Admin Authentication & Security -------------------------------- -The blog includes an **admin-only section** used for internal tools -(e.g. editor preview, drafts, future CMS features). +Authentication is **not handled by Next.js**. -Authentication is **not handled by Next.js**, but delegated to the -existing Spring Boot backend (`workspace-api`). +Instead, it is delegated to the Spring Boot backend (`workspace-api`). Key principles: -- Public blog remains fully accessible without login -- Admin routes require a valid backend session +- Single source of truth for auth (Spring) - Session-based authentication (JSESSIONID) -- No JWT, no duplicate auth system +- No JWT +- No duplicate auth logic in frontend +- Admin routes require a valid backend session Admin Route Protection (Frontend) --------------------------------- -Route: -- /admin → app/(site)/admin/page.tsx +Protection is implemented in: -Protection strategy: -- Implemented as an **async Server Component guard** -- On each request: +app/(site)/admin/layout.tsx + +Strategy: +- Admin layout is an async Server Component +- On every request: - Calls backend endpoint `/api/me` - - Forwards cookies manually + - Forwards incoming cookies manually - If response is 401 → redirect to /login - If response is 200 → render admin UI + - Optional role check (ROLE_ADMIN) -Important detail: -- Server-side fetch must forward cookies explicitly +Important technical detail: +- Server-side fetch MUST forward cookies explicitly +- Using headers().get("cookie") (async) - credentials: "include" is NOT sufficient in Server Components @@ -215,30 +204,30 @@ Login Flow (End-to-End) 1. User navigates to: http://localhost:3000/admin -2. Admin page fetches: +2. Admin layout fetches: GET http://localhost:8080/api/me 3. If not authenticated: - Backend returns 401 - - Next.js redirects to /login (frontend route) + - Next.js redirects to /login (frontend) -4. Frontend /login page redirects to backend: +4. Frontend /login redirects to backend: GET http://localhost:8080/login-redirect?redirect=http://localhost:3000/admin 5. Backend: - Stores redirect target in session - - Redirects to /login (without query params) + - Redirects to /login (no query params) 6. Spring Security default login page is shown 7. User submits credentials 8. On successful login: - - Custom successHandler reads redirect from session - - User is redirected to: + - Custom success handler reads redirect from session + - Redirects user back to: http://localhost:3000/admin -9. Admin page loads successfully +9. Admin UI loads successfully Backend Endpoints Involved @@ -248,7 +237,7 @@ Backend Endpoints Involved - Returns current authenticated user - 200 → logged in - 401 → not authenticated -- Never redirects (API-safe) +- Never redirects /login - Spring Security default login page @@ -259,23 +248,43 @@ Backend Endpoints Involved - Stores redirect target in session - Avoids unsupported query params on /login +/logout +- Invalidates session +- Clears JSESSIONID cookie + + +Admin Posts Manager +------------------- + +Route: +- /admin/posts + +Responsibilities: +- Reads files from content/posts on the server +- Detects available posts (.txt / .md / .mdx) +- Derives slugs from filenames +- Displays debug info (detected paths, files) +- Links to public blog pages + +This is intentionally read-only for now. + Why This Architecture --------------------- -- Single source of truth for authentication (Spring) -- No duplicate auth logic in frontend -- Clean separation: - - Public content → no auth - - Admin tools → backend session -- Works with SSR and Server Components -- Production-ready pattern for multi-app monorepos +- Clear separation of concerns +- Public content stays simple and fast +- Admin tools are protected and internal +- No auth duplication +- SSR-safe and production-ready +- Scales cleanly to future CMS features How to Extend (Future) ---------------------- -- Admin editor UI +- Admin post editor UI - Draft / preview mode -- Role-based admin features +- Role-based admin tools +- Publishing workflow - CMS integration - Protected preview links @@ -283,8 +292,9 @@ How to Extend (Future) Current Status -------------- - App Router fully set up -- Dynamic blog slugs working -- Shared public layout integrated -- MDX content loading stable +- Public blog routing stable +- Content loading via filesystem stable +- Tailwind styling active - Admin auth flow stable and tested -- Ready for styling, animations, and feature extensions \ No newline at end of file +- Admin dashboard + posts manager implemented +- Ready for further UX polish and feature extensions \ No newline at end of file diff --git a/apps/public-web/app/(site)/admin/LogoutButton.tsx b/apps/public-web/app/(site)/admin/LogoutButton.tsx new file mode 100644 index 0000000..8718a62 --- /dev/null +++ b/apps/public-web/app/(site)/admin/LogoutButton.tsx @@ -0,0 +1,11 @@ +"use client"; + +export default function LogoutButton() { + const backend = process.env.NEXT_PUBLIC_BACKEND_URL ?? "http://localhost:8080"; + + return ( +
+ +
+ ); +} \ No newline at end of file diff --git a/apps/public-web/app/(site)/admin/layout.tsx b/apps/public-web/app/(site)/admin/layout.tsx new file mode 100644 index 0000000..fbaaee7 --- /dev/null +++ b/apps/public-web/app/(site)/admin/layout.tsx @@ -0,0 +1,31 @@ +import { redirect } from "next/navigation"; +import { headers } from "next/headers"; + +async function cookieHeader() { + const h = await headers(); // ✅ await + return h.get("cookie") ?? ""; +} + +export default async function AdminLayout({ + children, + }: { + children: React.ReactNode; +}) { + const res = await fetch(`${process.env.BACKEND_URL}/api/me`, { + headers: { cookie: await cookieHeader() }, // ✅ await here too + cache: "no-store", + }); + + if (res.status === 401) redirect("/login"); + if (!res.ok) redirect("/login"); + + const me = await res.json(); + + const isAdmin = + Array.isArray(me?.authorities) && + me.authorities.some((a: any) => a.authority === "ROLE_ADMIN"); + + if (!isAdmin) redirect("/"); + + return <>{children}; +} \ No newline at end of file diff --git a/apps/public-web/app/(site)/admin/page.tsx b/apps/public-web/app/(site)/admin/page.tsx index cc9485e..47f9c2b 100644 --- a/apps/public-web/app/(site)/admin/page.tsx +++ b/apps/public-web/app/(site)/admin/page.tsx @@ -1,38 +1,97 @@ -import { redirect } from "next/navigation"; -import { cookies } from "next/headers"; +import Link from "next/link"; -export default async function AdminPage() { - const cookieHeader = (await cookies()) - .getAll() - .map(({ name, value }) => `${name}=${value}`) - .join("; "); - - const res = await fetch(`${process.env.BACKEND_URL}/api/me`, { - headers: { cookie: cookieHeader }, - cache: "no-store", - redirect: "manual", // IMPORTANT: don’t follow to /login - }); - - // Spring might answer 302 to /login when unauthenticated - if (res.status === 401 || res.status === 302) redirect("/login"); - - if (!res.ok) { - const text = await res.text(); - throw new Error(`Failed to load session: ${res.status} ${text.slice(0, 200)}`); - } - - const contentType = res.headers.get("content-type") ?? ""; - if (!contentType.includes("application/json")) { - // we got HTML (login page) or something else - redirect("/login"); - } - - const me = await res.json(); +export default function AdminPage() { + const backend = process.env.BACKEND_URL ?? "http://localhost:8080"; return ( -
-

Admin

-
{JSON.stringify(me, null, 2)}
-
+
+ {/* 1. SINGLE CLEAN SIDEBAR */} + + + {/* 2. MAIN CONTENT AREA */} +
{/* FAT VOYAGE HEADER */} + +
+ +

+ VOYAGE OFFICE +

+ +
+ +

+ System Connected: {backend} +

+
+
+
+ + {/* 3. SYMMETRICAL DASHBOARD CONTENT */} +
+ +
+ {/* Main Action Card */} +
+
+

Content

+

Post Management

+

+ Review and edit all current voyage publications. +

+
+ + Manage Posts → + +
+ + {/* Database Status Card */} +
+
+

Infrastructure

+

SQL Database

+

+ Status: Operational at {backend}/h2-console +

+
+ + Open Console ↗ + +
+
+ + {/* Large Summary Box */} +
+

Office Overview

+
+
+

+ "Efficiency is doing things right; effectiveness is doing the right things." +

+
+
+

100%

+

System Uptime

+
+
+
+ +
+
+
); } \ No newline at end of file diff --git a/apps/public-web/app/(site)/admin/posts/page.tsx b/apps/public-web/app/(site)/admin/posts/page.tsx new file mode 100644 index 0000000..e98c8cd --- /dev/null +++ b/apps/public-web/app/(site)/admin/posts/page.tsx @@ -0,0 +1,123 @@ +import fs from "node:fs"; +import path from "node:path"; +import Link from "next/link"; + +type Post = { + file: string; + slug: string; +}; + +type PostsResult = { + postsDir: string; + tried: string[]; + allFiles: string[]; + posts: Post[]; +}; + +function resolvePostsDir(): { postsDir: string; tried: string[] } { + const tried: string[] = []; + + // 1) When running `npm run dev` inside apps/public-web + const candidate1 = path.join(process.cwd(), "content", "posts"); + tried.push(candidate1); + if (fs.existsSync(candidate1)) return { postsDir: candidate1, tried }; + + // 2) When running Next from monorepo root (process.cwd() == repo root) + const candidate2 = path.join(process.cwd(), "apps", "public-web", "content", "posts"); + tried.push(candidate2); + if (fs.existsSync(candidate2)) return { postsDir: candidate2, tried }; + + // 3) Fallback: try relative to this file (best-effort) + const candidate3 = path.join(process.cwd(), "..", "content", "posts"); + tried.push(candidate3); + if (fs.existsSync(candidate3)) return { postsDir: candidate3, tried }; + + return { postsDir: candidate1, tried }; +} + +function getPosts(): PostsResult { + const { postsDir, tried } = resolvePostsDir(); + + if (!fs.existsSync(postsDir)) { + return { postsDir, tried, allFiles: [], posts: [] }; + } + + const allFiles = fs.readdirSync(postsDir); + + const supported = allFiles.filter((f) => + f.toLowerCase().endsWith(".txt") || + f.toLowerCase().endsWith(".md") || + f.toLowerCase().endsWith(".mdx") + ); + + const posts = supported.map((file) => { + const base = file.replace(/\.(txt|md|mdx)$/i, ""); + + // Strip date prefix (YY-MM-DD- or YYYY-MM-DD-) + const slug = base + .replace(/^\d{2}-\d{2}-\d{2}-/, "") + .replace(/^\d{4}-\d{2}-\d{2}-/, ""); + + return { file, slug }; + }); + + return { postsDir, tried, allFiles, posts }; +} + +export default function AdminPostsPage() { + const { postsDir, tried, allFiles, posts } = getPosts(); + + return ( +
+

Posts

+ +

+ ← Back to Admin +

+ +
+
+ postsDir: {postsDir} +
+
+ files in folder: {allFiles.length} +
+
+ posts detected: {posts.length} +
+ +
+ Debug: paths tried +
{tried.join("\n")}
+
+ + {allFiles.length > 0 && ( +
+ Debug: files seen +
{allFiles.join("\n")}
+
+ )} +
+ + {posts.length === 0 ? ( +

No posts found.

+ ) : ( + + )} +
+ ); +} \ No newline at end of file diff --git a/apps/public-web/node_modules/.package-lock.json b/apps/public-web/node_modules/.package-lock.json index 7da6794..cf1b1c2 100644 --- a/apps/public-web/node_modules/.package-lock.json +++ b/apps/public-web/node_modules/.package-lock.json @@ -81,6 +81,29 @@ "@img/sharp-libvips-darwin-arm64": "1.2.4" } }, + "node_modules/@img/sharp-darwin-x64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-darwin-x64/-/sharp-darwin-x64-0.34.5.tgz", + "integrity": "sha512-YNEFAF/4KQ/PeW0N+r+aVVsoIY0/qxxikF2SWdp+NRkmMB7y9LBZAVqQ4yhGCm/H3H270OSykqmQMKLBhBJDEw==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-darwin-x64": "1.2.4" + } + }, "node_modules/@img/sharp-libvips-darwin-arm64": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-arm64/-/sharp-libvips-darwin-arm64-1.2.4.tgz", @@ -97,6 +120,423 @@ "url": "https://opencollective.com/libvips" } }, + "node_modules/@img/sharp-libvips-darwin-x64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-x64/-/sharp-libvips-darwin-x64-1.2.4.tgz", + "integrity": "sha512-1IOd5xfVhlGwX+zXv2N93k0yMONvUlANylbJw1eTah8K/Jtpi15KC+WSiaX/nBmbm2HxRM1gZ0nSdjSsrZbGKg==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "darwin" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-arm": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm/-/sharp-libvips-linux-arm-1.2.4.tgz", + "integrity": "sha512-bFI7xcKFELdiNCVov8e44Ia4u2byA+l3XtsAj+Q8tfCwO6BQ8iDojYdvoPMqsKDkuoOo+X6HZA0s0q11ANMQ8A==", + "cpu": [ + "arm" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-arm64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm64/-/sharp-libvips-linux-arm64-1.2.4.tgz", + "integrity": "sha512-excjX8DfsIcJ10x1Kzr4RcWe1edC9PquDRRPx3YVCvQv+U5p7Yin2s32ftzikXojb1PIFc/9Mt28/y+iRklkrw==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-ppc64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-ppc64/-/sharp-libvips-linux-ppc64-1.2.4.tgz", + "integrity": "sha512-FMuvGijLDYG6lW+b/UvyilUWu5Ayu+3r2d1S8notiGCIyYU/76eig1UfMmkZ7vwgOrzKzlQbFSuQfgm7GYUPpA==", + "cpu": [ + "ppc64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-riscv64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-riscv64/-/sharp-libvips-linux-riscv64-1.2.4.tgz", + "integrity": "sha512-oVDbcR4zUC0ce82teubSm+x6ETixtKZBh/qbREIOcI3cULzDyb18Sr/Wcyx7NRQeQzOiHTNbZFF1UwPS2scyGA==", + "cpu": [ + "riscv64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-s390x": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-s390x/-/sharp-libvips-linux-s390x-1.2.4.tgz", + "integrity": "sha512-qmp9VrzgPgMoGZyPvrQHqk02uyjA0/QrTO26Tqk6l4ZV0MPWIW6LTkqOIov+J1yEu7MbFQaDpwdwJKhbJvuRxQ==", + "cpu": [ + "s390x" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linux-x64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-x64/-/sharp-libvips-linux-x64-1.2.4.tgz", + "integrity": "sha512-tJxiiLsmHc9Ax1bz3oaOYBURTXGIRDODBqhveVHonrHJ9/+k89qbLl0bcJns+e4t4rvaNBxaEZsFtSfAdquPrw==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linuxmusl-arm64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-arm64/-/sharp-libvips-linuxmusl-arm64-1.2.4.tgz", + "integrity": "sha512-FVQHuwx1IIuNow9QAbYUzJ+En8KcVm9Lk5+uGUQJHaZmMECZmOlix9HnH7n1TRkXMS0pGxIJokIVB9SuqZGGXw==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-libvips-linuxmusl-x64": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-x64/-/sharp-libvips-linuxmusl-x64-1.2.4.tgz", + "integrity": "sha512-+LpyBk7L44ZIXwz/VYfglaX/okxezESc6UxDSoyo2Ks6Jxc4Y7sGjpgU9s4PMgqgjj1gZCylTieNamqA1MF7Dg==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "LGPL-3.0-or-later", + "optional": true, + "os": [ + "linux" + ], + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-linux-arm": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm/-/sharp-linux-arm-0.34.5.tgz", + "integrity": "sha512-9dLqsvwtg1uuXBGZKsxem9595+ujv0sJ6Vi8wcTANSFpwV/GONat5eCkzQo/1O6zRIkh0m/8+5BjrRr7jDUSZw==", + "cpu": [ + "arm" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-arm": "1.2.4" + } + }, + "node_modules/@img/sharp-linux-arm64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm64/-/sharp-linux-arm64-0.34.5.tgz", + "integrity": "sha512-bKQzaJRY/bkPOXyKx5EVup7qkaojECG6NLYswgktOZjaXecSAeCWiZwwiFf3/Y+O1HrauiE3FVsGxFg8c24rZg==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-arm64": "1.2.4" + } + }, + "node_modules/@img/sharp-linux-ppc64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-ppc64/-/sharp-linux-ppc64-0.34.5.tgz", + "integrity": "sha512-7zznwNaqW6YtsfrGGDA6BRkISKAAE1Jo0QdpNYXNMHu2+0dTrPflTLNkpc8l7MUP5M16ZJcUvysVWWrMefZquA==", + "cpu": [ + "ppc64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-ppc64": "1.2.4" + } + }, + "node_modules/@img/sharp-linux-riscv64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-riscv64/-/sharp-linux-riscv64-0.34.5.tgz", + "integrity": "sha512-51gJuLPTKa7piYPaVs8GmByo7/U7/7TZOq+cnXJIHZKavIRHAP77e3N2HEl3dgiqdD/w0yUfiJnII77PuDDFdw==", + "cpu": [ + "riscv64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-riscv64": "1.2.4" + } + }, + "node_modules/@img/sharp-linux-s390x": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-s390x/-/sharp-linux-s390x-0.34.5.tgz", + "integrity": "sha512-nQtCk0PdKfho3eC5MrbQoigJ2gd1CgddUMkabUj+rBevs8tZ2cULOx46E7oyX+04WGfABgIwmMC0VqieTiR4jg==", + "cpu": [ + "s390x" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-s390x": "1.2.4" + } + }, + "node_modules/@img/sharp-linux-x64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linux-x64/-/sharp-linux-x64-0.34.5.tgz", + "integrity": "sha512-MEzd8HPKxVxVenwAa+JRPwEC7QFjoPWuS5NZnBt6B3pu7EG2Ge0id1oLHZpPJdn3OQK+BQDiw9zStiHBTJQQQQ==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linux-x64": "1.2.4" + } + }, + "node_modules/@img/sharp-linuxmusl-arm64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-arm64/-/sharp-linuxmusl-arm64-0.34.5.tgz", + "integrity": "sha512-fprJR6GtRsMt6Kyfq44IsChVZeGN97gTD331weR1ex1c1rypDEABN6Tm2xa1wE6lYb5DdEnk03NZPqA7Id21yg==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linuxmusl-arm64": "1.2.4" + } + }, + "node_modules/@img/sharp-linuxmusl-x64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-x64/-/sharp-linuxmusl-x64-0.34.5.tgz", + "integrity": "sha512-Jg8wNT1MUzIvhBFxViqrEhWDGzqymo3sV7z7ZsaWbZNDLXRJZoRGrjulp60YYtV4wfY8VIKcWidjojlLcWrd8Q==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "Apache-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + }, + "optionalDependencies": { + "@img/sharp-libvips-linuxmusl-x64": "1.2.4" + } + }, + "node_modules/@img/sharp-wasm32": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-wasm32/-/sharp-wasm32-0.34.5.tgz", + "integrity": "sha512-OdWTEiVkY2PHwqkbBI8frFxQQFekHaSSkUIJkwzclWZe64O1X4UlUjqqqLaPbUpMOQk6FBu/HtlGXNblIs0huw==", + "cpu": [ + "wasm32" + ], + "ideallyInert": true, + "license": "Apache-2.0 AND LGPL-3.0-or-later AND MIT", + "optional": true, + "dependencies": { + "@emnapi/runtime": "^1.7.0" + }, + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-win32-arm64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-win32-arm64/-/sharp-win32-arm64-0.34.5.tgz", + "integrity": "sha512-WQ3AgWCWYSb2yt+IG8mnC6Jdk9Whs7O0gxphblsLvdhSpSTtmu69ZG1Gkb6NuvxsNACwiPV6cNSZNzt0KPsw7g==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "Apache-2.0 AND LGPL-3.0-or-later", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-win32-ia32": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-win32-ia32/-/sharp-win32-ia32-0.34.5.tgz", + "integrity": "sha512-FV9m/7NmeCmSHDD5j4+4pNI8Cp3aW+JvLoXcTUo0IqyjSfAZJ8dIUmijx1qaJsIiU+Hosw6xM5KijAWRJCSgNg==", + "cpu": [ + "ia32" + ], + "ideallyInert": true, + "license": "Apache-2.0 AND LGPL-3.0-or-later", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + } + }, + "node_modules/@img/sharp-win32-x64": { + "version": "0.34.5", + "resolved": "https://registry.npmjs.org/@img/sharp-win32-x64/-/sharp-win32-x64-0.34.5.tgz", + "integrity": "sha512-+29YMsqY2/9eFEiW93eqWnuLcWcufowXewwSNIT6UwZdUUCrM3oFjMWH/Z6/TMmb4hlFenmfAVbpWeup2jryCw==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "Apache-2.0 AND LGPL-3.0-or-later", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^18.17.0 || ^20.3.0 || >=21.0.0" + }, + "funding": { + "url": "https://opencollective.com/libvips" + } + }, "node_modules/@japont/unicode-range": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/@japont/unicode-range/-/unicode-range-1.0.0.tgz", @@ -175,6 +615,125 @@ "node": ">= 10" } }, + "node_modules/@next/swc-darwin-x64": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-16.1.4.tgz", + "integrity": "sha512-AKC/qVjUGUQDSPI6gESTx0xOnOPQ5gttogNS3o6bA83yiaSZJek0Am5yXy82F1KcZCx3DdOwdGPZpQCluonuxg==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-arm64-gnu": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-16.1.4.tgz", + "integrity": "sha512-POQ65+pnYOkZNdngWfMEt7r53bzWiKkVNbjpmCt1Zb3V6lxJNXSsjwRuTQ8P/kguxDC8LRkqaL3vvsFrce4dMQ==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-arm64-musl": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-16.1.4.tgz", + "integrity": "sha512-3Wm0zGYVCs6qDFAiSSDL+Z+r46EdtCv/2l+UlIdMbAq9hPJBvGu/rZOeuvCaIUjbArkmXac8HnTyQPJFzFWA0Q==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-x64-gnu": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-16.1.4.tgz", + "integrity": "sha512-lWAYAezFinaJiD5Gv8HDidtsZdT3CDaCeqoPoJjeB57OqzvMajpIhlZFce5sCAH6VuX4mdkxCRqecCJFwfm2nQ==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-linux-x64-musl": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-16.1.4.tgz", + "integrity": "sha512-fHaIpT7x4gA6VQbdEpYUXRGyge/YbRrkG6DXM60XiBqDM2g2NcrsQaIuj375egnGFkJow4RHacgBOEsHfGbiUw==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-win32-arm64-msvc": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-16.1.4.tgz", + "integrity": "sha512-MCrXxrTSE7jPN1NyXJr39E+aNFBrQZtO154LoCz7n99FuKqJDekgxipoodLNWdQP7/DZ5tKMc/efybx1l159hw==", + "cpu": [ + "arm64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@next/swc-win32-x64-msvc": { + "version": "16.1.4", + "resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-16.1.4.tgz", + "integrity": "sha512-JSVlm9MDhmTXw/sO2PE/MRj+G6XOSMZB+BcZ0a7d6KwVFZVpkHcb2okyoYFBaco6LeiL53BBklRlOrDDbOeE5w==", + "cpu": [ + "x64" + ], + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, "node_modules/@swc/helpers": { "version": "0.5.15", "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.15.tgz", @@ -224,6 +783,24 @@ "@tailwindcss/oxide-win32-x64-msvc": "4.1.18" } }, + "node_modules/@tailwindcss/oxide-android-arm64": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-android-arm64/-/oxide-android-arm64-4.1.18.tgz", + "integrity": "sha512-dJHz7+Ugr9U/diKJA0W6N/6/cjI+ZTAoxPf9Iz9BFRF2GzEX8IvXxFIi/dZBloVJX/MZGvRuFA9rqwdiIEZQ0Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 10" + } + }, "node_modules/@tailwindcss/oxide-darwin-arm64": { "version": "4.1.18", "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.1.18.tgz", @@ -241,6 +818,199 @@ "node": ">= 10" } }, + "node_modules/@tailwindcss/oxide-darwin-x64": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-x64/-/oxide-darwin-x64-4.1.18.tgz", + "integrity": "sha512-FL5oxr2xQsFrc3X9o1fjHKBYBMD1QZNyc1Xzw/h5Qu4XnEBi3dZn96HcHm41c/euGV+GRiXFfh2hUCyKi/e+yw==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-freebsd-x64": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-freebsd-x64/-/oxide-freebsd-x64-4.1.18.tgz", + "integrity": "sha512-Fj+RHgu5bDodmV1dM9yAxlfJwkkWvLiRjbhuO2LEtwtlYlBgiAT4x/j5wQr1tC3SANAgD+0YcmWVrj8R9trVMA==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm-gnueabihf": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm-gnueabihf/-/oxide-linux-arm-gnueabihf-4.1.18.tgz", + "integrity": "sha512-Fp+Wzk/Ws4dZn+LV2Nqx3IilnhH51YZoRaYHQsVq3RQvEl+71VGKFpkfHrLM/Li+kt5c0DJe/bHXK1eHgDmdiA==", + "cpu": [ + "arm" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm64-gnu": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-gnu/-/oxide-linux-arm64-gnu-4.1.18.tgz", + "integrity": "sha512-S0n3jboLysNbh55Vrt7pk9wgpyTTPD0fdQeh7wQfMqLPM/Hrxi+dVsLsPrycQjGKEQk85Kgbx+6+QnYNiHalnw==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-linux-arm64-musl": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-musl/-/oxide-linux-arm64-musl-4.1.18.tgz", + "integrity": "sha512-1px92582HkPQlaaCkdRcio71p8bc8i/ap5807tPRDK/uw953cauQBT8c5tVGkOwrHMfc2Yh6UuxaH4vtTjGvHg==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-linux-x64-gnu": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-gnu/-/oxide-linux-x64-gnu-4.1.18.tgz", + "integrity": "sha512-v3gyT0ivkfBLoZGF9LyHmts0Isc8jHZyVcbzio6Wpzifg/+5ZJpDiRiUhDLkcr7f/r38SWNe7ucxmGW3j3Kb/g==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-linux-x64-musl": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-musl/-/oxide-linux-x64-musl-4.1.18.tgz", + "integrity": "sha512-bhJ2y2OQNlcRwwgOAGMY0xTFStt4/wyU6pvI6LSuZpRgKQwxTec0/3Scu91O8ir7qCR3AuepQKLU/kX99FouqQ==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-wasm32-wasi": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-wasm32-wasi/-/oxide-wasm32-wasi-4.1.18.tgz", + "integrity": "sha512-LffYTvPjODiP6PT16oNeUQJzNVyJl1cjIebq/rWWBF+3eDst5JGEFSc5cWxyRCJ0Mxl+KyIkqRxk1XPEs9x8TA==", + "bundleDependencies": [ + "@napi-rs/wasm-runtime", + "@emnapi/core", + "@emnapi/runtime", + "@tybys/wasm-util", + "@emnapi/wasi-threads", + "tslib" + ], + "cpu": [ + "wasm32" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "^1.7.1", + "@emnapi/runtime": "^1.7.1", + "@emnapi/wasi-threads": "^1.1.0", + "@napi-rs/wasm-runtime": "^1.1.0", + "@tybys/wasm-util": "^0.10.1", + "tslib": "^2.4.0" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@tailwindcss/oxide-win32-arm64-msvc": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-arm64-msvc/-/oxide-win32-arm64-msvc-4.1.18.tgz", + "integrity": "sha512-HjSA7mr9HmC8fu6bdsZvZ+dhjyGCLdotjVOgLA2vEqxEBZaQo9YTX4kwgEvPCpRh8o4uWc4J/wEoFzhEmjvPbA==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@tailwindcss/oxide-win32-x64-msvc": { + "version": "4.1.18", + "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-x64-msvc/-/oxide-win32-x64-msvc-4.1.18.tgz", + "integrity": "sha512-bJWbyYpUlqamC8dpR7pfjA0I7vdF6t5VpUGMWRkXVE3AXgIZjYUYAK7II1GNaxR8J1SSrSrppRar8G++JekE3Q==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 10" + } + }, "node_modules/@tailwindcss/postcss": { "version": "4.1.18", "resolved": "https://registry.npmjs.org/@tailwindcss/postcss/-/postcss-4.1.18.tgz", @@ -278,6 +1048,7 @@ "integrity": "sha512-Lpo8kgb/igvMIPeNV2rsYKTgaORYdO1XGVZ4Qz3akwOj0ySGYMPlQWa8BaLn0G63D1aSaAQ5ldR06wCpChQCjA==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "csstype": "^3.2.2" } @@ -428,6 +1199,7 @@ } ], "license": "MIT", + "peer": true, "dependencies": { "baseline-browser-mapping": "^2.9.0", "caniuse-lite": "^1.0.30001759", @@ -736,6 +1508,28 @@ "lightningcss-win32-x64-msvc": "1.30.2" } }, + "node_modules/lightningcss-android-arm64": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.30.2.tgz", + "integrity": "sha512-BH9sEdOCahSgmkVhBLeU7Hc9DWeZ1Eb6wNS6Da8igvUwAe0sqROHddIlvU06q3WyXVEOYDZ6ykBZQnjTbmo4+A==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, "node_modules/lightningcss-darwin-arm64": { "version": "1.30.2", "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.30.2.tgz", @@ -757,6 +1551,204 @@ "url": "https://opencollective.com/parcel" } }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.30.2.tgz", + "integrity": "sha512-oBZgKchomuDYxr7ilwLcyms6BCyLn0z8J0+ZZmfpjwg9fRVZIR5/GMXd7r9RH94iDhld3UmSjBM6nXWM2TfZTQ==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.30.2.tgz", + "integrity": "sha512-c2bH6xTrf4BDpK8MoGG4Bd6zAMZDAXS569UxCAGcA7IKbHNMlhGQ89eRmvpIUGfKWNVdbhSbkQaWhEoMGmGslA==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.30.2.tgz", + "integrity": "sha512-eVdpxh4wYcm0PofJIZVuYuLiqBIakQ9uFZmipf6LF/HRj5Bgm0eb3qL/mr1smyXIS1twwOxNWndd8z0E374hiA==", + "cpu": [ + "arm" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.30.2.tgz", + "integrity": "sha512-UK65WJAbwIJbiBFXpxrbTNArtfuznvxAJw4Q2ZGlU8kPeDIWEX1dg3rn2veBVUylA2Ezg89ktszWbaQnxD/e3A==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.30.2.tgz", + "integrity": "sha512-5Vh9dGeblpTxWHpOx8iauV02popZDsCYMPIgiuw97OJ5uaDsL86cnqSFs5LZkG3ghHoX5isLgWzMs+eD1YzrnA==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.30.2.tgz", + "integrity": "sha512-Cfd46gdmj1vQ+lR6VRTTadNHu6ALuw2pKR9lYq4FnhvgBc4zWY1EtZcAc6EffShbb1MFrIPfLDXD6Xprbnni4w==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.30.2.tgz", + "integrity": "sha512-XJaLUUFXb6/QG2lGIW6aIk6jKdtjtcffUT0NKvIqhSBY3hh9Ch+1LCeH80dR9q9LBjG3ewbDjnumefsLsP6aiA==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.30.2.tgz", + "integrity": "sha512-FZn+vaj7zLv//D/192WFFVA0RgHawIcHqLX9xuWiQt7P0PtdFEVaxgF9rjM/IRYHQXNnk61/H/gb2Ei+kUQ4xQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.30.2", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.30.2.tgz", + "integrity": "sha512-5g1yc73p+iAkid5phb4oVFMB45417DkRevRbt/El/gKXJk4jid+vPFF/AXbxn05Aky8PapwzZrdJShv5C0avjw==", + "cpu": [ + "x64" + ], + "dev": true, + "ideallyInert": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, "node_modules/magic-string": { "version": "0.30.21", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", @@ -962,6 +1954,7 @@ } ], "license": "MIT", + "peer": true, "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", @@ -983,6 +1976,7 @@ "resolved": "https://registry.npmjs.org/react/-/react-19.2.3.tgz", "integrity": "sha512-Ku/hhYbVjOQnXDZFv2+RibmLFGwFdeeKHFcOTlrt7xplBnya5OGn/hIRDsqDiSUcfORsDC7MPxwork8jBwsIWA==", "license": "MIT", + "peer": true, "engines": { "node": ">=0.10.0" } @@ -992,6 +1986,7 @@ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.3.tgz", "integrity": "sha512-yELu4WmLPw5Mr/lmeEpox5rw3RETacE++JgHqQzd2dg+YbJuat3jH4ingc+WPZhxaoFzdv9y33G+F7Nl5O0GBg==", "license": "MIT", + "peer": true, "dependencies": { "scheduler": "^0.27.0" }, diff --git a/apps/public-web/package-lock.json b/apps/public-web/package-lock.json index b42cd42..67f0c48 100644 --- a/apps/public-web/package-lock.json +++ b/apps/public-web/package-lock.json @@ -1031,6 +1031,7 @@ "integrity": "sha512-Lpo8kgb/igvMIPeNV2rsYKTgaORYdO1XGVZ4Qz3akwOj0ySGYMPlQWa8BaLn0G63D1aSaAQ5ldR06wCpChQCjA==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "csstype": "^3.2.2" } @@ -1181,6 +1182,7 @@ } ], "license": "MIT", + "peer": true, "dependencies": { "baseline-browser-mapping": "^2.9.0", "caniuse-lite": "^1.0.30001759", @@ -1925,6 +1927,7 @@ } ], "license": "MIT", + "peer": true, "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", @@ -1946,6 +1949,7 @@ "resolved": "https://registry.npmjs.org/react/-/react-19.2.3.tgz", "integrity": "sha512-Ku/hhYbVjOQnXDZFv2+RibmLFGwFdeeKHFcOTlrt7xplBnya5OGn/hIRDsqDiSUcfORsDC7MPxwork8jBwsIWA==", "license": "MIT", + "peer": true, "engines": { "node": ">=0.10.0" } @@ -1955,6 +1959,7 @@ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.3.tgz", "integrity": "sha512-yELu4WmLPw5Mr/lmeEpox5rw3RETacE++JgHqQzd2dg+YbJuat3jH4ingc+WPZhxaoFzdv9y33G+F7Nl5O0GBg==", "license": "MIT", + "peer": true, "dependencies": { "scheduler": "^0.27.0" }, diff --git a/apps/workspace-api/data/voyage-db.lock.db b/apps/workspace-api/data/voyage-db.lock.db deleted file mode 100644 index 25537e8..0000000 --- a/apps/workspace-api/data/voyage-db.lock.db +++ /dev/null @@ -1,6 +0,0 @@ -#FileLock -#Tue Jan 27 18:25:29 CET 2026 -hostName=macbook-air-von-melika.fritz.box -id=19c007d24ac0d06a483d34679bbb29160d6d33ed895 -method=file -server=192.168.178.68\:58736 diff --git a/apps/workspace-api/data/voyage-db.mv.db b/apps/workspace-api/data/voyage-db.mv.db index 50493b2..dc66f2c 100644 Binary files a/apps/workspace-api/data/voyage-db.mv.db and b/apps/workspace-api/data/voyage-db.mv.db differ