diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..ff9fbc9 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,30 @@ +name: CI + +on: + pull_request: + push: + branches: [main] + +permissions: + contents: read + +jobs: + check: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + with: + # Full history so `lint:changed` can resolve the merge-base with main. + fetch-depth: 0 + + - uses: oven-sh/setup-bun@v2 + with: + bun-version: "1.3" + + - name: Install dependencies + env: + PUPPETEER_SKIP_DOWNLOAD: "true" + run: bun install --frozen-lockfile + + - name: Strict gate (format / lint / typecheck / i18n / knip / react-compiler / security / tests) + run: bun run ci diff --git a/.gitignore b/.gitignore index 2d7433b..281e62b 100644 --- a/.gitignore +++ b/.gitignore @@ -41,12 +41,6 @@ yarn-error.log* /.claude/ /.superpowers/ -# menu bar app build outputs -/menubar/.build/ -/menubar/build/ -/menubar/.swiftpm/ -/menubar/Package.resolved - # vercel .vercel diff --git a/CLAUDE.md b/CLAUDE.md index bcd3918..1e53447 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -72,9 +72,17 @@ Key priorities (in order): ## Testing the app ```bash -npm run dev # starts on 127.0.0.1:3000 +bun dev # starts on 127.0.0.1:3000 ``` +The full CI gate (formatting, typecheck, i18n, knip, react-doctor, tests) is `bun run ci`. GitHub Actions runs the same script on every PR via `.github/workflows/ci.yml`. The five strict checks the project enforces: + +- `bun run format:check` — Biome formatter +- `bun run i18n:check` — `@lingual/i18n-check` for missing / orphan i18n keys (next-intl-recommended; wrapped in `scripts/check-i18n.mjs` with a baseline ignore list) +- `bun run knip` — dead code (files, deps, unlisted deps) +- `bun run react:doctor` — `react-compiler-healthcheck` +- `bun test` — Bun's built-in Jest-compatible test runner, with `--conditions react-server` so `server-only` resolves as a no-op + For end-to-end testing without real credentials, call the setup API directly: ```typescript diff --git a/README.md b/README.md index ffb142f..24dfc74 100644 --- a/README.md +++ b/README.md @@ -20,6 +20,7 @@ Encrypted. AI-categorized. Yours. [![SQLite](https://img.shields.io/badge/SQLite-WAL-003B57?logo=sqlite&logoColor=white&style=flat-square)](https://sqlite.org/) [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg?style=flat-square)](#license) [![Status: Beta](https://img.shields.io/badge/Status-Beta-blueviolet?style=flat-square)](#features) +[![CI](https://github.com/alon710/Spent/actions/workflows/ci.yml/badge.svg)](https://github.com/alon710/Spent/actions/workflows/ci.yml) @@ -36,7 +37,7 @@ Encrypted. AI-categorized. Yours. Israeli banks have terrible exports, YNAB doesn't speak ILS gracefully, and every "cloud finance" app wants you to hand over your bank password. Spent is the answer for people who'd rather just run something on their own laptop. -Your transactions get pulled directly from your bank with [`israeli-bank-scrapers`](https://github.com/eshaham/israeli-bank-scrapers), stored in a local SQLite file you can `cp` and back up like any other file, and categorized by an AI provider you choose: paid Claude, free local Ollama, or nothing at all. +Your transactions get pulled directly from your bank with [`israeli-bank-scrapers`](https://github.com/eshaham/israeli-bank-scrapers), stored in a local SQLite file you can `cp` and back up like any other file, and categorized by an AI provider you choose: paid Claude, Gemini, free local Ollama, or nothing at all. The trade-off is honest: you self-host, you trust the scraper, and you accept that banks may not love automation. In return you get a fast, beautiful, fully offline dashboard that never phones home. @@ -47,13 +48,13 @@ The trade-off is honest: you self-host, you trust the scraper, and you accept th ### 🏦 Israeli bank integration -Isracard, Bank Hapoalim, and Max work out of the box. Visa Cal and Bank Leumi are on the roadmap. +18 banks and card issuers ship enabled out of the box, from Isracard, Hapoalim, Leumi, Mizrahi, and Cal to One Zero (with programmatic SMS 2FA). ### 🤖 AI categorization -Choose Claude (Anthropic) for best accuracy, Ollama for fully local LLMs, or skip and categorize manually. +Choose Claude (Anthropic) for best accuracy, Gemini (Google) for a generous free tier, Ollama for fully local LLMs, or skip and categorize manually. @@ -78,8 +79,11 @@ Polished buttercream-and-sage palette in light mode, warm charcoal in dark. Syst -### 🍎 Menu bar / tray app -Native companion in your menu bar (macOS) or notification area (Windows). Status indicator, one-click open dashboard, sync, and start/stop/restart the service. +### 🔒 Runs entirely on your machine +No cloud, no account, no telemetry. The server binds to loopback only and your data never leaves `data/`. + +### 💬 Chat with your spending +A built-in AI chat agent at `/chat` answers questions about your transactions, budgets, and categories. Uses the same provider you picked for categorization. @@ -152,11 +156,11 @@ Toggle between English (default) and עברית from **Settings → Appearance** ```mermaid flowchart LR - Bank["🏦 Israeli bank
(Isracard / Hapoalim / Max)"] + Bank["🏦 Israeli bank
(Isracard, Hapoalim, Leumi,
Mizrahi, Cal, One Zero, …)"] Scraper["Puppeteer scraper
(israeli-bank-scrapers)"] DB[("📦 SQLite
data/spent.db
(WAL mode)")] - AI{"🤖 AI provider
Claude · Ollama · None"} - UI["🖥 Dashboard
http://spent.localhost:41234"] + AI{"🤖 AI provider
Claude · Gemini · Ollama · None"} + UI["🖥 Dashboard
http://127.0.0.1:2412"] Bank -->|HTTPS
credentials encrypted| Scraper Scraper -->|new transactions| DB @@ -171,67 +175,77 @@ flowchart LR end ``` -Everything inside the dashed box stays on your laptop. The only outbound traffic is to your bank (for scraping) and optionally `api.anthropic.com` (if you chose Claude) or `localhost:11434` (if you chose Ollama). +Everything inside the dashed box stays on your laptop. The only outbound traffic is to your bank (for scraping) and optionally `api.anthropic.com` (if you chose Claude), Google Gemini API endpoints (if you chose Gemini), or `localhost:11434` (if you chose Ollama). ## Supported banks -| Bank | Type | Status | +| Bank | Type | Notes | |---|---|---| -| **Isracard** | Credit card | ✅ Supported | -| **Bank Hapoalim** (incl. Poalim wallets) | Bank | ✅ Supported | -| **Max** (formerly Leumi Card) | Credit card | ✅ Supported | -| Visa Cal | Credit card | 🚧 Planned | -| Bank Leumi | Bank | 🚧 Planned | +| **Isracard** | Credit card | ID + last 6 of card + password | +| **Visa Cal** | Credit card | Username + password | +| **Max** (formerly Leumi Card) | Credit card | Username + password | +| **American Express IL** | Credit card | Isracard-issued; ID + last 6 + password | +| **Bank Hapoalim** (incl. Poalim wallets) | Bank | User code + password | +| **Bank Leumi** | Bank | Username + password | +| **Mizrahi Tefahot** | Bank | Username + password | +| **Bank Discount** | Bank | ID + password + account number | +| **Mercantile Discount** | Bank | ID + password + account number | +| **First International (FIBI / Beinleumi)** | Bank | Username + password | +| **Otsar Hahayal** | Bank | FIBI subsidiary; username + password | +| **Bank Pagi** | Bank | Username + password | +| **Bank Massad** | Bank | Username + password | +| **Bank Yahav** | Bank | Username + ID + password — 6 months history only | +| **Union Bank** | Bank | Merged into Mizrahi-Tefahot; legacy access | +| **Beyahad Bishvilha** | Card | Histadrut benefits; ID + password | +| **Behatsdaa** | Card | Histadrut subsidies; ID + password | +| **One Zero** | Bank | Programmatic SMS 2FA; email + password + phone | + +All of the above are wired through [`israeli-bank-scrapers`](https://github.com/eshaham/israeli-bank-scrapers) and shipped enabled. **One Zero** is the only provider with programmatic 2FA — for the others, disable 2FA on the bank side or use the `showBrowser` manual-2FA fallback. Don't see your bank? Adding a scraper is a small wrapper around `israeli-bank-scrapers` — see [Contributing](#contributing). ## AI providers -| | **Claude** (Anthropic) | **Ollama** (local) | **None** | -|---|---|---|---| -| Cost | ~₪0.004 per sync | Free | Free | -| Accuracy | Best | Good (depends on model) | Manual | -| Network | `api.anthropic.com` | `localhost:11434` | Offline | -| Setup | API key | Install Ollama + pull a model | Nothing | +| | **Claude** (Anthropic) | **Gemini** (Google) | **Ollama** (local) | **None** | +|---|---|---|---|---| +| Cost | ~₪0.004 per sync | Free tier available | Free | Free | +| Accuracy | Best | Strong | Good (depends on model) | Manual | +| Network | `api.anthropic.com` | Google Gemini API | `localhost:11434` | Offline | +| Setup | API key | API key from Google AI Studio + choose a model | Install Ollama + pull a model | Nothing | -Default model when Claude is selected: `claude-haiku-4-5` (cheap, fast, accurate for categorization). For Ollama, `llama3.2:3b` is the recommended default. +Default model when Claude is selected: `claude-haiku-4-5-20251001` (cheap, fast, accurate for categorization). Gemini defaults to `gemini-3.5-flash` and lets you choose from stable text models: `gemini-3.5-flash`, `gemini-3.1-flash-lite`, `gemini-2.5-flash`, `gemini-2.5-flash-lite`, and `gemini-2.5-pro`. For Ollama, `llama3.2:3b` is the recommended default. You can change providers any time from **Settings → AI provider**. Existing categorizations are kept. ## Requirements -- **Node.js 22+** -- **macOS 13+**, **Ubuntu 22+** (with systemd), or **Windows 11** -- **Build tools for the menubar** (only if you want the tray; `npm run setup` will offer to install these for you if they're missing): - - macOS: Xcode Command Line Tools (`xcode-select --install`) - - Windows: .NET 8 SDK (`winget install Microsoft.DotNet.SDK.8`) +- **[Bun 1.3+](https://bun.com)** (used as the package manager and script runner) +- **Node.js 22+** (the runtime for the production server via `next start`) +- **macOS 13+**, **Ubuntu 22+**, or **Windows 11** - A bank account with **2FA disabled** (most Israeli banks require this for automation — OneZero is the exception) ## Install -> Prefer a screenshot-by-screenshot walkthrough? The [step-by-step install guides](https://shaya16.github.io/Spent/getting-started/) on the docs site cover macOS and Windows separately, with build-tool setup and tray-app gotchas spelled out. - ```bash -git clone https://github.com/Shaya16/Spent.git +git clone https://github.com/alon710/Spent.git cd spent -npm install -npm run setup +bun install +bun run build +bun start ``` -`npm run setup` does everything: builds the Next.js app, installs the always-on service (LaunchAgent on macOS / systemd on Linux / Task Scheduler on Windows), builds the platform menubar from source, installs it to the standard location, registers it to auto-start at login, and opens the dashboard. On Windows it also writes a `127.0.0.1 spent.localhost` line to your hosts file (the only step that asks for Administrator). macOS and Linux resolve `*.localhost` natively, so setup runs sudo-free there. +`bun start` runs the production server bound to `127.0.0.1:2412` — loopback only, so it is never reachable from your LAN or the internet. Leave the process running (or wrap it in your own service manager / `tmux` / login item) for an always-on dashboard. -On Linux there is no native menubar. `npm run setup` installs the service and opens the browser. Control the service with `npm run service:*` (see below). +Open **`http://127.0.0.1:2412`** and bookmark it. -First launch of the menubar on macOS/Windows shows an unsigned-binary warning (Gatekeeper / SmartScreen). That's expected: you built it locally and didn't pay for a code-signing certificate. Right-click → Open (macOS) or "More info" → "Run anyway" (Windows). One-time. - -Open **`http://spent.localhost:41234`** and bookmark it. +To hack on the app with hot reload instead, run `bun dev` and open `http://127.0.0.1:3000`. ## First-time setup In the browser: 1. **Connect your bank** — credentials are AES-256-GCM encrypted before they touch disk. -2. **Choose an AI provider** — Claude (default), Ollama, or none. +2. **Choose an AI provider** — Claude (default), Gemini, Ollama, or none. 3. **Set your monthly ceiling** — total spend you want to stay under each month. 4. **Set per-category budgets** — type an amount on any category to budget it; leave blank to track without a limit. 5. **Done.** Sync starts automatically: 3 months of transactions, then AI categorization. @@ -240,49 +254,18 @@ In the browser: | What you want | Run | |---|---| -| Just use the app (no coding) | Open `http://spent.localhost:41234` | -| Code and see changes instantly | `npm run dev` → `http://127.0.0.1:3000` | -| Update the always-on app after editing | `npm run service:reload` | - -Rare cases: - -- Changed the menu bar app source → `npm run menubar:install:mac` (or `:windows`) to rebuild and reinstall. -- Changed install scripts or hostname → `npm run service:uninstall && npm run service:install`. - -## Service commands - -| Command | What it does | -|---|---| -| `npm run service:status` | Running? Bound to loopback? | -| `npm run service:start` / `:stop` | Start/stop now | -| `npm run service:reload` | Rebuild and restart | -| `npm run service:logs` | Tail server logs | -| `npm run service:open` | Open the app in your browser | -| `npm run service:uninstall` | Remove auto-start and hosts entry. `data/` is untouched. | +| Just use the app | `bun start` → `http://127.0.0.1:2412` | +| Code and see changes instantly | `bun dev` → `http://127.0.0.1:3000` | +| Rebuild after editing | `bun run build` then restart `bun start` | +| Run the full CI gate locally before pushing | `bun run ci` | ## Uninstall -```bash -npm run uninstall -``` - -Reverses everything `npm run setup` installed: - -- Stops the background service and removes the LaunchAgent / Task Scheduler entry / systemd unit. -- Windows: removes the `127.0.0.1 spent.localhost` line from your hosts file (asks for Administrator). macOS/Linux don't have a hosts entry to remove unless you're upgrading from an older install — in that case the legacy `spent.local` line is cleaned up automatically. -- Quits the menubar, removes the installed app, and removes it from Login Items / Startup. - -**Kept on purpose:** - -- `data/`: your transactions, budgets, and encryption key. To wipe your data: `rm -rf data/`. -- The repo itself. To remove Spent entirely: `rm -rf data/ && cd .. && rm -rf spent/`. - -If you only want to remove the menubar but keep the always-on web app: - -- **macOS**: `rm -rf ~/Applications/Spent.app` and remove "Spent" from System Settings → General → Login Items. -- **Windows**: delete `%LOCALAPPDATA%\Programs\Spent\` and remove `Spent.lnk` from `shell:startup`. +Spent installs nothing outside the project folder. To remove it: -If you only want to remove the always-on service but keep the menubar (so it's there if you reinstall later): `npm run service:uninstall`. +- Stop the running `bun start` (or `bun dev`) process. +- `rm -rf data/` to wipe your transactions, budgets, and encryption key. +- Delete the repository to remove Spent entirely: `cd .. && rm -rf spent/`. ## Security at a glance @@ -303,9 +286,8 @@ Full threat model and responsible-disclosure policy → [SECURITY.md](SECURITY.m - `data/spent.db` — transactions, categories, budgets, settings - `data/.encryption-key` — 32-byte AES key, mode `0600` -- `~/Library/Logs/Spent/` (macOS) / `~/.local/state/spent/log/` (Linux) — service logs -Back up `data/` like any other folder. To migrate to a new machine, copy `data/` over and run `npm run service:install`. +Back up `data/` like any other folder. To migrate to a new machine, copy `data/` over and start the app there. ## Architecture & code map @@ -322,17 +304,13 @@ spent/ │ │ └── settings/ Per-tab settings panels │ ├── lib/ Shared client-side types and helpers │ └── server/ -│ ├── ai/ Claude + Ollama provider implementations +│ ├── ai/ Claude + Gemini + Ollama provider implementations │ ├── db/ SQLite singleton, migrations, query helpers │ ├── lib/ Encryption, dedup, transfer detection, pace │ └── scrapers/ Wrapper around israeli-bank-scrapers -├── menubar/ Tray companions (built locally by `npm run setup`) -│ ├── mac/ Swift MenuBarExtra app -│ └── windows/ C# WinForms NotifyIcon app -├── scripts/service/ LaunchAgent / systemd / Task Scheduler installer +├── scripts/ Dev utilities + the i18n / changed-file-lint CI helpers ├── website/ Astro + Starlight docs site (auto-deploys to GitHub Pages) -├── .github/workflows/ CI — docs site deploy -├── Spent.sln Visual Studio solution for the Windows menubar project +├── .github/workflows/ CI gate + docs site deploy └── data/ SQLite + encryption key (gitignored) ``` @@ -340,21 +318,20 @@ spent/ > The [Troubleshooting docs](https://shaya16.github.io/Spent/troubleshooting/) cover Defender, Gatekeeper, Cloudflare bot challenges, and bank-specific quirks in more depth. -- **Port 41234 in use** → `lsof -nP -iTCP:41234 -sTCP:LISTEN` (Unix) or `netstat -ano | findstr :41234` (Windows). Kill the offender and re-run install. -- **Gatekeeper blocks `Spent.app`** → right-click → Open → Open. One-time. -- **Linux: "systemd user instance not available"** → `loginctl enable-linger $USER`. -- **Windows: hosts edit fails / `spent.localhost` doesn't resolve** → re-run install from an elevated PowerShell (Win+X → "Terminal (Admin)") so it can edit `C:\Windows\System32\drivers\etc\hosts`. After the edit, the installer flushes the DNS cache automatically; if you edited hosts manually, run `ipconfig /flushdns`. `http://127.0.0.1:41234` always works as a fallback. +- **Port 2412 in use** → `lsof -nP -iTCP:2412 -sTCP:LISTEN` (Unix) or `netstat -ano | findstr :2412` (Windows). Kill the offender and restart `bun start`. - **Bank scrape fails with "Cloudflare"** → temporarily run with `SPENT_DISABLE_CHROMIUM_SANDBOX=1` to let Puppeteer use a real Chrome profile. ## Roadmap - [x] Hebrew UI with full RTL layout -- [ ] Visa Cal scraper -- [ ] Bank Leumi scraper +- [x] Visa Cal, Bank Leumi, Mizrahi, Discount, FIBI, and the rest of the `israeli-bank-scrapers` roster +- [x] One Zero with programmatic SMS 2FA +- [x] Gemini as a third AI provider +- [x] AI chat agent for asking questions about your spending - [ ] CSV / OFX export - [ ] Custom user-defined categories - [ ] Mobile companion (Phase 2) -- [ ] Multiple workspaces in the menu bar / tray app +- [ ] Multiple workspaces ## Contributing @@ -370,6 +347,7 @@ Conventions: - TypeScript strict mode. No `any` without a comment. - Conventional commits: `feat:`, `fix:`, `chore:`, `docs:`, `refactor:`. - Comments only where the "why" isn't obvious. No em dashes in code, commits, or docs. +- Run `bun run ci` before opening a PR — same five checks GitHub Actions enforces strictly: formatter (Biome), TypeScript, i18n keys (next-intl-recommended `@lingual/i18n-check`), dead code (knip), React Compiler healthcheck, and `bun test`. ## License @@ -384,4 +362,4 @@ Built on the shoulders of: - [`shadcn/ui`](https://ui.shadcn.com/) on top of [`base-ui`](https://base-ui.com/) - [`better-sqlite3`](https://github.com/WiseLibs/better-sqlite3) - [`next-intl`](https://next-intl.dev/) for English / Hebrew i18n -- [Anthropic Claude](https://www.anthropic.com/) and the local-LLM crew at [Ollama](https://ollama.com/) +- [Anthropic Claude](https://www.anthropic.com/), [Google Gemini](https://ai.google.dev/), and the local-LLM crew at [Ollama](https://ollama.com/) diff --git a/SECURITY.md b/SECURITY.md index 4c94f0c..14e8c45 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -64,7 +64,7 @@ sandbox on. ## CSRF defense -Next.js middleware (`src/middleware.ts`) rejects any mutating API +Next.js proxy (`src/proxy.ts`) rejects any mutating API request (POST/PUT/PATCH/DELETE) whose `Origin` or `Referer` header doesn't match the app's own host. This prevents a malicious tab in your browser from triggering syncs / category changes against your @@ -140,64 +140,42 @@ defense. key only exposes one credential at a time. - **Audit log** of every API mutation with timestamps. -## Always-on service - -If you install Spent as a background service (`npm run service:install`), -the server runs from login to logout (or from boot to shutdown). This -changes how some surfaces look. The guarantees: - -**The server still binds only to `127.0.0.1`.** The `npm run start` -script hardcodes `-H 127.0.0.1 -p 41234`, and every per-OS template -(LaunchAgent plist, systemd unit, Task Scheduler XML) invokes it with -those flags. The installer runs a post-install check and refuses to -finish if it detects the server listening on a non-loopback address. - -**No new daemon runs as root or SYSTEM.** The installer refuses to run -as root. The LaunchAgent and systemd user unit run under your user. -The Windows scheduled task uses `LeastPrivilege` and runs as the -installing user, not as `SYSTEM`. - -**The hostname is loopback-only.** Spent uses `spent.localhost`, which -RFC 6761 reserves as loopback. macOS and Linux resolve `*.localhost` -to `127.0.0.1` natively through the system resolver, so no hosts file -edit is needed there. On Windows, `npm run service:install` appends -`127.0.0.1 spent.localhost` to the hosts file as a compatibility -fallback (the only step that requires elevation, and it prompts -interactively). The block is bracketed with markers and removed -cleanly by `npm run service:uninstall`. No mDNS / Bonjour service is -ever registered, and the loopback address never resolves from another -device on your network. +## Running it always-on + +Spent does not ship a service installer. You run the server yourself with +`bun start` and, if you want it always-on, wrap that command in your own +service manager (a macOS LaunchAgent, a systemd user unit, a Windows +scheduled task, `tmux`, etc.). Whatever you use, these properties hold: + +**The server binds only to `127.0.0.1`.** The `bun start` script hardcodes +`-H 127.0.0.1 -p 2412`, so the dashboard is reachable from your machine +only — never from your LAN or the internet. Do not change the host flag to +a non-loopback address. Run your service manager under your own user, not +as root / SYSTEM. **The health endpoint discloses minimum information.** `GET /api/health` returns `{ok, version, hasDb}` and nothing else. No transaction counts, no provider names, no setup status. Add to it only if you have thought carefully about what a local cross-app attacker could learn. -**The macOS menu bar app has no network entitlements beyond loopback.** -`Spent.app` ships with an `NSAppTransportSecurity > NSExceptionDomains` -entry whitelisting `127.0.0.1` and nothing else. It cannot reach the -internet even if its code were modified to try, without re-signing the -bundle with a different Info.plist. - -**Logs do not leak credentials.** macOS LaunchAgent stdout/stderr go to -`~/Library/Logs/Spent/{out,err}.log` (directory mode `0700`). Linux -systemd writes to `~/.local/state/spent/log/`. The app itself already -avoids logging credentials (see "What's protected at rest" above); -the always-on service does not change that. +**Logs do not leak credentials.** The app avoids logging credentials (see +"What's protected at rest" above). If you redirect `bun start` output to a +file, store it somewhere only your user can read (e.g. directory mode +`0700`). -**The encryption key file's permissions are now asserted at startup.** +**The encryption key file's permissions are asserted at startup.** `src/server/lib/encryption.ts` reads `data/.encryption-key` and refuses to start if the file mode is not `0600` (POSIX only; Windows relies on the user profile ACL). If you ever `chmod 644` the key file by accident, the server will fail loudly with the fix command. -**What the always-on service does not protect against:** +**What running always-on does not protect against:** - A local attacker who can already run code as your user. They can read - the DB and key file with or without the service running. + the DB and key file with or without the server running. - A malicious browser tab on your machine doing a CSRF against - `127.0.0.1:41234`. The same-origin middleware in - `src/middleware.ts` already blocks this on every mutating request, + `127.0.0.1:2412`. The same-origin proxy in + `src/proxy.ts` already blocks this on every mutating request, and that protection works the same whether the server runs on demand or always-on. diff --git a/Spent.sln b/Spent.sln deleted file mode 100644 index d2c573d..0000000 --- a/Spent.sln +++ /dev/null @@ -1,29 +0,0 @@ -Microsoft Visual Studio Solution File, Format Version 12.00 -# Visual Studio Version 17 -VisualStudioVersion = 17.5.2.0 -MinimumVisualStudioVersion = 10.0.40219.1 -Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "menubar", "menubar", "{6259D98C-762D-27BB-6DE2-96B890C42113}" -EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Spent", "menubar\windows\Spent.csproj", "{CBDF4A33-EA45-F131-781C-71B987CEAE1C}" -EndProject -Global - GlobalSection(SolutionConfigurationPlatforms) = preSolution - Debug|Any CPU = Debug|Any CPU - Release|Any CPU = Release|Any CPU - EndGlobalSection - GlobalSection(ProjectConfigurationPlatforms) = postSolution - {CBDF4A33-EA45-F131-781C-71B987CEAE1C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {CBDF4A33-EA45-F131-781C-71B987CEAE1C}.Debug|Any CPU.Build.0 = Debug|Any CPU - {CBDF4A33-EA45-F131-781C-71B987CEAE1C}.Release|Any CPU.ActiveCfg = Release|Any CPU - {CBDF4A33-EA45-F131-781C-71B987CEAE1C}.Release|Any CPU.Build.0 = Release|Any CPU - EndGlobalSection - GlobalSection(SolutionProperties) = preSolution - HideSolutionNode = FALSE - EndGlobalSection - GlobalSection(NestedProjects) = preSolution - {CBDF4A33-EA45-F131-781C-71B987CEAE1C} = {6259D98C-762D-27BB-6DE2-96B890C42113} - EndGlobalSection - GlobalSection(ExtensibilityGlobals) = postSolution - SolutionGuid = {BA14BD89-C500-45F3-9EFF-A8CA0AEDD4A9} - EndGlobalSection -EndGlobal diff --git a/biome.jsonc b/biome.jsonc new file mode 100644 index 0000000..6ae70e2 --- /dev/null +++ b/biome.jsonc @@ -0,0 +1,59 @@ +{ + "$schema": "https://biomejs.dev/schemas/2.4.15/schema.json", + "vcs": { + "enabled": true, + "clientKind": "git", + "useIgnoreFile": true + }, + "files": { + "ignoreUnknown": true, + "includes": [ + "**", + "!.context", + "!**/.next", + "!**/out", + "!**/build", + "!**/data", + "!**/coverage", + "!website", + "!**/*.lockb", + "!**/bun.lock", + "!src/components/ui" + ] + }, + "formatter": { + "enabled": true, + "indentStyle": "space", + "indentWidth": 2, + "lineWidth": 100 + }, + "linter": { + "enabled": false + }, + "javascript": { + "formatter": { + "quoteStyle": "double", + "trailingCommas": "all", + "semicolons": "always", + "arrowParentheses": "always" + } + }, + "json": { + "formatter": { + "trailingCommas": "none" + } + }, + "css": { + "parser": { + "tailwindDirectives": true + } + }, + "assist": { + "enabled": true, + "actions": { + "source": { + "organizeImports": "on" + } + } + } +} diff --git a/bun.lock b/bun.lock new file mode 100644 index 0000000..e8bd540 --- /dev/null +++ b/bun.lock @@ -0,0 +1,2367 @@ +{ + "lockfileVersion": 1, + "configVersion": 1, + "workspaces": { + "": { + "name": "spent", + "dependencies": { + "@ai-sdk/anthropic": "^3.0.79", + "@ai-sdk/google": "^3.0.80", + "@ai-sdk/react": "^3.0.193", + "@anthropic-ai/sdk": "0.95.2", + "@base-ui/react": "^1.4.1", + "@google/genai": "^2.6.0", + "@tanstack/react-query": "^5.100.10", + "ai": "^6.0.191", + "ai-sdk-ollama": "^3.8.4", + "better-sqlite3": "12.10.0", + "class-variance-authority": "^0.7.1", + "clsx": "^2.1.1", + "drizzle-orm": "^0.45.2", + "framer-motion": "^12.38.0", + "israeli-bank-scrapers": "6.7.4", + "lucide-react": "^1.14.0", + "next": "16.2.6", + "next-intl": "^4.12.0", + "next-themes": "^0.4.6", + "react": "19.2.4", + "react-dom": "19.2.4", + "react-markdown": "^10.1.0", + "recharts": "^3.8.1", + "remark-gfm": "^4.0.1", + "server-only": "^0.0.1", + "sonner": "^2.0.7", + "tailwind-merge": "^3.6.0", + "tw-animate-css": "^1.4.0", + "zod": "^4.4.3", + }, + "devDependencies": { + "@biomejs/biome": "^2.2.0", + "@tailwindcss/postcss": "^4", + "@types/better-sqlite3": "^7.6.13", + "@types/bun": "latest", + "@types/node": "^20", + "@types/react": "^19", + "@types/react-dom": "^19", + "drizzle-kit": "^0.31.10", + "eslint": "^9", + "eslint-config-next": "16.2.6", + "knip": "^5", + "postcss": "^8.5.15", + "puppeteer": "^24", + "shadcn": "^4.7.0", + "tailwindcss": "^4", + "typescript": "^5", + }, + }, + }, + "trustedDependencies": [ + "@google/genai", + "sharp", + "msw", + "@parcel/watcher", + "@swc/core", + "protobufjs", + "unrs-resolver", + "better-sqlite3", + "puppeteer", + ], + "overrides": { + "postcss": "8.5.15", + }, + "packages": { + "@ai-sdk/anthropic": ["@ai-sdk/anthropic@3.0.79", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-saEX+h5JDOkT9P/+REKDyikbnJiToFuLipgNcsmu4Zr3GW5kW1m9HhvrPK+vj63itIOsoZU6tmVIjkrePOlIUA=="], + + "@ai-sdk/gateway": ["@ai-sdk/gateway@3.0.120", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27", "@vercel/oidc": "3.2.0" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-MYKAeD2q7/sa1ZdqtL2tw0Me0B8Tok6Q/fhkJDhJl39dG8u+VBlWO9yk9lcdm784bM418o1EKObo4aOxs6+18Q=="], + + "@ai-sdk/google": ["@ai-sdk/google@3.0.80", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-5ORbm/yFUPO0MEvZsxBMN0cdKw2+lwU/wVn5KN3KF8Dmk1LughuDuUohMh/7iU/XFTiyB0OvmTW/tdV/J7O9zg=="], + + "@ai-sdk/provider": ["@ai-sdk/provider@3.0.10", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-Q3BZ27qfpYqnCYGvE3vt+Qi6LGOF9R5Nmzn+9JoM1lCRsD9mYaIhfJLkSunN48nfGXJ6n+XNV0J/XVpqGQl7Dw=="], + + "@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.27", "", { "dependencies": { "@ai-sdk/provider": "3.0.10", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.8" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-ubkAJ+xODouwtmN1tYlvTPphH1hPOBfZaEQe8U7skGvFAnIRs9PPpsq57bC2+Ky/MB4yzhd6YOsxTAx9sGpazw=="], + + "@ai-sdk/react": ["@ai-sdk/react@3.0.193", "", { "dependencies": { "@ai-sdk/provider-utils": "4.0.27", "ai": "6.0.191", "swr": "^2.2.5", "throttleit": "2.1.0" }, "peerDependencies": { "react": "^18 || ~19.0.1 || ~19.1.2 || ^19.2.1" } }, "sha512-El0jUZ/B7mvBHAD5rfSDqOAhWxutVTq7BCNhfGuwfDPT9SO0TMHybh2bMkieJQI7YOfl+qNBoWrRAOHHaFb99Q=="], + + "@alloc/quick-lru": ["@alloc/quick-lru@5.2.0", "", {}, "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw=="], + + "@anthropic-ai/sdk": ["@anthropic-ai/sdk@0.95.2", "", { "dependencies": { "json-schema-to-ts": "^3.1.1", "standardwebhooks": "^1.0.0" }, "peerDependencies": { "zod": "^3.25.0 || ^4.0.0" }, "optionalPeers": ["zod"], "bin": { "anthropic-ai-sdk": "bin/cli" } }, "sha512-Egddwo3sheo1PzUrMkZnH6VkQYwS0h/b/i8vSK8Ta9M45UQipAMeDFH57dYuDAfXMEUUGeKw6CMlremgMZgrSQ=="], + + "@babel/code-frame": ["@babel/code-frame@7.29.7", "", { "dependencies": { "@babel/helper-validator-identifier": "^7.29.7", "js-tokens": "^4.0.0", "picocolors": "^1.1.1" } }, "sha512-Aup7aUOfpbAUg2ROOJN6Iw5f9DMBlzu0mIkm/malLQFN/YQgO48wCj0Kxa3sEHJvPVFg7siR+qRInwXd2qhQKw=="], + + "@babel/compat-data": ["@babel/compat-data@7.29.7", "", {}, "sha512-locTkQyKvwIEgBzVrn8693ebc97F2U8ZHjbXwDXJ5Fn2TCpNwTlKcaKLkdHop5c/icOFE7qt7Q9JC5hnKNa6Gg=="], + + "@babel/core": ["@babel/core@7.29.7", "", { "dependencies": { "@babel/code-frame": "^7.29.7", "@babel/generator": "^7.29.7", "@babel/helper-compilation-targets": "^7.29.7", "@babel/helper-module-transforms": "^7.29.7", "@babel/helpers": "^7.29.7", "@babel/parser": "^7.29.7", "@babel/template": "^7.29.7", "@babel/traverse": "^7.29.7", "@babel/types": "^7.29.7", "@jridgewell/remapping": "^2.3.5", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", "json5": "^2.2.3", "semver": "^6.3.1" } }, "sha512-RgHBCvtjbOK2gXSNBNIkNoEc9qoVEtau3hj8gEqKQuL3HZAibKarWFEI3Lfm6EYKkLalOh8eSrj9b+ch9H/VBA=="], + + "@babel/generator": ["@babel/generator@7.29.7", "", { "dependencies": { "@babel/parser": "^7.29.7", "@babel/types": "^7.29.7", "@jridgewell/gen-mapping": "^0.3.12", "@jridgewell/trace-mapping": "^0.3.28", "jsesc": "^3.0.2" } }, "sha512-DkXD5OJQaAQIdZ1bt3UZdEnHAn9Imd3IVBdX03UFe+ony9Ojw5pzr9YVKGDY1jt+Gcn/FnGkNf8r+Vj5NOJWtQ=="], + + "@babel/helper-annotate-as-pure": ["@babel/helper-annotate-as-pure@7.29.7", "", { "dependencies": { "@babel/types": "^7.29.7" } }, "sha512-OoK6239jHPuSQOoS0kfTVKn0b/rVTk0seKq4Gd2UMLtmOVLjDC0ki3e+c90Trqv2gMfvJFqkiljrr568+qddiw=="], + + "@babel/helper-compilation-targets": ["@babel/helper-compilation-targets@7.29.7", "", { "dependencies": { "@babel/compat-data": "^7.29.7", "@babel/helper-validator-option": "^7.29.7", "browserslist": "^4.24.0", "lru-cache": "^5.1.1", "semver": "^6.3.1" } }, "sha512-wem6WaBj4NaVYVdNhLPPVacES6ZJ+KBBfSkTMD3YZxbP3rm3Di85tJU5ljaUNhaOynt+Aj0xruhYuzQBt8n71g=="], + + "@babel/helper-create-class-features-plugin": ["@babel/helper-create-class-features-plugin@7.29.7", "", { "dependencies": { "@babel/helper-annotate-as-pure": "^7.29.7", "@babel/helper-member-expression-to-functions": "^7.29.7", "@babel/helper-optimise-call-expression": "^7.29.7", "@babel/helper-replace-supers": "^7.29.7", "@babel/helper-skip-transparent-expression-wrappers": "^7.29.7", "@babel/traverse": "^7.29.7", "semver": "^6.3.1" }, "peerDependencies": { "@babel/core": "^7.0.0" } }, "sha512-IY3ZD9Tmooqr3TUhc3DUWxiuo8xx1DWLhd5M7hQ+ZWJamqM2BbalrBJb2MisSLoYorOj75U03qULCxQTY9r3hg=="], + + "@babel/helper-globals": ["@babel/helper-globals@7.29.7", "", {}, "sha512-3nQVUAtvkKH9zahfWgw96Jc/uFOmjACE1kQz82E2lqWmHBgjzbNlsC22nuQTfahmWeQtTq5nQ/4Nnd2A1wj4zA=="], + + "@babel/helper-member-expression-to-functions": ["@babel/helper-member-expression-to-functions@7.29.7", "", { "dependencies": { "@babel/traverse": "^7.29.7", "@babel/types": "^7.29.7" } }, "sha512-j+7JYmk1JYDtACIGj0QJqqWZjoUpMoEikQGADMaHgCMCSDqd2+P32rfcibUNrGOMWrlzK1WJBdxrB3JJQZwWtg=="], + + "@babel/helper-module-imports": ["@babel/helper-module-imports@7.29.7", "", { "dependencies": { "@babel/traverse": "^7.29.7", "@babel/types": "^7.29.7" } }, "sha512-ejHwrQQYcm9xnTivShn2IDOlIzInN34AXskvq9QicvCtEzq1Vzclu/tKF8Jq1Cg8JG2GL6/EmjgsCT7lXepE3g=="], + + "@babel/helper-module-transforms": ["@babel/helper-module-transforms@7.29.7", "", { "dependencies": { "@babel/helper-module-imports": "^7.29.7", "@babel/helper-validator-identifier": "^7.29.7", "@babel/traverse": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0" } }, "sha512-UPUVSyXbOh627KiCIGQSgwWzGeBKLkaJ9PJEdrngIwMSzxLR4jS4+f1f1jb7VzBbg8nFLaYotvVPFCTqdrmTAg=="], + + "@babel/helper-optimise-call-expression": ["@babel/helper-optimise-call-expression@7.29.7", "", { "dependencies": { "@babel/types": "^7.29.7" } }, "sha512-+kmGVjcT9RGYzoDwdwEqEvGgKe3BYq+O1iGzjFubaNgZHwYHP6lsF2Yghf4kEuv9BV7tYDZ913aBW9am6YKong=="], + + "@babel/helper-plugin-utils": ["@babel/helper-plugin-utils@7.29.7", "", {}, "sha512-G7sHYigPY17oO5SYWnfD/0MTBwVR781S/JI643e/JhUYgVgWE/61SoW3NH9KWUKyKq5LVh3npif99Wkt6j86Jw=="], + + "@babel/helper-replace-supers": ["@babel/helper-replace-supers@7.29.7", "", { "dependencies": { "@babel/helper-member-expression-to-functions": "^7.29.7", "@babel/helper-optimise-call-expression": "^7.29.7", "@babel/traverse": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0" } }, "sha512-atfGXWSeCiF4DnKZIfmJfQRkSw9b9gNNXR1kqKjbhG4pGYCOnkp8OcTB8E3NXjBu8NpheSnOeNKz8KT7UNFTmQ=="], + + "@babel/helper-skip-transparent-expression-wrappers": ["@babel/helper-skip-transparent-expression-wrappers@7.29.7", "", { "dependencies": { "@babel/traverse": "^7.29.7", "@babel/types": "^7.29.7" } }, "sha512-brcMGQaVzIeUb+6/bs1Av0f8YuNNjKY2JyvfRCsFuFsdKccEQ5Ges2y74D74NZ1Rz8lKJ9ksJkfqwQFJ/iNEyQ=="], + + "@babel/helper-string-parser": ["@babel/helper-string-parser@7.29.7", "", {}, "sha512-Pb5ijPrZ89GDH8223L4UP8i6QApWxs04RbPQJTeWDV0/keR2E36MeKnyr6LYmUUvqRRI+Iv87SuF1W6ErINzYw=="], + + "@babel/helper-validator-identifier": ["@babel/helper-validator-identifier@7.29.7", "", {}, "sha512-qehxGkRj55h/ff8EMaJ+cYhyaKlHIxqYDn682wQD7RNp9UujOQsHog2uS0r2vzr4pW+sXf90NeeayjcNaX3fFg=="], + + "@babel/helper-validator-option": ["@babel/helper-validator-option@7.29.7", "", {}, "sha512-N9ZErrD+yW5geCDtBqnOoxmR8+tNKiGuxKlDpuJxfsqpa2dFcexaziGAE/qoHLiDDreVNMupxGmSoNlyvsA3gw=="], + + "@babel/helpers": ["@babel/helpers@7.29.7", "", { "dependencies": { "@babel/template": "^7.29.7", "@babel/types": "^7.29.7" } }, "sha512-1k2lAGRMfHTcwuNYcCNUmaUffmQv8KWMfh2iJUUeRlwlwH4FdNG7mfPI10NPfLHJFThE4Tyr4mv7kTNZOiPuBg=="], + + "@babel/parser": ["@babel/parser@7.29.7", "", { "dependencies": { "@babel/types": "^7.29.7" }, "bin": "./bin/babel-parser.js" }, "sha512-hnORnjP/1P/zFEndoeX+n+t1RwWRJiJpM/jO7FW32Kn9r5+sJB2JWOdYo4L6k78j15eCwY3Gm/7364B1EMwtNg=="], + + "@babel/plugin-syntax-jsx": ["@babel/plugin-syntax-jsx@7.29.7", "", { "dependencies": { "@babel/helper-plugin-utils": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-TSu8+mHCoEaaCDEZ0I3+6mvTBYR4PCxQwf2z9/r5Tbztv6NaLR3B9thGTTxX2WGuGHJqRiAbKPeGTJ5XWXVg6A=="], + + "@babel/plugin-syntax-typescript": ["@babel/plugin-syntax-typescript@7.29.7", "", { "dependencies": { "@babel/helper-plugin-utils": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-ngr+82Sh0xMz25TPCZi+nC2iTzjfCdWS2ONXTp/PtSCHCgaCNBpdMqgvJ2ccdLlClVZ7sisIgB914j/JFe+RZA=="], + + "@babel/plugin-transform-modules-commonjs": ["@babel/plugin-transform-modules-commonjs@7.29.7", "", { "dependencies": { "@babel/helper-module-transforms": "^7.29.7", "@babel/helper-plugin-utils": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-j0vCldybPC5b5dwCQOJ21uKtHzt7hxLygJTg9eF1ScfaikEDNfzn94XoW5Fi+seBR0nCyL23xaBFFkq7dTM8XQ=="], + + "@babel/plugin-transform-typescript": ["@babel/plugin-transform-typescript@7.29.7", "", { "dependencies": { "@babel/helper-annotate-as-pure": "^7.29.7", "@babel/helper-create-class-features-plugin": "^7.29.7", "@babel/helper-plugin-utils": "^7.29.7", "@babel/helper-skip-transparent-expression-wrappers": "^7.29.7", "@babel/plugin-syntax-typescript": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-jK52h8LaLc7JarhQV2ofeFMts4H7vnOXnqZNA6fYglBTZewRBE51KWt3BUltW1P+KoPsYkHoJeXePuz4zo2LMw=="], + + "@babel/preset-typescript": ["@babel/preset-typescript@7.29.7", "", { "dependencies": { "@babel/helper-plugin-utils": "^7.29.7", "@babel/helper-validator-option": "^7.29.7", "@babel/plugin-syntax-jsx": "^7.29.7", "@babel/plugin-transform-modules-commonjs": "^7.29.7", "@babel/plugin-transform-typescript": "^7.29.7" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-/Foi8vKY2EVbed/1eZx0gJEEwHAIxogrySI7rULcRIvhZzbvoE/b5qG5Ghc0WKAFKOHA9SD1x7RsFlOYdutIiQ=="], + + "@babel/runtime": ["@babel/runtime@7.29.7", "", {}, "sha512-Nq8OhGWiZIZGV6hLHoyAKLLcJihP/xFeBMGJoUrxTX2psI8dCifzLhZISFb+VWS3wFMRDmCGw5R+dOySCqPLhw=="], + + "@babel/template": ["@babel/template@7.29.7", "", { "dependencies": { "@babel/code-frame": "^7.29.7", "@babel/parser": "^7.29.7", "@babel/types": "^7.29.7" } }, "sha512-puq+Gf35oI24FeN11LkoUQFqv9uwNeWpxXZi/Ji3rRIoKAzKnxRaZ+Gkj0vKS9ZCiTESfng1N9LyOyXvo+m+Gg=="], + + "@babel/traverse": ["@babel/traverse@7.29.7", "", { "dependencies": { "@babel/code-frame": "^7.29.7", "@babel/generator": "^7.29.7", "@babel/helper-globals": "^7.29.7", "@babel/parser": "^7.29.7", "@babel/template": "^7.29.7", "@babel/types": "^7.29.7", "debug": "^4.3.1" } }, "sha512-EhlfNQtZ+NK22w5BM61ciuiq1m58ed33Wr1Xan//ZRTy6hgjnwyCffRYwzsGXdASJSUJ1guZILsErh1eQcl+zw=="], + + "@babel/types": ["@babel/types@7.29.7", "", { "dependencies": { "@babel/helper-string-parser": "^7.29.7", "@babel/helper-validator-identifier": "^7.29.7" } }, "sha512-4zBIxpPzowiZpusoFkyGVwakdRJUyuH5PxQ/PrqghfdFWWasvnCdPfQXHrenDai+gyLARulZjZowCOj6fjT4pA=="], + + "@base-ui/react": ["@base-ui/react@1.5.0", "", { "dependencies": { "@babel/runtime": "^7.29.2", "@base-ui/utils": "0.2.9", "@floating-ui/react-dom": "^2.1.8", "@floating-ui/utils": "^0.2.11", "use-sync-external-store": "^1.6.0" }, "peerDependencies": { "@date-fns/tz": "^1.2.0", "@types/react": "^17 || ^18 || ^19", "date-fns": "^4.0.0", "react": "^17 || ^18 || ^19", "react-dom": "^17 || ^18 || ^19" }, "optionalPeers": ["@date-fns/tz", "@types/react", "date-fns"] }, "sha512-z1gSAlced1yY+iM+mHDEtIkD8UI3Ebs52MuBPxvV6f5hRutk+xvCH/wuB7hDqDzK9JG5FoMz5nhrqtSs1wjt1A=="], + + "@base-ui/utils": ["@base-ui/utils@0.2.9", "", { "dependencies": { "@babel/runtime": "^7.29.2", "@floating-ui/utils": "^0.2.11", "reselect": "^5.1.1", "use-sync-external-store": "^1.6.0" }, "peerDependencies": { "@types/react": "^17 || ^18 || ^19", "react": "^17 || ^18 || ^19", "react-dom": "^17 || ^18 || ^19" }, "optionalPeers": ["@types/react"] }, "sha512-x/PDDCYzoqPpjrdyb3VcyylTI2IjUXEtYDGi5foh7KsnmNJIIaVwA2GLgDH1dps1GgXiJbA60hM+AyuTfQzIvw=="], + + "@biomejs/biome": ["@biomejs/biome@2.4.15", "", { "optionalDependencies": { "@biomejs/cli-darwin-arm64": "2.4.15", "@biomejs/cli-darwin-x64": "2.4.15", "@biomejs/cli-linux-arm64": "2.4.15", "@biomejs/cli-linux-arm64-musl": "2.4.15", "@biomejs/cli-linux-x64": "2.4.15", "@biomejs/cli-linux-x64-musl": "2.4.15", "@biomejs/cli-win32-arm64": "2.4.15", "@biomejs/cli-win32-x64": "2.4.15" }, "bin": { "biome": "bin/biome" } }, "sha512-j5VH3a/h/HXTKBM50MDMxRCzkeLv9S2XJcW2WgnZT1+xyisi+0bISrXR82gCX+8S9lvK0skEvHJRN+3Ktr2hlw=="], + + "@biomejs/cli-darwin-arm64": ["@biomejs/cli-darwin-arm64@2.4.15", "", { "os": "darwin", "cpu": "arm64" }, "sha512-rF3PPqLq1yoST79zaQbDjVJwsuIeci/O+9bgNmC5QpgOqz6aqYuzA4abyAGx+mgyiDXn4A049xAN8gijbuR1Qg=="], + + "@biomejs/cli-darwin-x64": ["@biomejs/cli-darwin-x64@2.4.15", "", { "os": "darwin", "cpu": "x64" }, "sha512-/5KHXYMfSJs1fNXiX30xFtI8JcCFV6zaVVLxOa0M2sfqBKHkpQhRTv94yxQWxeTY2lzo2OuTlNvPC+hDQt2wcQ=="], + + "@biomejs/cli-linux-arm64": ["@biomejs/cli-linux-arm64@2.4.15", "", { "os": "linux", "cpu": "arm64" }, "sha512-owaAMZD/T4LrD0ELNCk0Km3qrRHuM0X6EAyVE1FSqGY0rbLoiDLrO4Us2tllm6cAeB2Ioa9C2C08NZPdr8+0Ug=="], + + "@biomejs/cli-linux-arm64-musl": ["@biomejs/cli-linux-arm64-musl@2.4.15", "", { "os": "linux", "cpu": "arm64" }, "sha512-ZPcxznxm0pogHBLZhYntyR3sR+MrZjqJIKEr7ZqVen0Rl+P/4upVmfYXjftizi9RoqZntg33fv/1fbdhbYXpEQ=="], + + "@biomejs/cli-linux-x64": ["@biomejs/cli-linux-x64@2.4.15", "", { "os": "linux", "cpu": "x64" }, "sha512-0jj7THz12GbUOLmMibktK6DZjqz2zV64KFxyBtcFTKPiiOIY0a7vns1elpO1dERvxpsZ5ik0oFfz0oGwFde1+g=="], + + "@biomejs/cli-linux-x64-musl": ["@biomejs/cli-linux-x64-musl@2.4.15", "", { "os": "linux", "cpu": "x64" }, "sha512-CNq/9W38SYSH023lfcQ4KKU8K0YX8T//FZUhcgtMMRABDojx5XsMV7jlweAvGSl389wJQB29Qo6Zb/a+jdvt+w=="], + + "@biomejs/cli-win32-arm64": ["@biomejs/cli-win32-arm64@2.4.15", "", { "os": "win32", "cpu": "arm64" }, "sha512-ouhkYdlhp/1GghEJPdWwD/Vi3gQ1nFxuSpMolWsbq3Lsq3QUR4jl6UdhhscdCugKU5vOEuMiJhvKj66O0OCq+w=="], + + "@biomejs/cli-win32-x64": ["@biomejs/cli-win32-x64@2.4.15", "", { "os": "win32", "cpu": "x64" }, "sha512-zBrGq5mx5wwpnow4+2BxUvleDM+GNd4sLbPaMapsSLQLD0NGRCquqPBTgN+7XkUteHvj7M+BstuI8tmnV7+HgQ=="], + + "@dotenvx/dotenvx": ["@dotenvx/dotenvx@1.69.1", "", { "dependencies": { "commander": "^11.1.0", "dotenv": "^17.2.1", "eciesjs": "^0.4.10", "enquirer": "^2.4.1", "execa": "^5.1.1", "fdir": "^6.2.0", "ignore": "^5.3.0", "object-treeify": "1.1.33", "picomatch": "^4.0.4", "which": "^4.0.0", "yocto-spinner": "^1.1.0" }, "bin": { "dotenvx": "src/cli/dotenvx.js" } }, "sha512-kwQB5KcAegxw/+NGUgXAo5ovyOSjlMhoXSSnSEpDhoHJwzMcMO0HE1U0VCYZ7jbAeCMGamed9XdWzOA5ixtTNg=="], + + "@drizzle-team/brocli": ["@drizzle-team/brocli@0.10.2", "", {}, "sha512-z33Il7l5dKjUgGULTqBsQBQwckHh5AbIuxhdsIxDDiZAzBOrZO6q9ogcWC65kU382AfynTfgNumVcNIjuIua6w=="], + + "@ecies/ciphers": ["@ecies/ciphers@0.2.6", "", { "peerDependencies": { "@noble/ciphers": "^1.0.0" } }, "sha512-patgsRPKGkhhoBjETV4XxD0En4ui5fbX0hzayqI3M8tvNMGUoUvmyYAIWwlxBc1KX5cturfqByYdj5bYGRpN9g=="], + + "@emnapi/core": ["@emnapi/core@1.10.0", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.1", "tslib": "^2.4.0" } }, "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw=="], + + "@emnapi/runtime": ["@emnapi/runtime@1.10.0", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA=="], + + "@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.2.1", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w=="], + + "@esbuild-kit/core-utils": ["@esbuild-kit/core-utils@3.3.2", "", { "dependencies": { "esbuild": "~0.18.20", "source-map-support": "^0.5.21" } }, "sha512-sPRAnw9CdSsRmEtnsl2WXWdyquogVpB3yZ3dgwJfe8zrOzTsV7cJvmwrKVa+0ma5BoiGJ+BoqkMvawbayKUsqQ=="], + + "@esbuild-kit/esm-loader": ["@esbuild-kit/esm-loader@2.6.5", "", { "dependencies": { "@esbuild-kit/core-utils": "^3.3.2", "get-tsconfig": "^4.7.0" } }, "sha512-FxEMIkJKnodyA1OaCUoEvbYRkoZlLZ4d/eXFu9Fh8CbBBgP5EmZxrfTRyN0qpXZ4vOvqnE5YdRdcrmUUXuU+dA=="], + + "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.25.12", "", { "os": "aix", "cpu": "ppc64" }, "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA=="], + + "@esbuild/android-arm": ["@esbuild/android-arm@0.25.12", "", { "os": "android", "cpu": "arm" }, "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg=="], + + "@esbuild/android-arm64": ["@esbuild/android-arm64@0.25.12", "", { "os": "android", "cpu": "arm64" }, "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg=="], + + "@esbuild/android-x64": ["@esbuild/android-x64@0.25.12", "", { "os": "android", "cpu": "x64" }, "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg=="], + + "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.25.12", "", { "os": "darwin", "cpu": "arm64" }, "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg=="], + + "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.25.12", "", { "os": "darwin", "cpu": "x64" }, "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA=="], + + "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.25.12", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg=="], + + "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.25.12", "", { "os": "freebsd", "cpu": "x64" }, "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ=="], + + "@esbuild/linux-arm": ["@esbuild/linux-arm@0.25.12", "", { "os": "linux", "cpu": "arm" }, "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw=="], + + "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.25.12", "", { "os": "linux", "cpu": "arm64" }, "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ=="], + + "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.25.12", "", { "os": "linux", "cpu": "ia32" }, "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA=="], + + "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng=="], + + "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw=="], + + "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.25.12", "", { "os": "linux", "cpu": "ppc64" }, "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA=="], + + "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w=="], + + "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.25.12", "", { "os": "linux", "cpu": "s390x" }, "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg=="], + + "@esbuild/linux-x64": ["@esbuild/linux-x64@0.25.12", "", { "os": "linux", "cpu": "x64" }, "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw=="], + + "@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg=="], + + "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.25.12", "", { "os": "none", "cpu": "x64" }, "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ=="], + + "@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.25.12", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A=="], + + "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.25.12", "", { "os": "openbsd", "cpu": "x64" }, "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw=="], + + "@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg=="], + + "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.25.12", "", { "os": "sunos", "cpu": "x64" }, "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w=="], + + "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.25.12", "", { "os": "win32", "cpu": "arm64" }, "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg=="], + + "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.25.12", "", { "os": "win32", "cpu": "ia32" }, "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ=="], + + "@esbuild/win32-x64": ["@esbuild/win32-x64@0.25.12", "", { "os": "win32", "cpu": "x64" }, "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA=="], + + "@eslint-community/eslint-utils": ["@eslint-community/eslint-utils@4.9.1", "", { "dependencies": { "eslint-visitor-keys": "^3.4.3" }, "peerDependencies": { "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" } }, "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ=="], + + "@eslint-community/regexpp": ["@eslint-community/regexpp@4.12.2", "", {}, "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew=="], + + "@eslint/config-array": ["@eslint/config-array@0.21.2", "", { "dependencies": { "@eslint/object-schema": "^2.1.7", "debug": "^4.3.1", "minimatch": "^3.1.5" } }, "sha512-nJl2KGTlrf9GjLimgIru+V/mzgSK0ABCDQRvxw5BjURL7WfH5uoWmizbH7QB6MmnMBd8cIC9uceWnezL1VZWWw=="], + + "@eslint/config-helpers": ["@eslint/config-helpers@0.4.2", "", { "dependencies": { "@eslint/core": "^0.17.0" } }, "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw=="], + + "@eslint/core": ["@eslint/core@0.17.0", "", { "dependencies": { "@types/json-schema": "^7.0.15" } }, "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ=="], + + "@eslint/eslintrc": ["@eslint/eslintrc@3.3.5", "", { "dependencies": { "ajv": "^6.14.0", "debug": "^4.3.2", "espree": "^10.0.1", "globals": "^14.0.0", "ignore": "^5.2.0", "import-fresh": "^3.2.1", "js-yaml": "^4.1.1", "minimatch": "^3.1.5", "strip-json-comments": "^3.1.1" } }, "sha512-4IlJx0X0qftVsN5E+/vGujTRIFtwuLbNsVUe7TO6zYPDR1O6nFwvwhIKEKSrl6dZchmYBITazxKoUYOjdtjlRg=="], + + "@eslint/js": ["@eslint/js@9.39.4", "", {}, "sha512-nE7DEIchvtiFTwBw4Lfbu59PG+kCofhjsKaCWzxTpt4lfRjRMqG6uMBzKXuEcyXhOHoUp9riAm7/aWYGhXZ9cw=="], + + "@eslint/object-schema": ["@eslint/object-schema@2.1.7", "", {}, "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA=="], + + "@eslint/plugin-kit": ["@eslint/plugin-kit@0.4.1", "", { "dependencies": { "@eslint/core": "^0.17.0", "levn": "^0.4.1" } }, "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA=="], + + "@floating-ui/core": ["@floating-ui/core@1.7.5", "", { "dependencies": { "@floating-ui/utils": "^0.2.11" } }, "sha512-1Ih4WTWyw0+lKyFMcBHGbb5U5FtuHJuujoyyr5zTaWS5EYMeT6Jb2AuDeftsCsEuchO+mM2ij5+q9crhydzLhQ=="], + + "@floating-ui/dom": ["@floating-ui/dom@1.7.6", "", { "dependencies": { "@floating-ui/core": "^1.7.5", "@floating-ui/utils": "^0.2.11" } }, "sha512-9gZSAI5XM36880PPMm//9dfiEngYoC6Am2izES1FF406YFsjvyBMmeJ2g4SAju3xWwtuynNRFL2s9hgxpLI5SQ=="], + + "@floating-ui/react-dom": ["@floating-ui/react-dom@2.1.8", "", { "dependencies": { "@floating-ui/dom": "^1.7.6" }, "peerDependencies": { "react": ">=16.8.0", "react-dom": ">=16.8.0" } }, "sha512-cC52bHwM/n/CxS87FH0yWdngEZrjdtLW/qVruo68qg+prK7ZQ4YGdut2GyDVpoGeAYe/h899rVeOVm6Oi40k2A=="], + + "@floating-ui/utils": ["@floating-ui/utils@0.2.11", "", {}, "sha512-RiB/yIh78pcIxl6lLMG0CgBXAZ2Y0eVHqMPYugu+9U0AeT6YBeiJpf7lbdJNIugFP5SIjwNRgo4DhR1Qxi26Gg=="], + + "@formatjs/fast-memoize": ["@formatjs/fast-memoize@3.1.5", "", {}, "sha512-KLi3fan6WnCHmigd9pmEEN8Hid0v4wiFBW576M/d07KMWYecf1CvyMI3n34vCmHT4AoVqG2n702kiHbXjzZX2A=="], + + "@formatjs/icu-messageformat-parser": ["@formatjs/icu-messageformat-parser@3.5.10", "", { "dependencies": { "@formatjs/icu-skeleton-parser": "2.1.9" } }, "sha512-XeJihYLy1lCe19xfK1KWKG/betBOK2rB0luL8lSkjfvJj0zP+LTJvkC+RKd0jsFI8mWxN71LrarHSrEXE8xxOQ=="], + + "@formatjs/icu-skeleton-parser": ["@formatjs/icu-skeleton-parser@2.1.9", "", {}, "sha512-rsxswgHMfU1zUgB2byc08fesf83wLGjFnzLCEtuf00mx2doiqc6pYrf67raI37XqdRcGUviQepk2UKGqpng74Q=="], + + "@formatjs/intl-localematcher": ["@formatjs/intl-localematcher@0.8.8", "", { "dependencies": { "@formatjs/fast-memoize": "3.1.5" } }, "sha512-pBr2hVKWvkHVnfXegW+53NT9U2uaVQCc+EgzLPCCwXqBA3nvM5fPbK9IcJlNjV+NMKGyZ2F3ZSG78iGdxAAqbA=="], + + "@google/genai": ["@google/genai@2.6.0", "", { "dependencies": { "google-auth-library": "^10.3.0", "p-retry": "^4.6.2", "protobufjs": "^7.5.4", "ws": "^8.18.0" }, "peerDependencies": { "@modelcontextprotocol/sdk": "^1.25.2" }, "optionalPeers": ["@modelcontextprotocol/sdk"] }, "sha512-HjoW3mPuEn7pnuKABJl9VbDoWDSF4nbwYKYvYYor7YjPeDxrrBxHzu2d1Prcd+BAuC4w+85UP6y7ZdcrQAoO7g=="], + + "@hono/node-server": ["@hono/node-server@1.19.14", "", { "peerDependencies": { "hono": "^4" } }, "sha512-GwtvgtXxnWsucXvbQXkRgqksiH2Qed37H9xHZocE5sA3N8O8O8/8FA3uclQXxXVzc9XBZuEOMK7+r02FmSpHtw=="], + + "@humanfs/core": ["@humanfs/core@0.19.2", "", { "dependencies": { "@humanfs/types": "^0.15.0" } }, "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA=="], + + "@humanfs/node": ["@humanfs/node@0.16.8", "", { "dependencies": { "@humanfs/core": "^0.19.2", "@humanfs/types": "^0.15.0", "@humanwhocodes/retry": "^0.4.0" } }, "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ=="], + + "@humanfs/types": ["@humanfs/types@0.15.0", "", {}, "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q=="], + + "@humanwhocodes/module-importer": ["@humanwhocodes/module-importer@1.0.1", "", {}, "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA=="], + + "@humanwhocodes/retry": ["@humanwhocodes/retry@0.4.3", "", {}, "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ=="], + + "@img/colour": ["@img/colour@1.1.0", "", {}, "sha512-Td76q7j57o/tLVdgS746cYARfSyxk8iEfRxewL9h4OMzYhbW4TAcppl0mT4eyqXddh6L/jwoM75mo7ixa/pCeQ=="], + + "@img/sharp-darwin-arm64": ["@img/sharp-darwin-arm64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-darwin-arm64": "1.2.4" }, "os": "darwin", "cpu": "arm64" }, "sha512-imtQ3WMJXbMY4fxb/Ndp6HBTNVtWCUI0WdobyheGf5+ad6xX8VIDO8u2xE4qc/fr08CKG/7dDseFtn6M6g/r3w=="], + + "@img/sharp-darwin-x64": ["@img/sharp-darwin-x64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-darwin-x64": "1.2.4" }, "os": "darwin", "cpu": "x64" }, "sha512-YNEFAF/4KQ/PeW0N+r+aVVsoIY0/qxxikF2SWdp+NRkmMB7y9LBZAVqQ4yhGCm/H3H270OSykqmQMKLBhBJDEw=="], + + "@img/sharp-libvips-darwin-arm64": ["@img/sharp-libvips-darwin-arm64@1.2.4", "", { "os": "darwin", "cpu": "arm64" }, "sha512-zqjjo7RatFfFoP0MkQ51jfuFZBnVE2pRiaydKJ1G/rHZvnsrHAOcQALIi9sA5co5xenQdTugCvtb1cuf78Vf4g=="], + + "@img/sharp-libvips-darwin-x64": ["@img/sharp-libvips-darwin-x64@1.2.4", "", { "os": "darwin", "cpu": "x64" }, "sha512-1IOd5xfVhlGwX+zXv2N93k0yMONvUlANylbJw1eTah8K/Jtpi15KC+WSiaX/nBmbm2HxRM1gZ0nSdjSsrZbGKg=="], + + "@img/sharp-libvips-linux-arm": ["@img/sharp-libvips-linux-arm@1.2.4", "", { "os": "linux", "cpu": "arm" }, "sha512-bFI7xcKFELdiNCVov8e44Ia4u2byA+l3XtsAj+Q8tfCwO6BQ8iDojYdvoPMqsKDkuoOo+X6HZA0s0q11ANMQ8A=="], + + "@img/sharp-libvips-linux-arm64": ["@img/sharp-libvips-linux-arm64@1.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-excjX8DfsIcJ10x1Kzr4RcWe1edC9PquDRRPx3YVCvQv+U5p7Yin2s32ftzikXojb1PIFc/9Mt28/y+iRklkrw=="], + + "@img/sharp-libvips-linux-ppc64": ["@img/sharp-libvips-linux-ppc64@1.2.4", "", { "os": "linux", "cpu": "ppc64" }, "sha512-FMuvGijLDYG6lW+b/UvyilUWu5Ayu+3r2d1S8notiGCIyYU/76eig1UfMmkZ7vwgOrzKzlQbFSuQfgm7GYUPpA=="], + + "@img/sharp-libvips-linux-riscv64": ["@img/sharp-libvips-linux-riscv64@1.2.4", "", { "os": "linux", "cpu": "none" }, "sha512-oVDbcR4zUC0ce82teubSm+x6ETixtKZBh/qbREIOcI3cULzDyb18Sr/Wcyx7NRQeQzOiHTNbZFF1UwPS2scyGA=="], + + "@img/sharp-libvips-linux-s390x": ["@img/sharp-libvips-linux-s390x@1.2.4", "", { "os": "linux", "cpu": "s390x" }, "sha512-qmp9VrzgPgMoGZyPvrQHqk02uyjA0/QrTO26Tqk6l4ZV0MPWIW6LTkqOIov+J1yEu7MbFQaDpwdwJKhbJvuRxQ=="], + + "@img/sharp-libvips-linux-x64": ["@img/sharp-libvips-linux-x64@1.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-tJxiiLsmHc9Ax1bz3oaOYBURTXGIRDODBqhveVHonrHJ9/+k89qbLl0bcJns+e4t4rvaNBxaEZsFtSfAdquPrw=="], + + "@img/sharp-libvips-linuxmusl-arm64": ["@img/sharp-libvips-linuxmusl-arm64@1.2.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-FVQHuwx1IIuNow9QAbYUzJ+En8KcVm9Lk5+uGUQJHaZmMECZmOlix9HnH7n1TRkXMS0pGxIJokIVB9SuqZGGXw=="], + + "@img/sharp-libvips-linuxmusl-x64": ["@img/sharp-libvips-linuxmusl-x64@1.2.4", "", { "os": "linux", "cpu": "x64" }, "sha512-+LpyBk7L44ZIXwz/VYfglaX/okxezESc6UxDSoyo2Ks6Jxc4Y7sGjpgU9s4PMgqgjj1gZCylTieNamqA1MF7Dg=="], + + "@img/sharp-linux-arm": ["@img/sharp-linux-arm@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-arm": "1.2.4" }, "os": "linux", "cpu": "arm" }, "sha512-9dLqsvwtg1uuXBGZKsxem9595+ujv0sJ6Vi8wcTANSFpwV/GONat5eCkzQo/1O6zRIkh0m/8+5BjrRr7jDUSZw=="], + + "@img/sharp-linux-arm64": ["@img/sharp-linux-arm64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-arm64": "1.2.4" }, "os": "linux", "cpu": "arm64" }, "sha512-bKQzaJRY/bkPOXyKx5EVup7qkaojECG6NLYswgktOZjaXecSAeCWiZwwiFf3/Y+O1HrauiE3FVsGxFg8c24rZg=="], + + "@img/sharp-linux-ppc64": ["@img/sharp-linux-ppc64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-ppc64": "1.2.4" }, "os": "linux", "cpu": "ppc64" }, "sha512-7zznwNaqW6YtsfrGGDA6BRkISKAAE1Jo0QdpNYXNMHu2+0dTrPflTLNkpc8l7MUP5M16ZJcUvysVWWrMefZquA=="], + + "@img/sharp-linux-riscv64": ["@img/sharp-linux-riscv64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-riscv64": "1.2.4" }, "os": "linux", "cpu": "none" }, "sha512-51gJuLPTKa7piYPaVs8GmByo7/U7/7TZOq+cnXJIHZKavIRHAP77e3N2HEl3dgiqdD/w0yUfiJnII77PuDDFdw=="], + + "@img/sharp-linux-s390x": ["@img/sharp-linux-s390x@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-s390x": "1.2.4" }, "os": "linux", "cpu": "s390x" }, "sha512-nQtCk0PdKfho3eC5MrbQoigJ2gd1CgddUMkabUj+rBevs8tZ2cULOx46E7oyX+04WGfABgIwmMC0VqieTiR4jg=="], + + "@img/sharp-linux-x64": ["@img/sharp-linux-x64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linux-x64": "1.2.4" }, "os": "linux", "cpu": "x64" }, "sha512-MEzd8HPKxVxVenwAa+JRPwEC7QFjoPWuS5NZnBt6B3pu7EG2Ge0id1oLHZpPJdn3OQK+BQDiw9zStiHBTJQQQQ=="], + + "@img/sharp-linuxmusl-arm64": ["@img/sharp-linuxmusl-arm64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linuxmusl-arm64": "1.2.4" }, "os": "linux", "cpu": "arm64" }, "sha512-fprJR6GtRsMt6Kyfq44IsChVZeGN97gTD331weR1ex1c1rypDEABN6Tm2xa1wE6lYb5DdEnk03NZPqA7Id21yg=="], + + "@img/sharp-linuxmusl-x64": ["@img/sharp-linuxmusl-x64@0.34.5", "", { "optionalDependencies": { "@img/sharp-libvips-linuxmusl-x64": "1.2.4" }, "os": "linux", "cpu": "x64" }, "sha512-Jg8wNT1MUzIvhBFxViqrEhWDGzqymo3sV7z7ZsaWbZNDLXRJZoRGrjulp60YYtV4wfY8VIKcWidjojlLcWrd8Q=="], + + "@img/sharp-wasm32": ["@img/sharp-wasm32@0.34.5", "", { "dependencies": { "@emnapi/runtime": "^1.7.0" }, "cpu": "none" }, "sha512-OdWTEiVkY2PHwqkbBI8frFxQQFekHaSSkUIJkwzclWZe64O1X4UlUjqqqLaPbUpMOQk6FBu/HtlGXNblIs0huw=="], + + "@img/sharp-win32-arm64": ["@img/sharp-win32-arm64@0.34.5", "", { "os": "win32", "cpu": "arm64" }, "sha512-WQ3AgWCWYSb2yt+IG8mnC6Jdk9Whs7O0gxphblsLvdhSpSTtmu69ZG1Gkb6NuvxsNACwiPV6cNSZNzt0KPsw7g=="], + + "@img/sharp-win32-ia32": ["@img/sharp-win32-ia32@0.34.5", "", { "os": "win32", "cpu": "ia32" }, "sha512-FV9m/7NmeCmSHDD5j4+4pNI8Cp3aW+JvLoXcTUo0IqyjSfAZJ8dIUmijx1qaJsIiU+Hosw6xM5KijAWRJCSgNg=="], + + "@img/sharp-win32-x64": ["@img/sharp-win32-x64@0.34.5", "", { "os": "win32", "cpu": "x64" }, "sha512-+29YMsqY2/9eFEiW93eqWnuLcWcufowXewwSNIT6UwZdUUCrM3oFjMWH/Z6/TMmb4hlFenmfAVbpWeup2jryCw=="], + + "@inquirer/ansi": ["@inquirer/ansi@2.0.6", "", {}, "sha512-I/INw4sHGlVZ/afZOckpLiDP9SmbMl1g/GCqeHjLw1Afw/0PlRs2tRFgTGWmdI0hoNuWZn3y2iHNmG1vyECyQQ=="], + + "@inquirer/confirm": ["@inquirer/confirm@6.1.0", "", { "dependencies": { "@inquirer/core": "^11.2.0", "@inquirer/type": "^4.0.6" }, "peerDependencies": { "@types/node": ">=18" }, "optionalPeers": ["@types/node"] }, "sha512-USpeB76eqK7yGricDlGAupxWlp4a59qpeZOoNWaxO/nJln7agpJveyNkQ1d5u8YXG6TOqxZtQpKPORQQDrdVsA=="], + + "@inquirer/core": ["@inquirer/core@11.2.0", "", { "dependencies": { "@inquirer/ansi": "^2.0.6", "@inquirer/figures": "^2.0.6", "@inquirer/type": "^4.0.6", "cli-width": "^4.1.0", "fast-wrap-ansi": "^0.2.0", "mute-stream": "^4.0.0", "signal-exit": "^4.1.0" }, "peerDependencies": { "@types/node": ">=18" }, "optionalPeers": ["@types/node"] }, "sha512-joR1YS2sI0us+9d0I8ViqFbrRLONO8CFTuyvBX4ZVBSch+VsZiugUABdrhBXXJR1VyEzvpz5SQCix3keETQ58g=="], + + "@inquirer/figures": ["@inquirer/figures@2.0.6", "", {}, "sha512-dsZgQtH2t5Q6ah3aPbZbeEZAxsD9qQu0DXf01AltuEfRTm+NoLN6+rLVbr+4edeEbNCp/wBNM6mALRWtsQpfkw=="], + + "@inquirer/type": ["@inquirer/type@4.0.6", "", { "peerDependencies": { "@types/node": ">=18" }, "optionalPeers": ["@types/node"] }, "sha512-J+9tdxOskuYuGjsvGaq00AamhDgjR7anhEW2dP4QdQpFCMPngCeC/bCYWQ5NsMWZRdsy53is7kAHb/+7cwDk2g=="], + + "@jridgewell/gen-mapping": ["@jridgewell/gen-mapping@0.3.13", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.0", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA=="], + + "@jridgewell/remapping": ["@jridgewell/remapping@2.3.5", "", { "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ=="], + + "@jridgewell/resolve-uri": ["@jridgewell/resolve-uri@3.1.2", "", {}, "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw=="], + + "@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.5", "", {}, "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og=="], + + "@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.31", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw=="], + + "@modelcontextprotocol/sdk": ["@modelcontextprotocol/sdk@1.29.0", "", { "dependencies": { "@hono/node-server": "^1.19.9", "ajv": "^8.17.1", "ajv-formats": "^3.0.1", "content-type": "^1.0.5", "cors": "^2.8.5", "cross-spawn": "^7.0.5", "eventsource": "^3.0.2", "eventsource-parser": "^3.0.0", "express": "^5.2.1", "express-rate-limit": "^8.2.1", "hono": "^4.11.4", "jose": "^6.1.3", "json-schema-typed": "^8.0.2", "pkce-challenge": "^5.0.0", "raw-body": "^3.0.0", "zod": "^3.25 || ^4.0", "zod-to-json-schema": "^3.25.1" }, "peerDependencies": { "@cfworker/json-schema": "^4.1.1" }, "optionalPeers": ["@cfworker/json-schema"] }, "sha512-zo37mZA9hJWpULgkRpowewez1y6ML5GsXJPY8FI0tBBCd77HEvza4jDqRKOXgHNn867PVGCyTdzqpz0izu5ZjQ=="], + + "@mswjs/interceptors": ["@mswjs/interceptors@0.41.9", "", { "dependencies": { "@open-draft/deferred-promise": "^2.2.0", "@open-draft/logger": "^0.3.0", "@open-draft/until": "^2.0.0", "is-node-process": "^1.2.0", "outvariant": "^1.4.3", "strict-event-emitter": "^0.5.1" } }, "sha512-VVPPgHyQ6ShqnrmDWuxjmUIsO9gWyOZFmuOfLd9LfBGQJwZfy0gvv9pbHSJuoFNIYC7ZDX9aoFwowjcdSC4E8w=="], + + "@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.4", "", { "dependencies": { "@tybys/wasm-util": "^0.10.1" }, "peerDependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1" } }, "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow=="], + + "@next/env": ["@next/env@16.2.6", "", {}, "sha512-gd8HoHN4ufj73WmR3JmVolrpJR47ILK6LouP5xElPglaVxir6e1a7VzvTvDWkOoPXT9rkkTzyCxBu4yeZfZwcw=="], + + "@next/eslint-plugin-next": ["@next/eslint-plugin-next@16.2.6", "", { "dependencies": { "fast-glob": "3.3.1" } }, "sha512-Z8l6o4JWKUl755x4R+wogD86KPeU+Ckw4K+SYG4kHeOJtRenDeK+OSbGcqZpDtbwn9DsJVdir2UxmwXuinUbUw=="], + + "@next/swc-darwin-arm64": ["@next/swc-darwin-arm64@16.2.6", "", { "os": "darwin", "cpu": "arm64" }, "sha512-ZJGkkcNfYgrrMkqOdZ7zoLa1TOy0qpcMfk/z4Mh/FKUz40gVO+HNQWqmLxf67Z5WB64DRp0dhEbyHfel+6sJUg=="], + + "@next/swc-darwin-x64": ["@next/swc-darwin-x64@16.2.6", "", { "os": "darwin", "cpu": "x64" }, "sha512-v/YLBHIY132Ced3puBJ7YJKw1lqsCrgcNo2aRJlCEyQrrCeRJlvGlnmxhPxNQI3KE3N1DN5r9TPNPvka3nq5RQ=="], + + "@next/swc-linux-arm64-gnu": ["@next/swc-linux-arm64-gnu@16.2.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-RPOvqlYBbcQjkz9VQQDZ2T2bARIjXZV1KFlt+V2Mr6SW/e4I9fcKsaA0hdyf2FHoTlsV2xnBd5Y912rP/1Ce6w=="], + + "@next/swc-linux-arm64-musl": ["@next/swc-linux-arm64-musl@16.2.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-URUTu1+dMkxJsPFgm+OeEvq9wf5sujw0EvgYy80TDGHTSLTnIHeqb0Eu8A3sC95IRgjejQL+kC4mw+4yPxiAXA=="], + + "@next/swc-linux-x64-gnu": ["@next/swc-linux-x64-gnu@16.2.6", "", { "os": "linux", "cpu": "x64" }, "sha512-DOj182mPV8G3UkrayLoREM5YEYI+Dk5wv7Ox9xl1fFibAELEsFD0lDPfHIeILlutMMfdyhlzYPELG3peuKaurw=="], + + "@next/swc-linux-x64-musl": ["@next/swc-linux-x64-musl@16.2.6", "", { "os": "linux", "cpu": "x64" }, "sha512-HKQ5SP/V/ub73UvF7n/zeJlxk2kLmtL7Wzrg4WfmkjmNos5onJ2tKu7yZOPdL18A6Svfn3max29ym+ry7NkK4g=="], + + "@next/swc-win32-arm64-msvc": ["@next/swc-win32-arm64-msvc@16.2.6", "", { "os": "win32", "cpu": "arm64" }, "sha512-LZXpTlPyS5v7HhSmnvsLGP3iIYgYOBnc8r8ArlT55sGHV89bR2HlDdBjWQ+PY6SJMmk8TuVGFuxalnP3k/0Dwg=="], + + "@next/swc-win32-x64-msvc": ["@next/swc-win32-x64-msvc@16.2.6", "", { "os": "win32", "cpu": "x64" }, "sha512-F0+4i0h9J6C4eE3EAPWsoCk7UW/dbzOjyzxY0qnDUOYFu6FFmdZ6l97/XdV3/Nz3VYyO7UWjyEJUXkGqcoXfMA=="], + + "@noble/ciphers": ["@noble/ciphers@1.3.0", "", {}, "sha512-2I0gnIVPtfnMw9ee9h1dJG7tp81+8Ob3OJb3Mv37rx5L40/b0i7djjCVvGOVqc9AEIQyvyu1i6ypKdFw8R8gQw=="], + + "@noble/curves": ["@noble/curves@1.9.7", "", { "dependencies": { "@noble/hashes": "1.8.0" } }, "sha512-gbKGcRUYIjA3/zCCNaWDciTMFI0dCkvou3TL8Zmy5Nc7sJ47a0jtOeZoTaMxkuqRo9cRhjOdZJXegxYE5FN/xw=="], + + "@noble/hashes": ["@noble/hashes@1.8.0", "", {}, "sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A=="], + + "@nodelib/fs.scandir": ["@nodelib/fs.scandir@2.1.5", "", { "dependencies": { "@nodelib/fs.stat": "2.0.5", "run-parallel": "^1.1.9" } }, "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g=="], + + "@nodelib/fs.stat": ["@nodelib/fs.stat@2.0.5", "", {}, "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A=="], + + "@nodelib/fs.walk": ["@nodelib/fs.walk@1.2.8", "", { "dependencies": { "@nodelib/fs.scandir": "2.1.5", "fastq": "^1.6.0" } }, "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg=="], + + "@nolyfill/is-core-module": ["@nolyfill/is-core-module@1.0.39", "", {}, "sha512-nn5ozdjYQpUCZlWGuxcJY/KpxkWQs4DcbMCmKojjyrYDEAGy4Ce19NN4v5MduafTwJlbKc99UA8YhSVqq9yPZA=="], + + "@open-draft/deferred-promise": ["@open-draft/deferred-promise@3.0.0", "", {}, "sha512-XW375UK8/9SqUVNVa6M0yEy8+iTi4QN5VZ7aZuRFQmy76LRwI9wy5F4YIBU6T+eTe2/DNDo8tqu8RHlwLHM6RA=="], + + "@open-draft/logger": ["@open-draft/logger@0.3.0", "", { "dependencies": { "is-node-process": "^1.2.0", "outvariant": "^1.4.0" } }, "sha512-X2g45fzhxH238HKO4xbSr7+wBS8Fvw6ixhTDuvLd5mqh6bJJCFAPwU9mPDxbcrRtfxv4u5IHCEH77BmxvXmmxQ=="], + + "@open-draft/until": ["@open-draft/until@2.1.0", "", {}, "sha512-U69T3ItWHvLwGg5eJ0n3I62nWuE6ilHlmz7zM0npLBRvPRd7e6NYmg54vvRtP5mZG7kZqZCFVdsTWo7BPtBujg=="], + + "@opentelemetry/api": ["@opentelemetry/api@1.9.1", "", {}, "sha512-gLyJlPHPZYdAk1JENA9LeHejZe1Ti77/pTeFm/nMXmQH/HFZlcS/O2XJB+L8fkbrNSqhdtlvjBVjxwUYanNH5Q=="], + + "@oxc-resolver/binding-android-arm-eabi": ["@oxc-resolver/binding-android-arm-eabi@11.19.1", "", { "os": "android", "cpu": "arm" }, "sha512-aUs47y+xyXHUKlbhqHUjBABjvycq6YSD7bpxSW7vplUmdzAlJ93yXY6ZR0c1o1x5A/QKbENCvs3+NlY8IpIVzg=="], + + "@oxc-resolver/binding-android-arm64": ["@oxc-resolver/binding-android-arm64@11.19.1", "", { "os": "android", "cpu": "arm64" }, "sha512-oolbkRX+m7Pq2LNjr/kKgYeC7bRDMVTWPgxBGMjSpZi/+UskVo4jsMU3MLheZV55jL6c3rNelPl4oD60ggYmqA=="], + + "@oxc-resolver/binding-darwin-arm64": ["@oxc-resolver/binding-darwin-arm64@11.19.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-nUC6d2i3R5B12sUW4O646qD5cnMXf2oBGPLIIeaRfU9doJRORAbE2SGv4eW6rMqhD+G7nf2Y8TTJTLiiO3Q/dQ=="], + + "@oxc-resolver/binding-darwin-x64": ["@oxc-resolver/binding-darwin-x64@11.19.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-cV50vE5+uAgNcFa3QY1JOeKDSkM/9ReIcc/9wn4TavhW/itkDGrXhw9jaKnkQnGbjJ198Yh5nbX/Gr2mr4Z5jQ=="], + + "@oxc-resolver/binding-freebsd-x64": ["@oxc-resolver/binding-freebsd-x64@11.19.1", "", { "os": "freebsd", "cpu": "x64" }, "sha512-xZOQiYGFxtk48PBKff+Zwoym7ScPAIVp4c14lfLxizO2LTTTJe5sx9vQNGrBymrf/vatSPNMD4FgsaaRigPkqw=="], + + "@oxc-resolver/binding-linux-arm-gnueabihf": ["@oxc-resolver/binding-linux-arm-gnueabihf@11.19.1", "", { "os": "linux", "cpu": "arm" }, "sha512-lXZYWAC6kaGe/ky2su94e9jN9t6M0/6c+GrSlCqL//XO1cxi5lpAhnJYdyrKfm0ZEr/c7RNyAx3P7FSBcBd5+A=="], + + "@oxc-resolver/binding-linux-arm-musleabihf": ["@oxc-resolver/binding-linux-arm-musleabihf@11.19.1", "", { "os": "linux", "cpu": "arm" }, "sha512-veG1kKsuK5+t2IsO9q0DErYVSw2azvCVvWHnfTOS73WE0STdLLB7Q1bB9WR+yHPQM76ASkFyRbogWo1GR1+WbQ=="], + + "@oxc-resolver/binding-linux-arm64-gnu": ["@oxc-resolver/binding-linux-arm64-gnu@11.19.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-heV2+jmXyYnUrpUXSPugqWDRpnsQcDm2AX4wzTuvgdlZfoNYO0O3W2AVpJYaDn9AG4JdM6Kxom8+foE7/BcSig=="], + + "@oxc-resolver/binding-linux-arm64-musl": ["@oxc-resolver/binding-linux-arm64-musl@11.19.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-jvo2Pjs1c9KPxMuMPIeQsgu0mOJF9rEb3y3TdpsrqwxRM+AN6/nDDwv45n5ZrUnQMsdBy5gIabioMKnQfWo9ew=="], + + "@oxc-resolver/binding-linux-ppc64-gnu": ["@oxc-resolver/binding-linux-ppc64-gnu@11.19.1", "", { "os": "linux", "cpu": "ppc64" }, "sha512-vLmdNxWCdN7Uo5suays6A/+ywBby2PWBBPXctWPg5V0+eVuzsJxgAn6MMB4mPlshskYbppjpN2Zg83ArHze9gQ=="], + + "@oxc-resolver/binding-linux-riscv64-gnu": ["@oxc-resolver/binding-linux-riscv64-gnu@11.19.1", "", { "os": "linux", "cpu": "none" }, "sha512-/b+WgR+VTSBxzgOhDO7TlMXC1ufPIMR6Vj1zN+/x+MnyXGW7prTLzU9eW85Aj7Th7CCEG9ArCbTeqxCzFWdg2w=="], + + "@oxc-resolver/binding-linux-riscv64-musl": ["@oxc-resolver/binding-linux-riscv64-musl@11.19.1", "", { "os": "linux", "cpu": "none" }, "sha512-YlRdeWb9j42p29ROh+h4eg/OQ3dTJlpHSa+84pUM9+p6i3djtPz1q55yLJhgW9XfDch7FN1pQ/Vd6YP+xfRIuw=="], + + "@oxc-resolver/binding-linux-s390x-gnu": ["@oxc-resolver/binding-linux-s390x-gnu@11.19.1", "", { "os": "linux", "cpu": "s390x" }, "sha512-EDpafVOQWF8/MJynsjOGFThcqhRHy417sRyLfQmeiamJ8qVhSKAn2Dn2VVKUGCjVB9C46VGjhNo7nOPUi1x6uA=="], + + "@oxc-resolver/binding-linux-x64-gnu": ["@oxc-resolver/binding-linux-x64-gnu@11.19.1", "", { "os": "linux", "cpu": "x64" }, "sha512-NxjZe+rqWhr+RT8/Ik+5ptA3oz7tUw361Wa5RWQXKnfqwSSHdHyrw6IdcTfYuml9dM856AlKWZIUXDmA9kkiBQ=="], + + "@oxc-resolver/binding-linux-x64-musl": ["@oxc-resolver/binding-linux-x64-musl@11.19.1", "", { "os": "linux", "cpu": "x64" }, "sha512-cM/hQwsO3ReJg5kR+SpI69DMfvNCp+A/eVR4b4YClE5bVZwz8rh2Nh05InhwI5HR/9cArbEkzMjcKgTHS6UaNw=="], + + "@oxc-resolver/binding-openharmony-arm64": ["@oxc-resolver/binding-openharmony-arm64@11.19.1", "", { "os": "none", "cpu": "arm64" }, "sha512-QF080IowFB0+9Rh6RcD19bdgh49BpQHUW5TajG1qvWHvmrQznTZZjYlgE2ltLXyKY+qs4F/v5xuX1XS7Is+3qA=="], + + "@oxc-resolver/binding-wasm32-wasi": ["@oxc-resolver/binding-wasm32-wasi@11.19.1", "", { "dependencies": { "@napi-rs/wasm-runtime": "^1.1.1" }, "cpu": "none" }, "sha512-w8UCKhX826cP/ZLokXDS6+milN8y4X7zidsAttEdWlVoamTNf6lhBJldaWr3ukTDiye7s4HRcuPEPOXNC432Vg=="], + + "@oxc-resolver/binding-win32-arm64-msvc": ["@oxc-resolver/binding-win32-arm64-msvc@11.19.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-nJ4AsUVZrVKwnU/QRdzPCCrO0TrabBqgJ8pJhXITdZGYOV28TIYystV1VFLbQ7DtAcaBHpocT5/ZJnF78YJPtQ=="], + + "@oxc-resolver/binding-win32-ia32-msvc": ["@oxc-resolver/binding-win32-ia32-msvc@11.19.1", "", { "os": "win32", "cpu": "ia32" }, "sha512-EW+ND5q2Tl+a3pH81l1QbfgbF3HmqgwLfDfVithRFheac8OTcnbXt/JxqD2GbDkb7xYEqy1zNaVFRr3oeG8npA=="], + + "@oxc-resolver/binding-win32-x64-msvc": ["@oxc-resolver/binding-win32-x64-msvc@11.19.1", "", { "os": "win32", "cpu": "x64" }, "sha512-6hIU3RQu45B+VNTY4Ru8ppFwjVS/S5qwYyGhBotmjxfEKk41I2DlGtRfGJndZ5+6lneE2pwloqunlOyZuX/XAw=="], + + "@parcel/watcher": ["@parcel/watcher@2.5.6", "", { "dependencies": { "detect-libc": "^2.0.3", "is-glob": "^4.0.3", "node-addon-api": "^7.0.0", "picomatch": "^4.0.3" }, "optionalDependencies": { "@parcel/watcher-android-arm64": "2.5.6", "@parcel/watcher-darwin-arm64": "2.5.6", "@parcel/watcher-darwin-x64": "2.5.6", "@parcel/watcher-freebsd-x64": "2.5.6", "@parcel/watcher-linux-arm-glibc": "2.5.6", "@parcel/watcher-linux-arm-musl": "2.5.6", "@parcel/watcher-linux-arm64-glibc": "2.5.6", "@parcel/watcher-linux-arm64-musl": "2.5.6", "@parcel/watcher-linux-x64-glibc": "2.5.6", "@parcel/watcher-linux-x64-musl": "2.5.6", "@parcel/watcher-win32-arm64": "2.5.6", "@parcel/watcher-win32-ia32": "2.5.6", "@parcel/watcher-win32-x64": "2.5.6" } }, "sha512-tmmZ3lQxAe/k/+rNnXQRawJ4NjxO2hqiOLTHvWchtGZULp4RyFeh6aU4XdOYBFe2KE1oShQTv4AblOs2iOrNnQ=="], + + "@parcel/watcher-android-arm64": ["@parcel/watcher-android-arm64@2.5.6", "", { "os": "android", "cpu": "arm64" }, "sha512-YQxSS34tPF/6ZG7r/Ih9xy+kP/WwediEUsqmtf0cuCV5TPPKw/PQHRhueUo6JdeFJaqV3pyjm0GdYjZotbRt/A=="], + + "@parcel/watcher-darwin-arm64": ["@parcel/watcher-darwin-arm64@2.5.6", "", { "os": "darwin", "cpu": "arm64" }, "sha512-Z2ZdrnwyXvvvdtRHLmM4knydIdU9adO3D4n/0cVipF3rRiwP+3/sfzpAwA/qKFL6i1ModaabkU7IbpeMBgiVEA=="], + + "@parcel/watcher-darwin-x64": ["@parcel/watcher-darwin-x64@2.5.6", "", { "os": "darwin", "cpu": "x64" }, "sha512-HgvOf3W9dhithcwOWX9uDZyn1lW9R+7tPZ4sug+NGrGIo4Rk1hAXLEbcH1TQSqxts0NYXXlOWqVpvS1SFS4fRg=="], + + "@parcel/watcher-freebsd-x64": ["@parcel/watcher-freebsd-x64@2.5.6", "", { "os": "freebsd", "cpu": "x64" }, "sha512-vJVi8yd/qzJxEKHkeemh7w3YAn6RJCtYlE4HPMoVnCpIXEzSrxErBW5SJBgKLbXU3WdIpkjBTeUNtyBVn8TRng=="], + + "@parcel/watcher-linux-arm-glibc": ["@parcel/watcher-linux-arm-glibc@2.5.6", "", { "os": "linux", "cpu": "arm" }, "sha512-9JiYfB6h6BgV50CCfasfLf/uvOcJskMSwcdH1PHH9rvS1IrNy8zad6IUVPVUfmXr+u+Km9IxcfMLzgdOudz9EQ=="], + + "@parcel/watcher-linux-arm-musl": ["@parcel/watcher-linux-arm-musl@2.5.6", "", { "os": "linux", "cpu": "arm" }, "sha512-Ve3gUCG57nuUUSyjBq/MAM0CzArtuIOxsBdQ+ftz6ho8n7s1i9E1Nmk/xmP323r2YL0SONs1EuwqBp2u1k5fxg=="], + + "@parcel/watcher-linux-arm64-glibc": ["@parcel/watcher-linux-arm64-glibc@2.5.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-f2g/DT3NhGPdBmMWYoxixqYr3v/UXcmLOYy16Bx0TM20Tchduwr4EaCbmxh1321TABqPGDpS8D/ggOTaljijOA=="], + + "@parcel/watcher-linux-arm64-musl": ["@parcel/watcher-linux-arm64-musl@2.5.6", "", { "os": "linux", "cpu": "arm64" }, "sha512-qb6naMDGlbCwdhLj6hgoVKJl2odL34z2sqkC7Z6kzir8b5W65WYDpLB6R06KabvZdgoHI/zxke4b3zR0wAbDTA=="], + + "@parcel/watcher-linux-x64-glibc": ["@parcel/watcher-linux-x64-glibc@2.5.6", "", { "os": "linux", "cpu": "x64" }, "sha512-kbT5wvNQlx7NaGjzPFu8nVIW1rWqV780O7ZtkjuWaPUgpv2NMFpjYERVi0UYj1msZNyCzGlaCWEtzc+exjMGbQ=="], + + "@parcel/watcher-linux-x64-musl": ["@parcel/watcher-linux-x64-musl@2.5.6", "", { "os": "linux", "cpu": "x64" }, "sha512-1JRFeC+h7RdXwldHzTsmdtYR/Ku8SylLgTU/reMuqdVD7CtLwf0VR1FqeprZ0eHQkO0vqsbvFLXUmYm/uNKJBg=="], + + "@parcel/watcher-win32-arm64": ["@parcel/watcher-win32-arm64@2.5.6", "", { "os": "win32", "cpu": "arm64" }, "sha512-3ukyebjc6eGlw9yRt678DxVF7rjXatWiHvTXqphZLvo7aC5NdEgFufVwjFfY51ijYEWpXbqF5jtrK275z52D4Q=="], + + "@parcel/watcher-win32-ia32": ["@parcel/watcher-win32-ia32@2.5.6", "", { "os": "win32", "cpu": "ia32" }, "sha512-k35yLp1ZMwwee3Ez/pxBi5cf4AoBKYXj00CZ80jUz5h8prpiaQsiRPKQMxoLstNuqe2vR4RNPEAEcjEFzhEz/g=="], + + "@parcel/watcher-win32-x64": ["@parcel/watcher-win32-x64@2.5.6", "", { "os": "win32", "cpu": "x64" }, "sha512-hbQlYcCq5dlAX9Qx+kFb0FHue6vbjlf0FrNzSKdYK2APUf7tGfGxQCk2ihEREmbR6ZMc0MVAD5RIX/41gpUzTw=="], + + "@protobufjs/aspromise": ["@protobufjs/aspromise@1.1.2", "", {}, "sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ=="], + + "@protobufjs/base64": ["@protobufjs/base64@1.1.2", "", {}, "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg=="], + + "@protobufjs/codegen": ["@protobufjs/codegen@2.0.5", "", {}, "sha512-zgXFLzW3Ap33e6d0Wlj4MGIm6Ce8O89n/apUaGNB/jx+hw+ruWEp7EwGUshdLKVRCxZW12fp9r40E1mQrf/34g=="], + + "@protobufjs/eventemitter": ["@protobufjs/eventemitter@1.1.1", "", {}, "sha512-vW1GmwMZNnL+gMRaovlh9yZX74kc+TTU3FObkkurpMaRtBfLP3ldjS9KQWlwZgraRE0+dheEEoAxdzcJQ8eXZg=="], + + "@protobufjs/fetch": ["@protobufjs/fetch@1.1.1", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.1" } }, "sha512-GpptLrs57adMSuHi3VNj0mAF8dwh36LMaYF6XyJ6JMWlVsc+t42tm1HSEDmOs3A8fC9yyeisgLhsTVQokOZ0zw=="], + + "@protobufjs/float": ["@protobufjs/float@1.0.2", "", {}, "sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ=="], + + "@protobufjs/inquire": ["@protobufjs/inquire@1.1.2", "", {}, "sha512-pa0vFRuws4wkvaXKK1uXZMAwAX4/t8ANaJo45iw/oQHNQ9q5xUzwgFmVJGXiga2BeN+zpX7Vf9vmsiIa2J+MUw=="], + + "@protobufjs/path": ["@protobufjs/path@1.1.2", "", {}, "sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA=="], + + "@protobufjs/pool": ["@protobufjs/pool@1.1.0", "", {}, "sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw=="], + + "@protobufjs/utf8": ["@protobufjs/utf8@1.1.1", "", {}, "sha512-oOAWABowe8EAbMyWKM0tYDKi8Yaox52D+HWZhAIJqQXbqe0xI/GV7FhLWqlEKreMkfDjshR5FKgi3mnle0h6Eg=="], + + "@puppeteer/browsers": ["@puppeteer/browsers@2.13.2", "", { "dependencies": { "debug": "^4.4.3", "extract-zip": "^2.0.1", "progress": "^2.0.3", "proxy-agent": "^6.5.0", "semver": "^7.7.4", "tar-fs": "^3.1.1", "yargs": "^17.7.2" }, "bin": { "browsers": "lib/cjs/main-cli.js" } }, "sha512-5EUZSUIc37H6aIXyWO0Z4y8NlF8NnjgmqeQgOGiswAU7pY0HOo16ho4+alIWmSfdZnjqBRawMsP3I5YqLSn6kw=="], + + "@reduxjs/toolkit": ["@reduxjs/toolkit@2.12.0", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "@standard-schema/utils": "^0.3.0", "immer": "^11.0.0", "redux": "^5.0.1", "redux-thunk": "^3.1.0", "reselect": "^5.1.0" }, "peerDependencies": { "react": "^16.9.0 || ^17.0.0 || ^18 || ^19", "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0" }, "optionalPeers": ["react", "react-redux"] }, "sha512-KiT+RzZbp6mQET+Mg+h2c97+9j1sNflUxQkIHI7Yuzf6Peu+OYpmkn6nbHWmLLWj+1ZODUJFwGZ7gx3L9R9EOw=="], + + "@rtsao/scc": ["@rtsao/scc@1.1.0", "", {}, "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g=="], + + "@schummar/icu-type-parser": ["@schummar/icu-type-parser@1.21.5", "", {}, "sha512-bXHSaW5jRTmke9Vd0h5P7BtWZG9Znqb8gSDxZnxaGSJnGwPLDPfS+3g0BKzeWqzgZPsIVZkM7m2tbo18cm5HBw=="], + + "@sec-ant/readable-stream": ["@sec-ant/readable-stream@0.4.1", "", {}, "sha512-831qok9r2t8AlxLko40y2ebgSDhenenCatLVeW/uBtnHPyhHOvG0C7TvfgecV+wHzIm5KUICgzmVpWS+IMEAeg=="], + + "@sindresorhus/merge-streams": ["@sindresorhus/merge-streams@4.0.0", "", {}, "sha512-tlqY9xq5ukxTUZBmoOp+m61cqwQD5pHJtFY3Mn8CA8ps6yghLH/Hw8UPdqg4OLmFW3IFlcXnQNmo/dh8HzXYIQ=="], + + "@stablelib/base64": ["@stablelib/base64@1.0.1", "", {}, "sha512-1bnPQqSxSuc3Ii6MhBysoWCg58j97aUjuCSZrGSmDxNqtytIi0k8utUenAwTZN4V5mXXYGsVUI9zeBqy+jBOSQ=="], + + "@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], + + "@standard-schema/utils": ["@standard-schema/utils@0.3.0", "", {}, "sha512-e7Mew686owMaPJVNNLs55PUvgz371nKgwsc4vxE49zsODpJEnxgxRo2y/OKrqueavXgZNMDVj3DdHFlaSAeU8g=="], + + "@swc/core": ["@swc/core@1.15.40", "", { "dependencies": { "@swc/counter": "^0.1.3", "@swc/types": "^0.1.26" }, "optionalDependencies": { "@swc/core-darwin-arm64": "1.15.40", "@swc/core-darwin-x64": "1.15.40", "@swc/core-linux-arm-gnueabihf": "1.15.40", "@swc/core-linux-arm64-gnu": "1.15.40", "@swc/core-linux-arm64-musl": "1.15.40", "@swc/core-linux-ppc64-gnu": "1.15.40", "@swc/core-linux-s390x-gnu": "1.15.40", "@swc/core-linux-x64-gnu": "1.15.40", "@swc/core-linux-x64-musl": "1.15.40", "@swc/core-win32-arm64-msvc": "1.15.40", "@swc/core-win32-ia32-msvc": "1.15.40", "@swc/core-win32-x64-msvc": "1.15.40" }, "peerDependencies": { "@swc/helpers": ">=0.5.17" }, "optionalPeers": ["@swc/helpers"] }, "sha512-2kwzJikRvgtNAG7MwVZY2vEzZjTxKIq5jXOihuSV/8U+Hej8Va22t65aKnJZs3P+NwojZvR8Mf8kyM7O+V8sQg=="], + + "@swc/core-darwin-arm64": ["@swc/core-darwin-arm64@1.15.40", "", { "os": "darwin", "cpu": "arm64" }, "sha512-PaYyclfmQ++77D8ityYvmmVzHv9aG8ROwt2GfG6/ccloy4Hgf80qtOnzb9VYvPsUT7Ty1uhuDRhv3XYpf62qhQ=="], + + "@swc/core-darwin-x64": ["@swc/core-darwin-x64@1.15.40", "", { "os": "darwin", "cpu": "x64" }, "sha512-HbbPzvfLBUXjIB1Ezks+//lNUjmLjfyd63XSwprJgrZaXYdm70kohXPJUWdqKZozolFxbPaO+xtBaiUp6BoueA=="], + + "@swc/core-linux-arm-gnueabihf": ["@swc/core-linux-arm-gnueabihf@1.15.40", "", { "os": "linux", "cpu": "arm" }, "sha512-SlRZsCjOCPR2LvFs0Ri/Xrx/5o5TCt8vl4gW6mX1hEZOG0a625RxzRHpHdAQNGykmAN/7IeaFAJG+QnNmxlHcA=="], + + "@swc/core-linux-arm64-gnu": ["@swc/core-linux-arm64-gnu@1.15.40", "", { "os": "linux", "cpu": "arm64" }, "sha512-Q8byxJt2fh8CR3EUX6snBpy47AoBVm+In/+Z3rjDHMjC38ZvR9/gtUUNCT0tfrn4EdVsO8/QPi59nxrxvqxvBQ=="], + + "@swc/core-linux-arm64-musl": ["@swc/core-linux-arm64-musl@1.15.40", "", { "os": "linux", "cpu": "arm64" }, "sha512-4z0MgHU+7M0pZDqBN1El7mFXDI1SBwinfcUkAyA4v8QrhOIUOZltySt2aStQLZGrdXVXM4Y4ylfiTC04ED+MoQ=="], + + "@swc/core-linux-ppc64-gnu": ["@swc/core-linux-ppc64-gnu@1.15.40", "", { "os": "linux", "cpu": "ppc64" }, "sha512-fLI4iUgeSZu0eRWUXwe6YzPFx9gHbFiPkl8Rp3mJfP8OpNR3nTQCGPvHdDh9xniW7mVvgMY4ni7A4VzqI1KrpA=="], + + "@swc/core-linux-s390x-gnu": ["@swc/core-linux-s390x-gnu@1.15.40", "", { "os": "linux", "cpu": "s390x" }, "sha512-YqeKMAb7d4nQSGMJQ454IlaCENpzcDqhvBE9+CPfdnYpnUXxd+BSrB6Xk0YjW8UyoEhUj4p6quATCxbsp6J3jg=="], + + "@swc/core-linux-x64-gnu": ["@swc/core-linux-x64-gnu@1.15.40", "", { "os": "linux", "cpu": "x64" }, "sha512-7HOuS1iGcme/j/TuL1TfmmLGiMQrjv/GmjyZeydl00FKPtpGXEldwqfI56xgd1YzrzoB2svWjxbGGyQ0TEASxg=="], + + "@swc/core-linux-x64-musl": ["@swc/core-linux-x64-musl@1.15.40", "", { "os": "linux", "cpu": "x64" }, "sha512-h4kZYHc7dpc9P9u4brRJaS8Pl7tPVHAeiLSzw7T5RfIJgAoSdaCMKzI/2Uay9gFhaw8uyCDl0L5q37r0EpAfIA=="], + + "@swc/core-win32-arm64-msvc": ["@swc/core-win32-arm64-msvc@1.15.40", "", { "os": "win32", "cpu": "arm64" }, "sha512-+mQgKZXSj6mV38Zh05QaxSjUDmGP/R2JWlXZTDLSPkDzHU6p3GxN9eeSf5dfyDVU86946fmCvSzyl/ucImx8+A=="], + + "@swc/core-win32-ia32-msvc": ["@swc/core-win32-ia32-msvc@1.15.40", "", { "os": "win32", "cpu": "ia32" }, "sha512-yvwdPLGd25mcj/mNatjNQ0lZujtQD6psH3v9PNmMb+fSzjbNG8KIDxjFWrcV+fsFVLOkyOmdJsFmX7NAFjVyPw=="], + + "@swc/core-win32-x64-msvc": ["@swc/core-win32-x64-msvc@1.15.40", "", { "os": "win32", "cpu": "x64" }, "sha512-OXtKsLU1bVtInzzDEAY2sYiF/rl4tvAnLLLpuMp3HzAOQZ5A+i69AKDhA1YLQTaMAqO3vzyYNVAYVRMPtSYD4w=="], + + "@swc/counter": ["@swc/counter@0.1.3", "", {}, "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ=="], + + "@swc/helpers": ["@swc/helpers@0.5.15", "", { "dependencies": { "tslib": "^2.8.0" } }, "sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g=="], + + "@swc/types": ["@swc/types@0.1.26", "", { "dependencies": { "@swc/counter": "^0.1.3" } }, "sha512-lyMwd7WGgG79RS7EERZV3T8wMdmPq3xwyg+1nmAM64kIhx5yl+juO2PYIHb7vTiPgPCj8LYjsNV2T5wiQHUEaw=="], + + "@tailwindcss/node": ["@tailwindcss/node@4.3.0", "", { "dependencies": { "@jridgewell/remapping": "^2.3.5", "enhanced-resolve": "^5.21.0", "jiti": "^2.6.1", "lightningcss": "1.32.0", "magic-string": "^0.30.21", "source-map-js": "^1.2.1", "tailwindcss": "4.3.0" } }, "sha512-aFb4gUhFOgdh9AXo4IzBEOzBkkAxm9VigwDJnMIYv3lcfXCJVesNfbEaBl4BNgVRyid92AmdviqwBUBRKSeY3g=="], + + "@tailwindcss/oxide": ["@tailwindcss/oxide@4.3.0", "", { "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.3.0", "@tailwindcss/oxide-darwin-arm64": "4.3.0", "@tailwindcss/oxide-darwin-x64": "4.3.0", "@tailwindcss/oxide-freebsd-x64": "4.3.0", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.3.0", "@tailwindcss/oxide-linux-arm64-gnu": "4.3.0", "@tailwindcss/oxide-linux-arm64-musl": "4.3.0", "@tailwindcss/oxide-linux-x64-gnu": "4.3.0", "@tailwindcss/oxide-linux-x64-musl": "4.3.0", "@tailwindcss/oxide-wasm32-wasi": "4.3.0", "@tailwindcss/oxide-win32-arm64-msvc": "4.3.0", "@tailwindcss/oxide-win32-x64-msvc": "4.3.0" } }, "sha512-F7HZGBeN9I0/AuuJS5PwcD8xayx5ri5GhjYUDBEVYUkexyA/giwbDNjRVrxSezE3T250OU2K/wp/ltWx3UOefg=="], + + "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.3.0", "", { "os": "android", "cpu": "arm64" }, "sha512-TJPiq67tKlLuObP6RkwvVGDoxCMBVtDgKkLfa/uyj7/FyxvQwHS+UOnVrXXgbEsfUaMgiVvC4KbJnRr26ho4Ng=="], + + "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.3.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-oMN/WZRb+SO37BmUElEgeEWuU8E/HXRkiODxJxLe1UTHVXLrdVSgfaJV7pSlhRGMSOiXLuxTIjfsF3wYvz8cgQ=="], + + "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.3.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-N6CUmu4a6bKVADfw77p+iw6Yd9Q3OBhe0veaDX+QazfuVYlQsHfDgxBrsjQ/IW+zywL8mTrNd0SdJT/zgtvMdA=="], + + "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.3.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-zDL5hBkQdH5C6MpqbK3gQAgP80tsMwSI26vjOzjJtNCMUo0lFgOItzHKBIupOZNQxt3ouPH7RPhvNhiTfCe5CQ=="], + + "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.3.0", "", { "os": "linux", "cpu": "arm" }, "sha512-R06HdNi7A7OEoMsf6d4tjZ71RCWnZQPHj2mnotSFURjNLdBC+cIgXQ7l81CqeoiQftjf6OOblxXMInMgN2VzMA=="], + + "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.3.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-qTJHELX8jetjhRQHCLilkVLmybpzNQAtaI/gaoVoidn/ufbNDbAo8KlK2J+yPoc8wQxvDxCmh/5lr8nC1+lTbg=="], + + "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.3.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-Z6sukiQsngnWO+l39X4pPbiWT81IC+PLKF+PHxIlyZbGNb9MODfYlXEVlFvej5BOZInWX01kVyzeLvHsXhfczQ=="], + + "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.3.0", "", { "os": "linux", "cpu": "x64" }, "sha512-DRNdQRpSGzRGfARVuVkxvM8Q12nh19l4BF/G7zGA1oe+9wcC6saFBHTISrpIcKzhiXtSrlSrluCfvMuledoCTQ=="], + + "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.3.0", "", { "os": "linux", "cpu": "x64" }, "sha512-Z0IADbDo8bh6I7h2IQMx601AdXBLfFpEdUotft86evd/8ZPflZe9COPO8Q1vw+pfLWIUo9zN/JGZvwuAJqduqg=="], + + "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.3.0", "", { "dependencies": { "@emnapi/core": "^1.10.0", "@emnapi/runtime": "^1.10.0", "@emnapi/wasi-threads": "^1.2.1", "@napi-rs/wasm-runtime": "^1.1.4", "@tybys/wasm-util": "^0.10.1", "tslib": "^2.8.1" }, "cpu": "none" }, "sha512-HNZGOUxEmElksYR7S6sC5jTeNGpobAsy9u7Gu0AskJ8/20FR9GqebUyB+HBcU/ax6BHuiuJi+Oda4B+YX6H1yA=="], + + "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.3.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-Pe+RPVTi1T+qymuuRpcdvwSVZjnll/f7n8gBxMMh3xLTctMDKqpdfGimbMyioqtLhUYZxdJ9wGNhV7MKHvgZsQ=="], + + "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.3.0", "", { "os": "win32", "cpu": "x64" }, "sha512-Mvrf2kXW/yeW/OTezZlCGOirXRcUuLIBx/5Y12BaPM7wJoryG6dfS/NJL8aBPqtTEx/Vm4T4vKzFUcKDT+TKUA=="], + + "@tailwindcss/postcss": ["@tailwindcss/postcss@4.3.0", "", { "dependencies": { "@alloc/quick-lru": "^5.2.0", "@tailwindcss/node": "4.3.0", "@tailwindcss/oxide": "4.3.0", "postcss": "^8.5.10", "tailwindcss": "4.3.0" } }, "sha512-Jm05Tjx+9yCLGv5qw1c+84Psds8MnyrEQYCB+FFk2lgGiUjlRqdxke4mVTuYrj2xnVZqKim2Apr5ySuQRYAw/w=="], + + "@tanstack/query-core": ["@tanstack/query-core@5.100.14", "", {}, "sha512-5X41dGpxgeaHISCRW2oYwcSycZeULZzAunaudXT9ov1KOTj9xwt0CH6hbwqP1/z74ZWF7rYFnDpyYH07XFcZew=="], + + "@tanstack/react-query": ["@tanstack/react-query@5.100.14", "", { "dependencies": { "@tanstack/query-core": "5.100.14" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-oOr6aRdSFEwWhzxEkD/9ZcItM3+LjBSkeVmadWKwUssAHTsqd/7bOjWrX4AbvEkoEhgAxzN0Xk6H/aYzXiYBAw=="], + + "@tootallnate/quickjs-emscripten": ["@tootallnate/quickjs-emscripten@0.23.0", "", {}, "sha512-C5Mc6rdnsaJDjO3UpGW/CQTHtCKaYlScZTly4JIu97Jxo/odCiH0ITnDXSJPTOrEKk/ycSZ0AOgTmkDtkOsvIA=="], + + "@ts-morph/common": ["@ts-morph/common@0.27.0", "", { "dependencies": { "fast-glob": "^3.3.3", "minimatch": "^10.0.1", "path-browserify": "^1.0.1" } }, "sha512-Wf29UqxWDpc+i61k3oIOzcUfQt79PIT9y/MWfAGlrkjg6lBC1hwDECLXPVJAhWjiGbfBCxZd65F/LIZF3+jeJQ=="], + + "@tybys/wasm-util": ["@tybys/wasm-util@0.10.2", "", { "dependencies": { "tslib": "^2.4.0" } }, "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg=="], + + "@types/better-sqlite3": ["@types/better-sqlite3@7.6.13", "", { "dependencies": { "@types/node": "*" } }, "sha512-NMv9ASNARoKksWtsq/SHakpYAYnhBrQgGD8zkLYk/jaK8jUGn08CfEdTRgYhMypUQAfzSP8W6gNLe0q19/t4VA=="], + + "@types/bun": ["@types/bun@1.3.14", "", { "dependencies": { "bun-types": "1.3.14" } }, "sha512-h1hFqFVcvAvD9j9K7ZW7vd82aSA+rTdznZa+5bwvCwqSB1jmmfLcbIWhOLx1/+boy/xmjgCs/OMUL8hRJSmnPw=="], + + "@types/d3-array": ["@types/d3-array@3.2.2", "", {}, "sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw=="], + + "@types/d3-color": ["@types/d3-color@3.1.3", "", {}, "sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A=="], + + "@types/d3-ease": ["@types/d3-ease@3.0.2", "", {}, "sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA=="], + + "@types/d3-interpolate": ["@types/d3-interpolate@3.0.4", "", { "dependencies": { "@types/d3-color": "*" } }, "sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA=="], + + "@types/d3-path": ["@types/d3-path@3.1.1", "", {}, "sha512-VMZBYyQvbGmWyWVea0EHs/BwLgxc+MKi1zLDCONksozI4YJMcTt8ZEuIR4Sb1MMTE8MMW49v0IwI5+b7RmfWlg=="], + + "@types/d3-scale": ["@types/d3-scale@4.0.9", "", { "dependencies": { "@types/d3-time": "*" } }, "sha512-dLmtwB8zkAeO/juAMfnV+sItKjlsw2lKdZVVy6LRr0cBmegxSABiLEpGVmSJJ8O08i4+sGR6qQtb6WtuwJdvVw=="], + + "@types/d3-shape": ["@types/d3-shape@3.1.8", "", { "dependencies": { "@types/d3-path": "*" } }, "sha512-lae0iWfcDeR7qt7rA88BNiqdvPS5pFVPpo5OfjElwNaT2yyekbM0C9vK+yqBqEmHr6lDkRnYNoTBYlAgJa7a4w=="], + + "@types/d3-time": ["@types/d3-time@3.0.4", "", {}, "sha512-yuzZug1nkAAaBlBBikKZTgzCeA+k1uy4ZFwWANOfKw5z5LRhV0gNA7gNkKm7HoK+HRN0wX3EkxGk0fpbWhmB7g=="], + + "@types/d3-timer": ["@types/d3-timer@3.0.2", "", {}, "sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw=="], + + "@types/debug": ["@types/debug@4.1.13", "", { "dependencies": { "@types/ms": "*" } }, "sha512-KSVgmQmzMwPlmtljOomayoR89W4FynCAi3E8PPs7vmDVPe84hT+vGPKkJfThkmXs0x0jAaa9U8uW8bbfyS2fWw=="], + + "@types/estree": ["@types/estree@1.0.9", "", {}, "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg=="], + + "@types/estree-jsx": ["@types/estree-jsx@1.0.5", "", { "dependencies": { "@types/estree": "*" } }, "sha512-52CcUVNFyfb1A2ALocQw/Dd1BQFNmSdkuC3BkZ6iqhdMfQz7JWOFRuJFloOzjk+6WijU56m9oKXFAXc7o3Towg=="], + + "@types/hast": ["@types/hast@3.0.4", "", { "dependencies": { "@types/unist": "*" } }, "sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ=="], + + "@types/json-schema": ["@types/json-schema@7.0.15", "", {}, "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA=="], + + "@types/json5": ["@types/json5@0.0.29", "", {}, "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ=="], + + "@types/mdast": ["@types/mdast@4.0.4", "", { "dependencies": { "@types/unist": "*" } }, "sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA=="], + + "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="], + + "@types/node": ["@types/node@20.19.41", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-ECymXOukMnOoVkC2bb1Vc/w/836DXncOg5m8Xj1RH7xSHZJWNYY6Zh7EH477vcnD5egKNNfy2RpNOmuChhFPgQ=="], + + "@types/react": ["@types/react@19.2.15", "", { "dependencies": { "csstype": "^3.2.2" } }, "sha512-eRwcGNHve+E8qtEQSSRl6urh+rFop4v8gm6O8rGv25CodbvFdLjA1vVQ1KkiFE0w0UPOnb8tDiFKL5lp0rtY5Q=="], + + "@types/react-dom": ["@types/react-dom@19.2.3", "", { "peerDependencies": { "@types/react": "^19.2.0" } }, "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ=="], + + "@types/retry": ["@types/retry@0.12.0", "", {}, "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA=="], + + "@types/set-cookie-parser": ["@types/set-cookie-parser@2.4.10", "", { "dependencies": { "@types/node": "*" } }, "sha512-GGmQVGpQWUe5qglJozEjZV/5dyxbOOZ0LHe/lqyWssB88Y4svNfst0uqBVscdDeIKl5Jy5+aPSvy7mI9tYRguw=="], + + "@types/statuses": ["@types/statuses@2.0.6", "", {}, "sha512-xMAgYwceFhRA2zY+XbEA7mxYbA093wdiW8Vu6gZPGWy9cmOyU9XesH1tNcEWsKFd5Vzrqx5T3D38PWx1FIIXkA=="], + + "@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="], + + "@types/use-sync-external-store": ["@types/use-sync-external-store@0.0.6", "", {}, "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg=="], + + "@types/validate-npm-package-name": ["@types/validate-npm-package-name@4.0.2", "", {}, "sha512-lrpDziQipxCEeK5kWxvljWYhUvOiB2A9izZd9B2AFarYAkqZshb4lPbRs7zKEic6eGtH8V/2qJW+dPp9OtF6bw=="], + + "@types/yauzl": ["@types/yauzl@2.10.3", "", { "dependencies": { "@types/node": "*" } }, "sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q=="], + + "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.60.0", "", { "dependencies": { "@eslint-community/regexpp": "^4.12.2", "@typescript-eslint/scope-manager": "8.60.0", "@typescript-eslint/type-utils": "8.60.0", "@typescript-eslint/utils": "8.60.0", "@typescript-eslint/visitor-keys": "8.60.0", "ignore": "^7.0.5", "natural-compare": "^1.4.0", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.60.0", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-QYb/sa74/s7OKMbACMjrYnGspj9Hs5YI5aaffSL65UfeBUzVzBJfVo3oWSpbzPurvm7yaCCo2Lk7lVj610HqKw=="], + + "@typescript-eslint/parser": ["@typescript-eslint/parser@8.60.0", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.60.0", "@typescript-eslint/types": "8.60.0", "@typescript-eslint/typescript-estree": "8.60.0", "@typescript-eslint/visitor-keys": "8.60.0", "debug": "^4.4.3" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-fcqpj/MyK4sxDPcbe7STNPbpQL4RLZOPWuaTmwZYuc+hJKzRf58yRxfhqGpc6PIq9ZyfSBpfHgmUHmHs0KwHwg=="], + + "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.60.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.60.0", "@typescript-eslint/types": "^8.60.0", "debug": "^4.4.3" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-aZu74NNKJeUWqCjDddzdiKaS82dgYgV/vmf+Ui3ZdZejmgfXR/q+pRumgobnQ2cCJTgGTWp4ypiwsuofFubavg=="], + + "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.60.0", "", { "dependencies": { "@typescript-eslint/types": "8.60.0", "@typescript-eslint/visitor-keys": "8.60.0" } }, "sha512-pFzqhllJMs+jghLQWzV00ds39xLzuyqPSev5pd8f4Ir0rtKR3ZLUB4/4dhjOFighWb9larvtfJvqL+4yKDI3Xw=="], + + "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.60.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-BZPR3RGYlAXnly6ymAxfkVn5rCbZzQNou0rxv3GfWZ8cTQp+hhVd73khbGLAd8k1TlAPLISH337M+tAgAnaJDQ=="], + + "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.60.0", "", { "dependencies": { "@typescript-eslint/types": "8.60.0", "@typescript-eslint/typescript-estree": "8.60.0", "@typescript-eslint/utils": "8.60.0", "debug": "^4.4.3", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-SX46wEUtitCpq7AN38HkUU/+zvUpdKf7ephtWAFgckH8O7PQIyL5gvrhQgBLuEYgLfuKWOVvWVskMbuFHAz5xg=="], + + "@typescript-eslint/types": ["@typescript-eslint/types@8.60.0", "", {}, "sha512-AsE7x2XaAK+CVbeih0Fvbn+r1qHxtpLDJ3XUuFcIinT318T90yHMJC+Zgv+jUuDjQQd06HKwxnDu6sz1IcTilA=="], + + "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.60.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.60.0", "@typescript-eslint/tsconfig-utils": "8.60.0", "@typescript-eslint/types": "8.60.0", "@typescript-eslint/visitor-keys": "8.60.0", "debug": "^4.4.3", "minimatch": "^10.2.2", "semver": "^7.7.3", "tinyglobby": "^0.2.15", "ts-api-utils": "^2.5.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.1.0" } }, "sha512-3AcZNBGMClm6CXDyo8kYvVGT/sx29sS0oBsIb9oZI2gunA4Vm2M3YHzRLPvsUBBsl+yB5FPtltq7gGH0iTlp9g=="], + + "@typescript-eslint/utils": ["@typescript-eslint/utils@8.60.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.9.1", "@typescript-eslint/scope-manager": "8.60.0", "@typescript-eslint/types": "8.60.0", "@typescript-eslint/typescript-estree": "8.60.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-HtXuPfrHTyBDkameWpl+vJb1Uevu2tznAyahM1Oc4AENidCLTPiZDWIo4GfcxNdC/RcfGcadzzkqbRG87dUrQA=="], + + "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.60.0", "", { "dependencies": { "@typescript-eslint/types": "8.60.0", "eslint-visitor-keys": "^5.0.0" } }, "sha512-9WI52t8ZGLVGrPMBet25yAftqY/n95+zmoUUtJBBQTKDSKUu7OsPTroT2op7U9JatkoRccL0YkWDNMFfC4Sjxg=="], + + "@ungap/structured-clone": ["@ungap/structured-clone@1.3.1", "", {}, "sha512-mUFwbeTqrVgDQxFveS+df2yfap6iuP20NAKAsBt5jDEoOTDew+zwLAOilHCeQJOVSvmgCX4ogqIrA0mnyr08yQ=="], + + "@unrs/resolver-binding-android-arm-eabi": ["@unrs/resolver-binding-android-arm-eabi@1.12.2", "", { "os": "android", "cpu": "arm" }, "sha512-g5T90pqg1bo/7mytQx6F4iBNC0Wsh9cu+z9veDbFjc7HjpesJFWD7QMS0NGStXM075+7dJPPVvBbpZlnrdpi/w=="], + + "@unrs/resolver-binding-android-arm64": ["@unrs/resolver-binding-android-arm64@1.12.2", "", { "os": "android", "cpu": "arm64" }, "sha512-YGCRZv/9GLhwmz6mYDeTsm/92BAyR28l6c2ReweVW5pWgfsitWLY8upvfRlGdoyD8HjeTHSYJWyZGD4KJA/nFQ=="], + + "@unrs/resolver-binding-darwin-arm64": ["@unrs/resolver-binding-darwin-arm64@1.12.2", "", { "os": "darwin", "cpu": "arm64" }, "sha512-u9DiNT1auQMO20A9SyTuG3wUgQWB9Z7KjAg0uFuCDR1FsAY8A0CG2S6JpHS1xwm/w1G08bjXZDcyOCjv1WAm2w=="], + + "@unrs/resolver-binding-darwin-x64": ["@unrs/resolver-binding-darwin-x64@1.12.2", "", { "os": "darwin", "cpu": "x64" }, "sha512-f7rPLi/T1HVKZu/u6t87lroib16n8vrSzcyxI7lg4BGO9UF26KhQL44sd9eOUgrTYhvRXtWOIZT5PejdPyJfUA=="], + + "@unrs/resolver-binding-freebsd-x64": ["@unrs/resolver-binding-freebsd-x64@1.12.2", "", { "os": "freebsd", "cpu": "x64" }, "sha512-BpcOjWCJub6nRZUS2zA20pmLvjtqAtGejETaIyRLiZiQf++cbrjltLA5NN/xaXfqeOBOSlMFbemIl5/S5tljmg=="], + + "@unrs/resolver-binding-linux-arm-gnueabihf": ["@unrs/resolver-binding-linux-arm-gnueabihf@1.12.2", "", { "os": "linux", "cpu": "arm" }, "sha512-vZTDvdSISZjJx66OzJqtsOhzifbqRjbmI1Mnu49fQDwog5GtDI4QidRiEAYbZCRj9C8YZEW+3ZjqsyS9GR4k2A=="], + + "@unrs/resolver-binding-linux-arm-musleabihf": ["@unrs/resolver-binding-linux-arm-musleabihf@1.12.2", "", { "os": "linux", "cpu": "arm" }, "sha512-BiPI+IrIlwcW4nLLMM21+B1dFPzd55yAVgVGrdgDjNef+ch03GdxrcyaIz8X9SsQirh/kCQ7mviyWlMxdh2D7g=="], + + "@unrs/resolver-binding-linux-arm64-gnu": ["@unrs/resolver-binding-linux-arm64-gnu@1.12.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-zJc0H99FEPoFfSrNpa91HYfxzfAJCr502oxNK1cfdC9hlaFI43RT+JFCann9JUgZmLzzntChHyn13Sgn9ljHNg=="], + + "@unrs/resolver-binding-linux-arm64-musl": ["@unrs/resolver-binding-linux-arm64-musl@1.12.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-KQ3Lki6l+Pz1k/eBipN41ES+YUK30beLGb9YqcB1O542cyLCNE6GaxrfcY3T6EezmGGk84wb5XyO9loTM9tkcA=="], + + "@unrs/resolver-binding-linux-loong64-gnu": ["@unrs/resolver-binding-linux-loong64-gnu@1.12.2", "", { "os": "linux", "cpu": "none" }, "sha512-3SJGEh1DborhG6pyxvhPzCT4bbSIVihsvgJc13P1bHG7KLdNDaF9T3gsTwFc7Jw/5Y5/iWOjkEx7Zy0NvCGX3Q=="], + + "@unrs/resolver-binding-linux-loong64-musl": ["@unrs/resolver-binding-linux-loong64-musl@1.12.2", "", { "os": "linux", "cpu": "none" }, "sha512-jiuG/Obbel7uw1PwHNFfrkiKhLAF6mnyZ6aWlOAVN9WqKm8v0OFGnciJIHu8+CMvXLQ8AD51LPzAoUfT21D5Ew=="], + + "@unrs/resolver-binding-linux-ppc64-gnu": ["@unrs/resolver-binding-linux-ppc64-gnu@1.12.2", "", { "os": "linux", "cpu": "ppc64" }, "sha512-q7xRvVpmcfeL+LlZg8Pbbo6QaTZwDU5BaGZbwfhkEsXJn3Was8xYfE0RBH266xZt0rM6B7i8xAYIvjthuUIWHg=="], + + "@unrs/resolver-binding-linux-riscv64-gnu": ["@unrs/resolver-binding-linux-riscv64-gnu@1.12.2", "", { "os": "linux", "cpu": "none" }, "sha512-0CVdx6lcnT3Q9inOH8tsMIOJ6ImndllMjqJHg8RLVdB7Vq4SfkEXl9mCSsVNuNA4MCYycRicCUxPCabVHJRr6A=="], + + "@unrs/resolver-binding-linux-riscv64-musl": ["@unrs/resolver-binding-linux-riscv64-musl@1.12.2", "", { "os": "linux", "cpu": "none" }, "sha512-iOwlRo9vnp6R6ohHQS11n0NnfdXx/omhkocmIfaPRpQhKZ+3BDMkkdRVh53qjkFkpPddf+FETA28NwGN7l5l+w=="], + + "@unrs/resolver-binding-linux-s390x-gnu": ["@unrs/resolver-binding-linux-s390x-gnu@1.12.2", "", { "os": "linux", "cpu": "s390x" }, "sha512-HYJtLfXq94q8iZNFT1lknx258wlkkWhZeUXJRqzKBBUJ00CvZ+N33zgbCqimLjsyw5Va6uUxhVa12mI+kaveEw=="], + + "@unrs/resolver-binding-linux-x64-gnu": ["@unrs/resolver-binding-linux-x64-gnu@1.12.2", "", { "os": "linux", "cpu": "x64" }, "sha512-mPsUhunKKDih5O96Y6enDQyHc1SqBPlY1E/SfMWDM3EdJ95Z9CArPeCVwCCqbP45ljvivdEk8Fxn+SIb1rDAJQ=="], + + "@unrs/resolver-binding-linux-x64-musl": ["@unrs/resolver-binding-linux-x64-musl@1.12.2", "", { "os": "linux", "cpu": "x64" }, "sha512-azrt6+5ydLd8Vt210AAFis/lZevSfPw93EJRIJG+xPu4WCJ8K0kppCTpMyLPcKT7H15M4Jnt2tMp5bOvCkRC6A=="], + + "@unrs/resolver-binding-openharmony-arm64": ["@unrs/resolver-binding-openharmony-arm64@1.12.2", "", { "os": "none", "cpu": "arm64" }, "sha512-YZ9hP4O0X9PQb8eO980qmLNGH4zT3I9+SZTdt0Pr0YyuGQhYKoOZkV02VzrzyOZJ5xIJ3UFIenKkUkGg8GjgWQ=="], + + "@unrs/resolver-binding-wasm32-wasi": ["@unrs/resolver-binding-wasm32-wasi@1.12.2", "", { "dependencies": { "@emnapi/core": "1.10.0", "@emnapi/runtime": "1.10.0", "@napi-rs/wasm-runtime": "^1.1.4" }, "cpu": "none" }, "sha512-tYFDIkMxSflfEc/h92ZWNsZlHSwgimbNHSO3PL2JWQHfCuC2q316jMyYU9TIWZsFK2bQwyK5VAdYgn8ygPj69A=="], + + "@unrs/resolver-binding-win32-arm64-msvc": ["@unrs/resolver-binding-win32-arm64-msvc@1.12.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-qzNyg3xL0VPQmCaUh+N5jSitce6k+uCBfMDesWRnlULOZaqUkaJ0ybdT+UqlAWJoQjuqfIU/0Ptx9bteN4D82g=="], + + "@unrs/resolver-binding-win32-ia32-msvc": ["@unrs/resolver-binding-win32-ia32-msvc@1.12.2", "", { "os": "win32", "cpu": "ia32" }, "sha512-WD9sY00OfpHVGfsnHZoA8jVT+esS/Bg8z8jzxp5BnDCjjwsuKsPQrzswwpFy4J1AUJbXPRfkpcX0mXrzeXW79g=="], + + "@unrs/resolver-binding-win32-x64-msvc": ["@unrs/resolver-binding-win32-x64-msvc@1.12.2", "", { "os": "win32", "cpu": "x64" }, "sha512-nAB74NfSNKknqQ1RrYj6uz8FcXEomu/MATJZxh/x+BArzN2U3JbOYC0APYzUIGhVY3m5hRxA8VPNdPBoG8txlA=="], + + "@vercel/oidc": ["@vercel/oidc@3.2.0", "", {}, "sha512-UycprH3T6n3jH0k44NHMa7pnFHGu/N05MjojYr+Mc6I7obkoLIJujSWwin1pCvdy/eOxrI/l3uDLQsmcrOb4ug=="], + + "accepts": ["accepts@2.0.0", "", { "dependencies": { "mime-types": "^3.0.0", "negotiator": "^1.0.0" } }, "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng=="], + + "acorn": ["acorn@8.16.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw=="], + + "acorn-jsx": ["acorn-jsx@5.3.2", "", { "peerDependencies": { "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ=="], + + "agent-base": ["agent-base@7.1.4", "", {}, "sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ=="], + + "ai": ["ai@6.0.191", "", { "dependencies": { "@ai-sdk/gateway": "3.0.120", "@ai-sdk/provider": "3.0.10", "@ai-sdk/provider-utils": "4.0.27", "@opentelemetry/api": "^1.9.0" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-zAxvjKebQE7YkSyyNIl0OM7i6/zygnKeF+yNUjD4nWOelYrG+LpDd6RnH6mjySI4zUpZ7o4wbnmAy8jc6u98vQ=="], + + "ai-sdk-ollama": ["ai-sdk-ollama@3.8.4", "", { "dependencies": { "@ai-sdk/provider": "^3.0.10", "@ai-sdk/provider-utils": "^4.0.27", "jsonrepair": "^3.14.0", "ollama": "^0.6.3" }, "peerDependencies": { "ai": "^6.0.177" } }, "sha512-vvhLHo9MrOhDxsxRWfOGqBmQsnbDPKQHqDG5vSYMkdu6q8RPxJIYag001jQIAKA2MO37Nf96gCAPWM+dufvglw=="], + + "ajv": ["ajv@6.15.0", "", { "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", "json-schema-traverse": "^0.4.1", "uri-js": "^4.2.2" } }, "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw=="], + + "ajv-formats": ["ajv-formats@3.0.1", "", { "dependencies": { "ajv": "^8.0.0" } }, "sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ=="], + + "ansi-colors": ["ansi-colors@4.1.3", "", {}, "sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw=="], + + "ansi-regex": ["ansi-regex@6.2.2", "", {}, "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg=="], + + "ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="], + + "argparse": ["argparse@2.0.1", "", {}, "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q=="], + + "aria-query": ["aria-query@5.3.2", "", {}, "sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw=="], + + "array-buffer-byte-length": ["array-buffer-byte-length@1.0.2", "", { "dependencies": { "call-bound": "^1.0.3", "is-array-buffer": "^3.0.5" } }, "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw=="], + + "array-includes": ["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.0", "es-object-atoms": "^1.1.1", "get-intrinsic": "^1.3.0", "is-string": "^1.1.1", "math-intrinsics": "^1.1.0" } }, "sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ=="], + + "array.prototype.findlast": ["array.prototype.findlast@1.2.5", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-abstract": "^1.23.2", "es-errors": "^1.3.0", "es-object-atoms": "^1.0.0", "es-shim-unscopables": "^1.0.2" } }, "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ=="], + + "array.prototype.findlastindex": ["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.23.9", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "es-shim-unscopables": "^1.1.0" } }, "sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ=="], + + "array.prototype.flat": ["array.prototype.flat@1.3.3", "", { "dependencies": { "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-abstract": "^1.23.5", "es-shim-unscopables": "^1.0.2" } }, "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg=="], + + "array.prototype.flatmap": ["array.prototype.flatmap@1.3.3", "", { "dependencies": { "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-abstract": "^1.23.5", "es-shim-unscopables": "^1.0.2" } }, "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg=="], + + "array.prototype.tosorted": ["array.prototype.tosorted@1.1.4", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-abstract": "^1.23.3", "es-errors": "^1.3.0", "es-shim-unscopables": "^1.0.2" } }, "sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA=="], + + "arraybuffer.prototype.slice": ["arraybuffer.prototype.slice@1.0.4", "", { "dependencies": { "array-buffer-byte-length": "^1.0.1", "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-abstract": "^1.23.5", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.6", "is-array-buffer": "^3.0.4" } }, "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ=="], + + "ast-types": ["ast-types@0.16.1", "", { "dependencies": { "tslib": "^2.0.1" } }, "sha512-6t10qk83GOG8p0vKmaCr8eiilZwO171AvbROMtvvNiwrTly62t+7XkA8RdIIVbpMhCASAsxgAzdRSwh6nw/5Dg=="], + + "ast-types-flow": ["ast-types-flow@0.0.8", "", {}, "sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ=="], + + "async-function": ["async-function@1.0.0", "", {}, "sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA=="], + + "available-typed-arrays": ["available-typed-arrays@1.0.7", "", { "dependencies": { "possible-typed-array-names": "^1.0.0" } }, "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ=="], + + "axe-core": ["axe-core@4.11.4", "", {}, "sha512-KunSNx+TVpkAw/6ULfhnx+HWRecjqZGTOyquAoWHYLRSdK1tB5Ihce1ZW+UY3fj33bYAFWPu7W/GRSmmrCGuxA=="], + + "axobject-query": ["axobject-query@4.1.0", "", {}, "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ=="], + + "b4a": ["b4a@1.8.1", "", { "peerDependencies": { "react-native-b4a": "*" }, "optionalPeers": ["react-native-b4a"] }, "sha512-aiqre1Nr0B/6DgE2N5vwTc+2/oQZ4Wh1t4NznYY4E00y8LCt6NqdRv81so00oo27D8MVKTpUa/MwUUtBLXCoDw=="], + + "bail": ["bail@2.0.2", "", {}, "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw=="], + + "balanced-match": ["balanced-match@1.0.2", "", {}, "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw=="], + + "bare-events": ["bare-events@2.8.3", "", { "peerDependencies": { "bare-abort-controller": "*" }, "optionalPeers": ["bare-abort-controller"] }, "sha512-HdUm8EMQBLaJvGUdidNNbqpA1kYkwNcb+MYxkxCLAPJGQzlv9J0C24h8V65Z4c5GLd/JEALDvpFCQgpLJqc0zw=="], + + "bare-fs": ["bare-fs@4.7.1", "", { "dependencies": { "bare-events": "^2.5.4", "bare-path": "^3.0.0", "bare-stream": "^2.6.4", "bare-url": "^2.2.2", "fast-fifo": "^1.3.2" }, "peerDependencies": { "bare-buffer": "*" }, "optionalPeers": ["bare-buffer"] }, "sha512-WDRsyVN52eAx/lBamKD6uyw8H4228h/x0sGGGegOamM2cd7Pag88GfMQalobXI+HaEUxpCkbKQUDOQqt9wawRw=="], + + "bare-os": ["bare-os@3.9.1", "", {}, "sha512-6M5XjcnsygQNPMCMPXSK379xrJFiZ/AEMNBmFEmQW8d/789VQATvriyi5r0HYTL9TkQ26rn3kgdTG3aisbrXkQ=="], + + "bare-path": ["bare-path@3.0.0", "", { "dependencies": { "bare-os": "^3.0.1" } }, "sha512-tyfW2cQcB5NN8Saijrhqn0Zh7AnFNsnczRcuWODH0eYAXBsJ5gVxAUuNr7tsHSC6IZ77cA0SitzT+s47kot8Mw=="], + + "bare-stream": ["bare-stream@2.13.1", "", { "dependencies": { "streamx": "^2.25.0", "teex": "^1.0.1" }, "peerDependencies": { "bare-abort-controller": "*", "bare-buffer": "*", "bare-events": "*" }, "optionalPeers": ["bare-abort-controller", "bare-buffer", "bare-events"] }, "sha512-Vp0cnjYyrEC4whYTymQ+YZi6pBpfiICZO3cfRG8sy67ZNWe951urv1x4eW1BKNngw3U+3fPYb5JQvHbCtxH7Ow=="], + + "bare-url": ["bare-url@2.4.3", "", { "dependencies": { "bare-path": "^3.0.0" } }, "sha512-Kccpc7ACfXaxfeInfqKcZtW4pT5YBn1mesc4sCsun6sRwtbJ4h+sNOaksUpYEJUKfN65YWC6Bw2OJEFiKxq8nQ=="], + + "base64-js": ["base64-js@1.5.1", "", {}, "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="], + + "baseline-browser-mapping": ["baseline-browser-mapping@2.10.32", "", { "bin": { "baseline-browser-mapping": "dist/cli.cjs" } }, "sha512-wbPvpyjJPC0zdfdKXxqEL3Ea+bOMD/87X4lftiJkkaBiuG6ALQy1SLmEd7BSmVCuwCQsBrCamgBoLyfFDD1EPg=="], + + "basic-ftp": ["basic-ftp@5.3.1", "", {}, "sha512-bopVNp6ugyA150DDuZfPFdt1KZ5a94ZDiwX4hMgZDzF+GttD80lEy8kj98kbyhLXnPvhtIo93mdnLIjpCAeeOw=="], + + "better-sqlite3": ["better-sqlite3@12.10.0", "", { "dependencies": { "bindings": "^1.5.0", "prebuild-install": "^7.1.1" } }, "sha512-CyzaZRQKyHkB2ZInfTTl2nvT33EbDpjkLEbE8/Zck3Ll6O0qqvuGdrJ45HgtH+HykRg88ITY3AdreBGN70aBSQ=="], + + "bignumber.js": ["bignumber.js@9.3.1", "", {}, "sha512-Ko0uX15oIUS7wJ3Rb30Fs6SkVbLmPBAKdlm7q9+ak9bbIeFf0MwuBsQV6z7+X768/cHsfg+WlysDWJcmthjsjQ=="], + + "bindings": ["bindings@1.5.0", "", { "dependencies": { "file-uri-to-path": "1.0.0" } }, "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ=="], + + "bl": ["bl@4.1.0", "", { "dependencies": { "buffer": "^5.5.0", "inherits": "^2.0.4", "readable-stream": "^3.4.0" } }, "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w=="], + + "body-parser": ["body-parser@2.2.2", "", { "dependencies": { "bytes": "^3.1.2", "content-type": "^1.0.5", "debug": "^4.4.3", "http-errors": "^2.0.0", "iconv-lite": "^0.7.0", "on-finished": "^2.4.1", "qs": "^6.14.1", "raw-body": "^3.0.1", "type-is": "^2.0.1" } }, "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA=="], + + "brace-expansion": ["brace-expansion@1.1.15", "", { "dependencies": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" } }, "sha512-EwOCDEex4quD37XhqM3omwtMoJjr//isUZz1JopUNWms+4Z2ViyM/k1YIRePpoVNnQhENnxtFjLaxNHrT7xIUg=="], + + "braces": ["braces@3.0.3", "", { "dependencies": { "fill-range": "^7.1.1" } }, "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA=="], + + "browserslist": ["browserslist@4.28.2", "", { "dependencies": { "baseline-browser-mapping": "^2.10.12", "caniuse-lite": "^1.0.30001782", "electron-to-chromium": "^1.5.328", "node-releases": "^2.0.36", "update-browserslist-db": "^1.2.3" }, "bin": { "browserslist": "cli.js" } }, "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg=="], + + "buffer": ["buffer@5.7.1", "", { "dependencies": { "base64-js": "^1.3.1", "ieee754": "^1.1.13" } }, "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ=="], + + "buffer-crc32": ["buffer-crc32@0.2.13", "", {}, "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ=="], + + "buffer-equal-constant-time": ["buffer-equal-constant-time@1.0.1", "", {}, "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA=="], + + "buffer-from": ["buffer-from@1.1.2", "", {}, "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ=="], + + "bun-types": ["bun-types@1.3.14", "", { "dependencies": { "@types/node": "*" } }, "sha512-4N0ig0fEomHt5R0KCFWjovxow98rIoRwKolrYdCcknNwMekCXRnWEUvgu5soYV8QXtVsrUD8B95MBOZGPvr6KQ=="], + + "bundle-name": ["bundle-name@4.1.0", "", { "dependencies": { "run-applescript": "^7.0.0" } }, "sha512-tjwM5exMg6BGRI+kNmTntNsvdZS1X8BFYS6tnJ2hdH0kVxM6/eVZ2xy+FqStSWvYmtfFMDLIxurorHwDKfDz5Q=="], + + "bytes": ["bytes@3.1.2", "", {}, "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg=="], + + "call-bind": ["call-bind@1.0.9", "", { "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" } }, "sha512-a/hy+pNsFUTR+Iz8TCJvXudKVLAnz/DyeSUo10I5yvFDQJBFU2s9uqQpoSrJlroHUKoKqzg+epxyP9lqFdzfBQ=="], + + "call-bind-apply-helpers": ["call-bind-apply-helpers@1.0.2", "", { "dependencies": { "es-errors": "^1.3.0", "function-bind": "^1.1.2" } }, "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ=="], + + "call-bound": ["call-bound@1.0.4", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "get-intrinsic": "^1.3.0" } }, "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg=="], + + "callsites": ["callsites@3.1.0", "", {}, "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ=="], + + "caniuse-lite": ["caniuse-lite@1.0.30001793", "", {}, "sha512-iwSsYWaCOoh26cV8NwNRViHlrfUvYsHDfRVcbtmw0Kg6PJIZZXwMkj1442FYLBGkeUf1juAsU3DTfxW579mrPA=="], + + "ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="], + + "chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="], + + "character-entities": ["character-entities@2.0.2", "", {}, "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ=="], + + "character-entities-html4": ["character-entities-html4@2.1.0", "", {}, "sha512-1v7fgQRj6hnSwFpq1Eu0ynr/CDEw0rXo2B61qXrLNdHZmPKgb7fqS1a2JwF0rISo9q77jDI8VMEHoApn8qDoZA=="], + + "character-entities-legacy": ["character-entities-legacy@3.0.0", "", {}, "sha512-RpPp0asT/6ufRm//AJVwpViZbGM/MkjQFxJccQRHmISF/22NBtsHqAWmL+/pmkPWoIUJdWyeVleTl1wydHATVQ=="], + + "character-reference-invalid": ["character-reference-invalid@2.0.1", "", {}, "sha512-iBZ4F4wRbyORVsu0jPV7gXkOsGYjGHPmAyv+HiHG8gi5PtC9KI2j1+v8/tlibRvjoWX027ypmG/n0HtO5t7unw=="], + + "chownr": ["chownr@1.1.4", "", {}, "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg=="], + + "chromium-bidi": ["chromium-bidi@14.0.0", "", { "dependencies": { "mitt": "^3.0.1", "zod": "^3.24.1" }, "peerDependencies": { "devtools-protocol": "*" } }, "sha512-9gYlLtS6tStdRWzrtXaTMnqcM4dudNegMXJxkR0I/CXObHalYeYcAMPrL19eroNZHtJ8DQmu1E+ZNOYu/IXMXw=="], + + "class-variance-authority": ["class-variance-authority@0.7.1", "", { "dependencies": { "clsx": "^2.1.1" } }, "sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg=="], + + "cli-cursor": ["cli-cursor@5.0.0", "", { "dependencies": { "restore-cursor": "^5.0.0" } }, "sha512-aCj4O5wKyszjMmDT4tZj93kxyydN/K5zPWSCe6/0AV/AA1pqe5ZBIw0a2ZfPQV7lL5/yb5HsUreJ6UFAF1tEQw=="], + + "cli-spinners": ["cli-spinners@2.9.2", "", {}, "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg=="], + + "cli-width": ["cli-width@4.1.0", "", {}, "sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ=="], + + "client-only": ["client-only@0.0.1", "", {}, "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA=="], + + "cliui": ["cliui@8.0.1", "", { "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.1", "wrap-ansi": "^7.0.0" } }, "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ=="], + + "clsx": ["clsx@2.1.1", "", {}, "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA=="], + + "code-block-writer": ["code-block-writer@13.0.3", "", {}, "sha512-Oofo0pq3IKnsFtuHqSF7TqBfr71aeyZDVJ0HpmqB7FBM2qEigL0iPONSCZSO9pE9dZTAxANe5XHG9Uy0YMv8cg=="], + + "color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="], + + "color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="], + + "comma-separated-tokens": ["comma-separated-tokens@2.0.3", "", {}, "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg=="], + + "commander": ["commander@14.0.3", "", {}, "sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw=="], + + "concat-map": ["concat-map@0.0.1", "", {}, "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg=="], + + "content-disposition": ["content-disposition@1.1.0", "", {}, "sha512-5jRCH9Z/+DRP7rkvY83B+yGIGX96OYdJmzngqnw2SBSxqCFPd0w2km3s5iawpGX8krnwSGmF0FW5Nhr0Hfai3g=="], + + "content-type": ["content-type@1.0.5", "", {}, "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA=="], + + "convert-source-map": ["convert-source-map@2.0.0", "", {}, "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg=="], + + "cookie": ["cookie@1.1.1", "", {}, "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ=="], + + "cookie-signature": ["cookie-signature@1.2.2", "", {}, "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg=="], + + "cors": ["cors@2.8.6", "", { "dependencies": { "object-assign": "^4", "vary": "^1" } }, "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw=="], + + "cosmiconfig": ["cosmiconfig@9.0.1", "", { "dependencies": { "env-paths": "^2.2.1", "import-fresh": "^3.3.0", "js-yaml": "^4.1.0", "parse-json": "^5.2.0" }, "peerDependencies": { "typescript": ">=4.9.5" }, "optionalPeers": ["typescript"] }, "sha512-hr4ihw+DBqcvrsEDioRO31Z17x71pUYoNe/4h6Z0wB72p7MU7/9gH8Q3s12NFhHPfYBBOV3qyfUxmr/Yn3shnQ=="], + + "cross-spawn": ["cross-spawn@7.0.6", "", { "dependencies": { "path-key": "^3.1.0", "shebang-command": "^2.0.0", "which": "^2.0.1" } }, "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA=="], + + "cssesc": ["cssesc@3.0.0", "", { "bin": { "cssesc": "bin/cssesc" } }, "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg=="], + + "csstype": ["csstype@3.2.3", "", {}, "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ=="], + + "d3-array": ["d3-array@3.2.4", "", { "dependencies": { "internmap": "1 - 2" } }, "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg=="], + + "d3-color": ["d3-color@3.1.0", "", {}, "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA=="], + + "d3-ease": ["d3-ease@3.0.1", "", {}, "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w=="], + + "d3-format": ["d3-format@3.1.2", "", {}, "sha512-AJDdYOdnyRDV5b6ArilzCPPwc1ejkHcoyFarqlPqT7zRYjhavcT3uSrqcMvsgh2CgoPbK3RCwyHaVyxYcP2Arg=="], + + "d3-interpolate": ["d3-interpolate@3.0.1", "", { "dependencies": { "d3-color": "1 - 3" } }, "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g=="], + + "d3-path": ["d3-path@3.1.0", "", {}, "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ=="], + + "d3-scale": ["d3-scale@4.0.2", "", { "dependencies": { "d3-array": "2.10.0 - 3", "d3-format": "1 - 3", "d3-interpolate": "1.2.0 - 3", "d3-time": "2.1.1 - 3", "d3-time-format": "2 - 4" } }, "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ=="], + + "d3-shape": ["d3-shape@3.2.0", "", { "dependencies": { "d3-path": "^3.1.0" } }, "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA=="], + + "d3-time": ["d3-time@3.1.0", "", { "dependencies": { "d3-array": "2 - 3" } }, "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q=="], + + "d3-time-format": ["d3-time-format@4.1.0", "", { "dependencies": { "d3-time": "1 - 3" } }, "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg=="], + + "d3-timer": ["d3-timer@3.0.1", "", {}, "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA=="], + + "damerau-levenshtein": ["damerau-levenshtein@1.0.8", "", {}, "sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA=="], + + "data-uri-to-buffer": ["data-uri-to-buffer@4.0.1", "", {}, "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A=="], + + "data-view-buffer": ["data-view-buffer@1.0.2", "", { "dependencies": { "call-bound": "^1.0.3", "es-errors": "^1.3.0", "is-data-view": "^1.0.2" } }, "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ=="], + + "data-view-byte-length": ["data-view-byte-length@1.0.2", "", { "dependencies": { "call-bound": "^1.0.3", "es-errors": "^1.3.0", "is-data-view": "^1.0.2" } }, "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ=="], + + "data-view-byte-offset": ["data-view-byte-offset@1.0.1", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "is-data-view": "^1.0.1" } }, "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ=="], + + "debug": ["debug@4.4.3", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA=="], + + "decimal.js-light": ["decimal.js-light@2.5.1", "", {}, "sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg=="], + + "decode-named-character-reference": ["decode-named-character-reference@1.3.0", "", { "dependencies": { "character-entities": "^2.0.0" } }, "sha512-GtpQYB283KrPp6nRw50q3U9/VfOutZOe103qlN7BPP6Ad27xYnOIWv4lPzo8HCAL+mMZofJ9KEy30fq6MfaK6Q=="], + + "decompress-response": ["decompress-response@6.0.0", "", { "dependencies": { "mimic-response": "^3.1.0" } }, "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ=="], + + "dedent": ["dedent@1.7.2", "", { "peerDependencies": { "babel-plugin-macros": "^3.1.0" }, "optionalPeers": ["babel-plugin-macros"] }, "sha512-WzMx3mW98SN+zn3hgemf4OzdmyNhhhKz5Ay0pUfQiMQ3e1g+xmTJWp/pKdwKVXhdSkAEGIIzqeuWrL3mV/AXbA=="], + + "deep-extend": ["deep-extend@0.6.0", "", {}, "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA=="], + + "deep-is": ["deep-is@0.1.4", "", {}, "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ=="], + + "deepmerge": ["deepmerge@4.3.1", "", {}, "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A=="], + + "default-browser": ["default-browser@5.5.0", "", { "dependencies": { "bundle-name": "^4.1.0", "default-browser-id": "^5.0.0" } }, "sha512-H9LMLr5zwIbSxrmvikGuI/5KGhZ8E2zH3stkMgM5LpOWDutGM2JZaj460Udnf1a+946zc7YBgrqEWwbk7zHvGw=="], + + "default-browser-id": ["default-browser-id@5.0.1", "", {}, "sha512-x1VCxdX4t+8wVfd1so/9w+vQ4vx7lKd2Qp5tDRutErwmR85OgmfX7RlLRMWafRMY7hbEiXIbudNrjOAPa/hL8Q=="], + + "define-data-property": ["define-data-property@1.1.4", "", { "dependencies": { "es-define-property": "^1.0.0", "es-errors": "^1.3.0", "gopd": "^1.0.1" } }, "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A=="], + + "define-lazy-prop": ["define-lazy-prop@3.0.0", "", {}, "sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg=="], + + "define-properties": ["define-properties@1.2.1", "", { "dependencies": { "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" } }, "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg=="], + + "degenerator": ["degenerator@5.0.1", "", { "dependencies": { "ast-types": "^0.13.4", "escodegen": "^2.1.0", "esprima": "^4.0.1" } }, "sha512-TllpMR/t0M5sqCXfj85i4XaAzxmS5tVA16dqvdkMwGmzI+dXLXnw3J+3Vdv7VKw+ThlTMboK6i9rnZ6Nntj5CQ=="], + + "depd": ["depd@2.0.0", "", {}, "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="], + + "dequal": ["dequal@2.0.3", "", {}, "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA=="], + + "detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="], + + "devlop": ["devlop@1.1.0", "", { "dependencies": { "dequal": "^2.0.0" } }, "sha512-RWmIqhcFf1lRYBvNmr7qTNuyCt/7/ns2jbpp1+PalgE/rDQcBT0fioSMUpJ93irlUhC5hrg4cYqe6U+0ImW0rA=="], + + "devtools-protocol": ["devtools-protocol@0.0.1608973", "", {}, "sha512-Tpm17fxYzt+J7VrGdc1k8YdRqS3YV7se/M6KeemEqvUbq/n7At1rWVuXMxQgpWkdwSdIEKYbU//Bve+Shm4YNQ=="], + + "diff": ["diff@8.0.4", "", {}, "sha512-DPi0FmjiSU5EvQV0++GFDOJ9ASQUVFh5kD+OzOnYdi7n3Wpm9hWWGfB/O2blfHcMVTL5WkQXSnRiK9makhrcnw=="], + + "doctrine": ["doctrine@2.1.0", "", { "dependencies": { "esutils": "^2.0.2" } }, "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw=="], + + "dotenv": ["dotenv@17.4.2", "", {}, "sha512-nI4U3TottKAcAD9LLud4Cb7b2QztQMUEfHbvhTH09bqXTxnSie8WnjPALV/WMCrJZ6UV/qHJ6L03OqO3LcdYZw=="], + + "drizzle-kit": ["drizzle-kit@0.31.10", "", { "dependencies": { "@drizzle-team/brocli": "^0.10.2", "@esbuild-kit/esm-loader": "^2.5.5", "esbuild": "^0.25.4", "tsx": "^4.21.0" }, "bin": { "drizzle-kit": "bin.cjs" } }, "sha512-7OZcmQUrdGI+DUNNsKBn1aW8qSoKuTH7d0mYgSP8bAzdFzKoovxEFnoGQp2dVs82EOJeYycqRtciopszwUf8bw=="], + + "drizzle-orm": ["drizzle-orm@0.45.2", "", { "peerDependencies": { "@aws-sdk/client-rds-data": ">=3", "@cloudflare/workers-types": ">=4", "@electric-sql/pglite": ">=0.2.0", "@libsql/client": ">=0.10.0", "@libsql/client-wasm": ">=0.10.0", "@neondatabase/serverless": ">=0.10.0", "@op-engineering/op-sqlite": ">=2", "@opentelemetry/api": "^1.4.1", "@planetscale/database": ">=1.13", "@prisma/client": "*", "@tidbcloud/serverless": "*", "@types/better-sqlite3": "*", "@types/pg": "*", "@types/sql.js": "*", "@upstash/redis": ">=1.34.7", "@vercel/postgres": ">=0.8.0", "@xata.io/client": "*", "better-sqlite3": ">=7", "bun-types": "*", "expo-sqlite": ">=14.0.0", "gel": ">=2", "knex": "*", "kysely": "*", "mysql2": ">=2", "pg": ">=8", "postgres": ">=3", "sql.js": ">=1", "sqlite3": ">=5" }, "optionalPeers": ["@aws-sdk/client-rds-data", "@cloudflare/workers-types", "@electric-sql/pglite", "@libsql/client", "@libsql/client-wasm", "@neondatabase/serverless", "@op-engineering/op-sqlite", "@opentelemetry/api", "@planetscale/database", "@prisma/client", "@tidbcloud/serverless", "@types/better-sqlite3", "@types/pg", "@types/sql.js", "@upstash/redis", "@vercel/postgres", "@xata.io/client", "better-sqlite3", "bun-types", "expo-sqlite", "gel", "knex", "kysely", "mysql2", "pg", "postgres", "sql.js", "sqlite3"] }, "sha512-kY0BSaTNYWnoDMVoyY8uxmyHjpJW1geOmBMdSSicKo9CIIWkSxMIj2rkeSR51b8KAPB7m+qysjuHme5nKP+E5Q=="], + + "dunder-proto": ["dunder-proto@1.0.1", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.1", "es-errors": "^1.3.0", "gopd": "^1.2.0" } }, "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A=="], + + "ecdsa-sig-formatter": ["ecdsa-sig-formatter@1.0.11", "", { "dependencies": { "safe-buffer": "^5.0.1" } }, "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ=="], + + "eciesjs": ["eciesjs@0.4.18", "", { "dependencies": { "@ecies/ciphers": "^0.2.5", "@noble/ciphers": "^1.3.0", "@noble/curves": "^1.9.7", "@noble/hashes": "^1.8.0" } }, "sha512-wG99Zcfcys9fZux7Cft8BAX/YrOJLJSZ3jyYPfhZHqN2E+Ffx+QXBDsv3gubEgPtV6dTzJMSQUwk1H98/t/0wQ=="], + + "ee-first": ["ee-first@1.1.1", "", {}, "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="], + + "electron-to-chromium": ["electron-to-chromium@1.5.362", "", {}, "sha512-PUY2DrLvkjkUuWqq+KPL2iWshrJsZOcIojzRQ7eXFacc9dWga7MGMJAa15VbiejSZB1PAXaRLAiKgruHP8LB1w=="], + + "emoji-regex": ["emoji-regex@9.2.2", "", {}, "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg=="], + + "encodeurl": ["encodeurl@2.0.0", "", {}, "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg=="], + + "end-of-stream": ["end-of-stream@1.4.5", "", { "dependencies": { "once": "^1.4.0" } }, "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg=="], + + "enhanced-resolve": ["enhanced-resolve@5.22.0", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.3.3" } }, "sha512-xYcDWrpELkFzz9SpZ3PlI6Eu6eD93Yf0WLDRxikGhWJ3MAir2SNZTIVCVZqZ/NUyx8AdMc2gT9C0gPiw18kG+A=="], + + "enquirer": ["enquirer@2.4.1", "", { "dependencies": { "ansi-colors": "^4.1.1", "strip-ansi": "^6.0.1" } }, "sha512-rRqJg/6gd538VHvR3PSrdRBb/1Vy2YfzHqzvbhGIQpDRKIa4FgV/54b5Q1xYSxOOwKvjXweS26E0Q+nAMwp2pQ=="], + + "env-paths": ["env-paths@2.2.1", "", {}, "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A=="], + + "error-ex": ["error-ex@1.3.4", "", { "dependencies": { "is-arrayish": "^0.2.1" } }, "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ=="], + + "es-abstract": ["es-abstract@1.24.2", "", { "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.19" } }, "sha512-2FpH9Q5i2RRwyEP1AylXe6nYLR5OhaJTZwmlcP0dL/+JCbgg7yyEo/sEK6HeGZRf3dFpWwThaRHVApXSkW3xeg=="], + + "es-define-property": ["es-define-property@1.0.1", "", {}, "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g=="], + + "es-errors": ["es-errors@1.3.0", "", {}, "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw=="], + + "es-iterator-helpers": ["es-iterator-helpers@1.3.2", "", { "dependencies": { "call-bind": "^1.0.9", "call-bound": "^1.0.4", "define-properties": "^1.2.1", "es-abstract": "^1.24.2", "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", "math-intrinsics": "^1.1.0" } }, "sha512-HVLACW1TppGYjJ8H6/jqH/pqOtKRw6wMlrB23xfExmFWxFquAIWCmwoLsOyN96K4a5KbmOf5At9ZUO3GZbetAw=="], + + "es-object-atoms": ["es-object-atoms@1.1.2", "", { "dependencies": { "es-errors": "^1.3.0" } }, "sha512-HWcBoN6NileqtSydK2FqHbS/LoDd2pqrnQHLyJzBj4kOp/ky2MWMN694xOfkK8/SnUsW2DH7EfyVlydKCsm1Zw=="], + + "es-set-tostringtag": ["es-set-tostringtag@2.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "get-intrinsic": "^1.2.6", "has-tostringtag": "^1.0.2", "hasown": "^2.0.2" } }, "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA=="], + + "es-shim-unscopables": ["es-shim-unscopables@1.1.0", "", { "dependencies": { "hasown": "^2.0.2" } }, "sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw=="], + + "es-to-primitive": ["es-to-primitive@1.3.0", "", { "dependencies": { "is-callable": "^1.2.7", "is-date-object": "^1.0.5", "is-symbol": "^1.0.4" } }, "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g=="], + + "es-toolkit": ["es-toolkit@1.47.0", "", {}, "sha512-n1GuoD0WEQZMBk5tttoZSqwgyLx01oqa5XsBmCHwPyNe1S9jPBEmtR2pSgp2kJuWE3ciFZ6yRHmY4pM4C3OOkw=="], + + "esbuild": ["esbuild@0.25.12", "", { "optionalDependencies": { "@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" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg=="], + + "escalade": ["escalade@3.2.0", "", {}, "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA=="], + + "escape-html": ["escape-html@1.0.3", "", {}, "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="], + + "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="], + + "escodegen": ["escodegen@2.1.0", "", { "dependencies": { "esprima": "^4.0.1", "estraverse": "^5.2.0", "esutils": "^2.0.2" }, "optionalDependencies": { "source-map": "~0.6.1" }, "bin": { "esgenerate": "bin/esgenerate.js", "escodegen": "bin/escodegen.js" } }, "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w=="], + + "eslint": ["eslint@9.39.4", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.8.0", "@eslint-community/regexpp": "^4.12.1", "@eslint/config-array": "^0.21.2", "@eslint/config-helpers": "^0.4.2", "@eslint/core": "^0.17.0", "@eslint/eslintrc": "^3.3.5", "@eslint/js": "9.39.4", "@eslint/plugin-kit": "^0.4.1", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", "@types/estree": "^1.0.6", "ajv": "^6.14.0", "chalk": "^4.0.0", "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", "eslint-scope": "^8.4.0", "eslint-visitor-keys": "^4.2.1", "espree": "^10.4.0", "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", "json-stable-stringify-without-jsonify": "^1.0.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.5", "natural-compare": "^1.4.0", "optionator": "^0.9.3" }, "peerDependencies": { "jiti": "*" }, "optionalPeers": ["jiti"], "bin": { "eslint": "bin/eslint.js" } }, "sha512-XoMjdBOwe/esVgEvLmNsD3IRHkm7fbKIUGvrleloJXUZgDHig2IPWNniv+GwjyJXzuNqVjlr5+4yVUZjycJwfQ=="], + + "eslint-config-next": ["eslint-config-next@16.2.6", "", { "dependencies": { "@next/eslint-plugin-next": "16.2.6", "eslint-import-resolver-node": "^0.3.6", "eslint-import-resolver-typescript": "^3.5.2", "eslint-plugin-import": "^2.32.0", "eslint-plugin-jsx-a11y": "^6.10.0", "eslint-plugin-react": "^7.37.0", "eslint-plugin-react-hooks": "^7.0.0", "globals": "16.4.0", "typescript-eslint": "^8.46.0" }, "peerDependencies": { "eslint": ">=9.0.0", "typescript": ">=3.3.1" }, "optionalPeers": ["typescript"] }, "sha512-z2ELYSkyrrJ6cuunTU8vhsT/RpouPkjaSah06nVW6Rg2Hpg0Vs8s497/e5s8G8qtdp4ccsiovz5P1rv+5VSW2Q=="], + + "eslint-import-resolver-node": ["eslint-import-resolver-node@0.3.10", "", { "dependencies": { "debug": "^3.2.7", "is-core-module": "^2.16.1", "resolve": "^2.0.0-next.6" } }, "sha512-tRrKqFyCaKict5hOd244sL6EQFNycnMQnBe+j8uqGNXYzsImGbGUU4ibtoaBmv5FLwJwcFJNeg1GeVjQfbMrDQ=="], + + "eslint-import-resolver-typescript": ["eslint-import-resolver-typescript@3.10.1", "", { "dependencies": { "@nolyfill/is-core-module": "1.0.39", "debug": "^4.4.0", "get-tsconfig": "^4.10.0", "is-bun-module": "^2.0.0", "stable-hash": "^0.0.5", "tinyglobby": "^0.2.13", "unrs-resolver": "^1.6.2" }, "peerDependencies": { "eslint": "*", "eslint-plugin-import": "*", "eslint-plugin-import-x": "*" }, "optionalPeers": ["eslint-plugin-import", "eslint-plugin-import-x"] }, "sha512-A1rHYb06zjMGAxdLSkN2fXPBwuSaQ0iO5M/hdyS0Ajj1VBaRp0sPD3dn1FhME3c/JluGFbwSxyCfqdSbtQLAHQ=="], + + "eslint-module-utils": ["eslint-module-utils@2.12.1", "", { "dependencies": { "debug": "^3.2.7" } }, "sha512-L8jSWTze7K2mTg0vos/RuLRS5soomksDPoJLXIslC7c8Wmut3bx7CPpJijDcBZtxQ5lrbUdM+s0OlNbz0DCDNw=="], + + "eslint-plugin-import": ["eslint-plugin-import@2.32.0", "", { "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-import-resolver-node": "^0.3.9", "eslint-module-utils": "^2.12.1", "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" }, "peerDependencies": { "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9" } }, "sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA=="], + + "eslint-plugin-jsx-a11y": ["eslint-plugin-jsx-a11y@6.10.2", "", { "dependencies": { "aria-query": "^5.3.2", "array-includes": "^3.1.8", "array.prototype.flatmap": "^1.3.2", "ast-types-flow": "^0.0.8", "axe-core": "^4.10.0", "axobject-query": "^4.1.0", "damerau-levenshtein": "^1.0.8", "emoji-regex": "^9.2.2", "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.0.3", "string.prototype.includes": "^2.0.1" }, "peerDependencies": { "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9" } }, "sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q=="], + + "eslint-plugin-react": ["eslint-plugin-react@7.37.5", "", { "dependencies": { "array-includes": "^3.1.8", "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.1", "estraverse": "^5.3.0", "hasown": "^2.0.2", "jsx-ast-utils": "^2.4.1 || ^3.0.0", "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" }, "peerDependencies": { "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7" } }, "sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA=="], + + "eslint-plugin-react-hooks": ["eslint-plugin-react-hooks@7.1.1", "", { "dependencies": { "@babel/core": "^7.24.4", "@babel/parser": "^7.24.4", "hermes-parser": "^0.25.1", "zod": "^3.25.0 || ^4.0.0", "zod-validation-error": "^3.5.0 || ^4.0.0" }, "peerDependencies": { "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 || ^10.0.0" } }, "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g=="], + + "eslint-scope": ["eslint-scope@8.4.0", "", { "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg=="], + + "eslint-visitor-keys": ["eslint-visitor-keys@4.2.1", "", {}, "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ=="], + + "espree": ["espree@10.4.0", "", { "dependencies": { "acorn": "^8.15.0", "acorn-jsx": "^5.3.2", "eslint-visitor-keys": "^4.2.1" } }, "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ=="], + + "esprima": ["esprima@4.0.1", "", { "bin": { "esparse": "./bin/esparse.js", "esvalidate": "./bin/esvalidate.js" } }, "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A=="], + + "esquery": ["esquery@1.7.0", "", { "dependencies": { "estraverse": "^5.1.0" } }, "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g=="], + + "esrecurse": ["esrecurse@4.3.0", "", { "dependencies": { "estraverse": "^5.2.0" } }, "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag=="], + + "estraverse": ["estraverse@5.3.0", "", {}, "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA=="], + + "estree-util-is-identifier-name": ["estree-util-is-identifier-name@3.0.0", "", {}, "sha512-hFtqIDZTIUZ9BXLb8y4pYGyk6+wekIivNVTcmvk8NoOh+VeRn5y6cEHzbURrWbfp1fIqdVipilzj+lfaadNZmg=="], + + "esutils": ["esutils@2.0.3", "", {}, "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g=="], + + "etag": ["etag@1.8.1", "", {}, "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg=="], + + "eventemitter3": ["eventemitter3@5.0.4", "", {}, "sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw=="], + + "events-universal": ["events-universal@1.0.1", "", { "dependencies": { "bare-events": "^2.7.0" } }, "sha512-LUd5euvbMLpwOF8m6ivPCbhQeSiYVNb8Vs0fQ8QjXo0JTkEHpz8pxdQf0gStltaPpw0Cca8b39KxvK9cfKRiAw=="], + + "eventsource": ["eventsource@3.0.7", "", { "dependencies": { "eventsource-parser": "^3.0.1" } }, "sha512-CRT1WTyuQoD771GW56XEZFQ/ZoSfWid1alKGDYMmkt2yl8UXrVR4pspqWNEcqKvVIzg6PAltWjxcSSPrboA4iA=="], + + "eventsource-parser": ["eventsource-parser@3.0.8", "", {}, "sha512-70QWGkr4snxr0OXLRWsFLeRBIRPuQOvt4s8QYjmUlmlkyTZkRqS7EDVRZtzU3TiyDbXSzaOeF0XUKy8PchzukQ=="], + + "execa": ["execa@9.6.1", "", { "dependencies": { "@sindresorhus/merge-streams": "^4.0.0", "cross-spawn": "^7.0.6", "figures": "^6.1.0", "get-stream": "^9.0.0", "human-signals": "^8.0.1", "is-plain-obj": "^4.1.0", "is-stream": "^4.0.1", "npm-run-path": "^6.0.0", "pretty-ms": "^9.2.0", "signal-exit": "^4.1.0", "strip-final-newline": "^4.0.0", "yoctocolors": "^2.1.1" } }, "sha512-9Be3ZoN4LmYR90tUoVu2te2BsbzHfhJyfEiAVfz7N5/zv+jduIfLrV2xdQXOHbaD6KgpGdO9PRPM1Y4Q9QkPkA=="], + + "expand-template": ["expand-template@2.0.3", "", {}, "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg=="], + + "express": ["express@5.2.1", "", { "dependencies": { "accepts": "^2.0.0", "body-parser": "^2.2.1", "content-disposition": "^1.0.0", "content-type": "^1.0.5", "cookie": "^0.7.1", "cookie-signature": "^1.2.1", "debug": "^4.4.0", "depd": "^2.0.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "finalhandler": "^2.1.0", "fresh": "^2.0.0", "http-errors": "^2.0.0", "merge-descriptors": "^2.0.0", "mime-types": "^3.0.0", "on-finished": "^2.4.1", "once": "^1.4.0", "parseurl": "^1.3.3", "proxy-addr": "^2.0.7", "qs": "^6.14.0", "range-parser": "^1.2.1", "router": "^2.2.0", "send": "^1.1.0", "serve-static": "^2.2.0", "statuses": "^2.0.1", "type-is": "^2.0.1", "vary": "^1.1.2" } }, "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw=="], + + "express-rate-limit": ["express-rate-limit@8.5.2", "", { "dependencies": { "ip-address": "^10.2.0" }, "peerDependencies": { "express": ">= 4.11" } }, "sha512-5Kb34ipNX694DH48vN9irak1Qx30nb0PLYHXfJgw4YEjiC3ZEmZJhwOp+VfiCYwFzvFTdB9QkArYS5kXa2cx2A=="], + + "extend": ["extend@3.0.2", "", {}, "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="], + + "extract-zip": ["extract-zip@2.0.1", "", { "dependencies": { "debug": "^4.1.1", "get-stream": "^5.1.0", "yauzl": "^2.10.0" }, "optionalDependencies": { "@types/yauzl": "^2.9.1" }, "bin": { "extract-zip": "cli.js" } }, "sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg=="], + + "fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="], + + "fast-fifo": ["fast-fifo@1.3.2", "", {}, "sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ=="], + + "fast-glob": ["fast-glob@3.3.3", "", { "dependencies": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", "glob-parent": "^5.1.2", "merge2": "^1.3.0", "micromatch": "^4.0.8" } }, "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg=="], + + "fast-json-stable-stringify": ["fast-json-stable-stringify@2.1.0", "", {}, "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw=="], + + "fast-levenshtein": ["fast-levenshtein@2.0.6", "", {}, "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw=="], + + "fast-sha256": ["fast-sha256@1.3.0", "", {}, "sha512-n11RGP/lrWEFI/bWdygLxhI+pVeo1ZYIVwvvPkW7azl/rOy+F3HYRZ2K5zeE9mmkhQppyv9sQFx0JM9UabnpPQ=="], + + "fast-string-truncated-width": ["fast-string-truncated-width@3.0.3", "", {}, "sha512-0jjjIEL6+0jag3l2XWWizO64/aZVtpiGE3t0Zgqxv0DPuxiMjvB3M24fCyhZUO4KomJQPj3LTSUnDP3GpdwC0g=="], + + "fast-string-width": ["fast-string-width@3.0.2", "", { "dependencies": { "fast-string-truncated-width": "^3.0.2" } }, "sha512-gX8LrtNEI5hq8DVUfRQMbr5lpaS4nMIWV+7XEbXk2b8kiQIizgnlr12B4dA3ZEx3308ze0O4Q1R+cHts8kyUJg=="], + + "fast-uri": ["fast-uri@3.1.2", "", {}, "sha512-rVjf7ArG3LTk+FS6Yw81V1DLuZl1bRbNrev6Tmd/9RaroeeRRJhAt7jg/6YFxbvAQXUCavSoZhPPj6oOx+5KjQ=="], + + "fast-wrap-ansi": ["fast-wrap-ansi@0.2.2", "", { "dependencies": { "fast-string-width": "^3.0.2" } }, "sha512-7F2Fl+TjRSenLqlU3UjSH0iyqopqoZIu7eZVpEirP2g1GtWa2G/ecEmBdgz31+Mxr+ELclgg6sokpSFIQiZ02Q=="], + + "fastq": ["fastq@1.20.1", "", { "dependencies": { "reusify": "^1.0.4" } }, "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw=="], + + "fd-package-json": ["fd-package-json@2.0.0", "", { "dependencies": { "walk-up-path": "^4.0.0" } }, "sha512-jKmm9YtsNXN789RS/0mSzOC1NUq9mkVd65vbSSVsKdjGvYXBuE4oWe2QOEoFeRmJg+lPuZxpmrfFclNhoRMneQ=="], + + "fd-slicer": ["fd-slicer@1.1.0", "", { "dependencies": { "pend": "~1.2.0" } }, "sha512-cE1qsB/VwyQozZ+q1dGxR8LBYNZeofhEdUNGSMbQD3Gw2lAzX9Zb3uIU6Ebc/Fmyjo9AWWfnn0AUCHqtevs/8g=="], + + "fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="], + + "fetch-blob": ["fetch-blob@3.2.0", "", { "dependencies": { "node-domexception": "^1.0.0", "web-streams-polyfill": "^3.0.3" } }, "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ=="], + + "figures": ["figures@6.1.0", "", { "dependencies": { "is-unicode-supported": "^2.0.0" } }, "sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg=="], + + "file-entry-cache": ["file-entry-cache@8.0.0", "", { "dependencies": { "flat-cache": "^4.0.0" } }, "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ=="], + + "file-uri-to-path": ["file-uri-to-path@1.0.0", "", {}, "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw=="], + + "fill-range": ["fill-range@7.1.1", "", { "dependencies": { "to-regex-range": "^5.0.1" } }, "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg=="], + + "finalhandler": ["finalhandler@2.1.1", "", { "dependencies": { "debug": "^4.4.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "on-finished": "^2.4.1", "parseurl": "^1.3.3", "statuses": "^2.0.1" } }, "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA=="], + + "find-up": ["find-up@5.0.0", "", { "dependencies": { "locate-path": "^6.0.0", "path-exists": "^4.0.0" } }, "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng=="], + + "flat-cache": ["flat-cache@4.0.1", "", { "dependencies": { "flatted": "^3.2.9", "keyv": "^4.5.4" } }, "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw=="], + + "flatted": ["flatted@3.4.2", "", {}, "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA=="], + + "for-each": ["for-each@0.3.5", "", { "dependencies": { "is-callable": "^1.2.7" } }, "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg=="], + + "formatly": ["formatly@0.3.0", "", { "dependencies": { "fd-package-json": "^2.0.0" }, "bin": { "formatly": "bin/index.mjs" } }, "sha512-9XNj/o4wrRFyhSMJOvsuyMwy8aUfBaZ1VrqHVfohyXf0Sw0e+yfKG+xZaY3arGCOMdwFsqObtzVOc1gU9KiT9w=="], + + "formdata-polyfill": ["formdata-polyfill@4.0.10", "", { "dependencies": { "fetch-blob": "^3.1.2" } }, "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g=="], + + "forwarded": ["forwarded@0.2.0", "", {}, "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="], + + "framer-motion": ["framer-motion@12.40.0", "", { "dependencies": { "motion-dom": "^12.40.0", "motion-utils": "^12.39.0", "tslib": "^2.4.0" }, "peerDependencies": { "@emotion/is-prop-valid": "*", "react": "^18.0.0 || ^19.0.0", "react-dom": "^18.0.0 || ^19.0.0" }, "optionalPeers": ["@emotion/is-prop-valid", "react", "react-dom"] }, "sha512-uaBd3qC1v3KQqBEjwTUd183K6PbS+j0yR9w9VmEOLWA/tnUcSn8Xa3uck7t4dgpDoUss8xQTcj8W2L07lrnLFg=="], + + "fresh": ["fresh@2.0.0", "", {}, "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A=="], + + "fs-constants": ["fs-constants@1.0.0", "", {}, "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow=="], + + "fs-extra": ["fs-extra@11.3.5", "", { "dependencies": { "graceful-fs": "^4.2.0", "jsonfile": "^6.0.1", "universalify": "^2.0.0" } }, "sha512-eKpRKAovdpZtR1WopLHxlBWvAgPny3c4gX1G5Jhwmmw4XJj0ifSD5qB5TOo8hmA0wlRKDAOAhEE1yVPgs6Fgcg=="], + + "fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="], + + "function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="], + + "function.prototype.name": ["function.prototype.name@1.1.8", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "define-properties": "^1.2.1", "functions-have-names": "^1.2.3", "hasown": "^2.0.2", "is-callable": "^1.2.7" } }, "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q=="], + + "functions-have-names": ["functions-have-names@1.2.3", "", {}, "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ=="], + + "fuzzysort": ["fuzzysort@3.1.0", "", {}, "sha512-sR9BNCjBg6LNgwvxlBd0sBABvQitkLzoVY9MYYROQVX/FvfJ4Mai9LsGhDgd8qYdds0bY77VzYd5iuB+v5rwQQ=="], + + "gaxios": ["gaxios@7.1.4", "", { "dependencies": { "extend": "^3.0.2", "https-proxy-agent": "^7.0.1", "node-fetch": "^3.3.2" } }, "sha512-bTIgTsM2bWn3XklZISBTQX7ZSddGW+IO3bMdGaemHZ3tbqExMENHLx6kKZ/KlejgrMtj8q7wBItt51yegqalrA=="], + + "gcp-metadata": ["gcp-metadata@8.1.2", "", { "dependencies": { "gaxios": "^7.0.0", "google-logging-utils": "^1.0.0", "json-bigint": "^1.0.0" } }, "sha512-zV/5HKTfCeKWnxG0Dmrw51hEWFGfcF2xiXqcA3+J90WDuP0SvoiSO5ORvcBsifmx/FoIjgQN3oNOGaQ5PhLFkg=="], + + "generator-function": ["generator-function@2.0.1", "", {}, "sha512-SFdFmIJi+ybC0vjlHN0ZGVGHc3lgE0DxPAT0djjVg+kjOnSqclqmj0KQ7ykTOLP6YxoqOvuAODGdcHJn+43q3g=="], + + "gensync": ["gensync@1.0.0-beta.2", "", {}, "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg=="], + + "get-caller-file": ["get-caller-file@2.0.5", "", {}, "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg=="], + + "get-east-asian-width": ["get-east-asian-width@1.6.0", "", {}, "sha512-QRbvDIbx6YklUe6RxeTeleMR0yv3cYH6PsPZHcnVn7xv7zO1BHN8r0XETu8n6Ye3Q+ahtSarc3WgtNWmehIBfA=="], + + "get-intrinsic": ["get-intrinsic@1.3.0", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", "math-intrinsics": "^1.1.0" } }, "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ=="], + + "get-own-enumerable-keys": ["get-own-enumerable-keys@1.0.0", "", {}, "sha512-PKsK2FSrQCyxcGHsGrLDcK0lx+0Ke+6e8KFFozA9/fIQLhQzPaRvJFdcz7+Axg3jUH/Mq+NI4xa5u/UT2tQskA=="], + + "get-proto": ["get-proto@1.0.1", "", { "dependencies": { "dunder-proto": "^1.0.1", "es-object-atoms": "^1.0.0" } }, "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g=="], + + "get-stream": ["get-stream@9.0.1", "", { "dependencies": { "@sec-ant/readable-stream": "^0.4.1", "is-stream": "^4.0.1" } }, "sha512-kVCxPF3vQM/N0B1PmoqVUqgHP+EeVjmZSQn+1oCRPxd2P21P2F19lIgbR3HBosbB1PUhOAoctJnfEn2GbN2eZA=="], + + "get-symbol-description": ["get-symbol-description@1.1.0", "", { "dependencies": { "call-bound": "^1.0.3", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.6" } }, "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg=="], + + "get-tsconfig": ["get-tsconfig@4.14.0", "", { "dependencies": { "resolve-pkg-maps": "^1.0.0" } }, "sha512-yTb+8DXzDREzgvYmh6s9vHsSVCHeC0G3PI5bEXNBHtmshPnO+S5O7qgLEOn0I5QvMy6kpZN8K1NKGyilLb93wA=="], + + "get-uri": ["get-uri@6.0.5", "", { "dependencies": { "basic-ftp": "^5.0.2", "data-uri-to-buffer": "^6.0.2", "debug": "^4.3.4" } }, "sha512-b1O07XYq8eRuVzBNgJLstU6FYc1tS6wnMtF1I1D9lE8LxZSOGZ7LhxN54yPP6mGw5f2CkXY2BQUL9Fx41qvcIg=="], + + "github-from-package": ["github-from-package@0.0.0", "", {}, "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw=="], + + "glob-parent": ["glob-parent@6.0.2", "", { "dependencies": { "is-glob": "^4.0.3" } }, "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A=="], + + "globals": ["globals@16.4.0", "", {}, "sha512-ob/2LcVVaVGCYN+r14cnwnoDPUufjiYgSqRhiFD0Q1iI4Odora5RE8Iv1D24hAz5oMophRGkGz+yuvQmmUMnMw=="], + + "globalthis": ["globalthis@1.0.4", "", { "dependencies": { "define-properties": "^1.2.1", "gopd": "^1.0.1" } }, "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ=="], + + "google-auth-library": ["google-auth-library@10.6.2", "", { "dependencies": { "base64-js": "^1.3.0", "ecdsa-sig-formatter": "^1.0.11", "gaxios": "^7.1.4", "gcp-metadata": "8.1.2", "google-logging-utils": "1.1.3", "jws": "^4.0.0" } }, "sha512-e27Z6EThmVNNvtYASwQxose/G57rkRuaRbQyxM2bvYLLX/GqWZ5chWq2EBoUchJbCc57eC9ArzO5wMsEmWftCw=="], + + "google-logging-utils": ["google-logging-utils@1.1.3", "", {}, "sha512-eAmLkjDjAFCVXg7A1unxHsLf961m6y17QFqXqAXGj/gVkKFrEICfStRfwUlGNfeCEjNRa32JEWOUTlYXPyyKvA=="], + + "gopd": ["gopd@1.2.0", "", {}, "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg=="], + + "graceful-fs": ["graceful-fs@4.2.11", "", {}, "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ=="], + + "graphql": ["graphql@16.14.0", "", {}, "sha512-BBvQ/406p+4CZbTpCbVPSxfzrZrbnuWSP1ELYgyS6B+hNeKzgrdB4JczCa5VZUBQrDa9hUngm0KnexY6pJRN5Q=="], + + "has-bigints": ["has-bigints@1.1.0", "", {}, "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg=="], + + "has-flag": ["has-flag@4.0.0", "", {}, "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ=="], + + "has-property-descriptors": ["has-property-descriptors@1.0.2", "", { "dependencies": { "es-define-property": "^1.0.0" } }, "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg=="], + + "has-proto": ["has-proto@1.2.0", "", { "dependencies": { "dunder-proto": "^1.0.0" } }, "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ=="], + + "has-symbols": ["has-symbols@1.1.0", "", {}, "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ=="], + + "has-tostringtag": ["has-tostringtag@1.0.2", "", { "dependencies": { "has-symbols": "^1.0.3" } }, "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw=="], + + "hasown": ["hasown@2.0.3", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg=="], + + "hast-util-to-jsx-runtime": ["hast-util-to-jsx-runtime@2.3.6", "", { "dependencies": { "@types/estree": "^1.0.0", "@types/hast": "^3.0.0", "@types/unist": "^3.0.0", "comma-separated-tokens": "^2.0.0", "devlop": "^1.0.0", "estree-util-is-identifier-name": "^3.0.0", "hast-util-whitespace": "^3.0.0", "mdast-util-mdx-expression": "^2.0.0", "mdast-util-mdx-jsx": "^3.0.0", "mdast-util-mdxjs-esm": "^2.0.0", "property-information": "^7.0.0", "space-separated-tokens": "^2.0.0", "style-to-js": "^1.0.0", "unist-util-position": "^5.0.0", "vfile-message": "^4.0.0" } }, "sha512-zl6s8LwNyo1P9uw+XJGvZtdFF1GdAkOg8ujOw+4Pyb76874fLps4ueHXDhXWdk6YHQ6OgUtinliG7RsYvCbbBg=="], + + "hast-util-whitespace": ["hast-util-whitespace@3.0.0", "", { "dependencies": { "@types/hast": "^3.0.0" } }, "sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw=="], + + "headers-polyfill": ["headers-polyfill@5.0.1", "", { "dependencies": { "@types/set-cookie-parser": "^2.4.10", "set-cookie-parser": "^3.0.1" } }, "sha512-1TJ6Fih/b8h5TIcv+1+Hw0PDQWJTKDKzFZzcKOiW1wJza3XoAQlkCuXLbymPYB8+ZQyw8mHvdw560e8zVFIWyA=="], + + "hermes-estree": ["hermes-estree@0.25.1", "", {}, "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw=="], + + "hermes-parser": ["hermes-parser@0.25.1", "", { "dependencies": { "hermes-estree": "0.25.1" } }, "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA=="], + + "hono": ["hono@4.12.23", "", {}, "sha512-eIaZ9qDgu7XV0pxOCrg7/WhnQ6Ivm22UcxhXx/A3dcbqbbYgBEkc6e/J/s7j2tS96zoB0S9VBdLwQNCWwUo4LA=="], + + "html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="], + + "http-errors": ["http-errors@2.0.1", "", { "dependencies": { "depd": "~2.0.0", "inherits": "~2.0.4", "setprototypeof": "~1.2.0", "statuses": "~2.0.2", "toidentifier": "~1.0.1" } }, "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ=="], + + "http-proxy-agent": ["http-proxy-agent@7.0.2", "", { "dependencies": { "agent-base": "^7.1.0", "debug": "^4.3.4" } }, "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig=="], + + "https-proxy-agent": ["https-proxy-agent@7.0.6", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "4" } }, "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw=="], + + "human-signals": ["human-signals@8.0.1", "", {}, "sha512-eKCa6bwnJhvxj14kZk5NCPc6Hb6BdsU9DZcOnmQKSnO1VKrfV0zCvtttPZUsBvjmNDn8rpcJfpwSYnHBjc95MQ=="], + + "iconv-lite": ["iconv-lite@0.7.2", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw=="], + + "icu-minify": ["icu-minify@4.12.0", "", { "dependencies": { "@formatjs/icu-messageformat-parser": "^3.4.0" } }, "sha512-zDmM05uav3t3+kxSfRrNlmyXOdj2b+uHA+p04CG32eJabtaHbugXujuL+YfRkwP9joAnf0Uh+RMGCKD5NLa5rQ=="], + + "ieee754": ["ieee754@1.2.1", "", {}, "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA=="], + + "ignore": ["ignore@5.3.2", "", {}, "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g=="], + + "immer": ["immer@10.2.0", "", {}, "sha512-d/+XTN3zfODyjr89gM3mPq1WNX2B8pYsu7eORitdwyA2sBubnTl3laYlBk4sXY5FUa5qTZGBDPJICVbvqzjlbw=="], + + "import-fresh": ["import-fresh@3.3.1", "", { "dependencies": { "parent-module": "^1.0.0", "resolve-from": "^4.0.0" } }, "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ=="], + + "imurmurhash": ["imurmurhash@0.1.4", "", {}, "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA=="], + + "inherits": ["inherits@2.0.4", "", {}, "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="], + + "ini": ["ini@1.3.8", "", {}, "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew=="], + + "inline-style-parser": ["inline-style-parser@0.2.7", "", {}, "sha512-Nb2ctOyNR8DqQoR0OwRG95uNWIC0C1lCgf5Naz5H6Ji72KZ8OcFZLz2P5sNgwlyoJ8Yif11oMuYs5pBQa86csA=="], + + "internal-slot": ["internal-slot@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "hasown": "^2.0.2", "side-channel": "^1.1.0" } }, "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw=="], + + "internmap": ["internmap@2.0.3", "", {}, "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg=="], + + "intl-messageformat": ["intl-messageformat@11.2.7", "", { "dependencies": { "@formatjs/fast-memoize": "3.1.5", "@formatjs/icu-messageformat-parser": "3.5.10" } }, "sha512-+q6Ktg119nULZEpZ8YTuGOst9MyEzFtjD63FTGBlN1mLz0Z/MOUYDIvnpVKwq17eezIEh+cfJIebfJoCetpiNw=="], + + "ip-address": ["ip-address@10.2.0", "", {}, "sha512-/+S6j4E9AHvW9SWMSEY9Xfy66O5PWvVEJ08O0y5JGyEKQpojb0K0GKpz/v5HJ/G0vi3D2sjGK78119oXZeE0qA=="], + + "ipaddr.js": ["ipaddr.js@1.9.1", "", {}, "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="], + + "is-alphabetical": ["is-alphabetical@2.0.1", "", {}, "sha512-FWyyY60MeTNyeSRpkM2Iry0G9hpr7/9kD40mD/cGQEuilcZYS4okz8SN2Q6rLCJ8gbCt6fN+rC+6tMGS99LaxQ=="], + + "is-alphanumerical": ["is-alphanumerical@2.0.1", "", { "dependencies": { "is-alphabetical": "^2.0.0", "is-decimal": "^2.0.0" } }, "sha512-hmbYhX/9MUMF5uh7tOXyK/n0ZvWpad5caBA17GsC6vyuCqaWliRG5K1qS9inmUhEMaOBIW7/whAnSwveW/LtZw=="], + + "is-array-buffer": ["is-array-buffer@3.0.5", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "get-intrinsic": "^1.2.6" } }, "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A=="], + + "is-arrayish": ["is-arrayish@0.2.1", "", {}, "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg=="], + + "is-async-function": ["is-async-function@2.1.1", "", { "dependencies": { "async-function": "^1.0.0", "call-bound": "^1.0.3", "get-proto": "^1.0.1", "has-tostringtag": "^1.0.2", "safe-regex-test": "^1.1.0" } }, "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ=="], + + "is-bigint": ["is-bigint@1.1.0", "", { "dependencies": { "has-bigints": "^1.0.2" } }, "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ=="], + + "is-boolean-object": ["is-boolean-object@1.2.2", "", { "dependencies": { "call-bound": "^1.0.3", "has-tostringtag": "^1.0.2" } }, "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A=="], + + "is-bun-module": ["is-bun-module@2.0.0", "", { "dependencies": { "semver": "^7.7.1" } }, "sha512-gNCGbnnnnFAUGKeZ9PdbyeGYJqewpmc2aKHUEMO5nQPWU9lOmv7jcmQIv+qHD8fXW6W7qfuCwX4rY9LNRjXrkQ=="], + + "is-callable": ["is-callable@1.2.7", "", {}, "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA=="], + + "is-core-module": ["is-core-module@2.16.2", "", { "dependencies": { "hasown": "^2.0.3" } }, "sha512-evOr8xfXKxE6qSR0hSXL2r3sd7ALj8+7jQEUvPYcm5sgZFdJ+AYzT6yNmJenvIYQBgIGwfwz08sL8zoL7yq2BA=="], + + "is-data-view": ["is-data-view@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "get-intrinsic": "^1.2.6", "is-typed-array": "^1.1.13" } }, "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw=="], + + "is-date-object": ["is-date-object@1.1.0", "", { "dependencies": { "call-bound": "^1.0.2", "has-tostringtag": "^1.0.2" } }, "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg=="], + + "is-decimal": ["is-decimal@2.0.1", "", {}, "sha512-AAB9hiomQs5DXWcRB1rqsxGUstbRroFOPPVAomNk/3XHR5JyEZChOyTWe2oayKnsSsr/kcGqF+z6yuH6HHpN0A=="], + + "is-docker": ["is-docker@3.0.0", "", { "bin": { "is-docker": "cli.js" } }, "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ=="], + + "is-extglob": ["is-extglob@2.1.1", "", {}, "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ=="], + + "is-finalizationregistry": ["is-finalizationregistry@1.1.1", "", { "dependencies": { "call-bound": "^1.0.3" } }, "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg=="], + + "is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="], + + "is-generator-function": ["is-generator-function@1.1.2", "", { "dependencies": { "call-bound": "^1.0.4", "generator-function": "^2.0.0", "get-proto": "^1.0.1", "has-tostringtag": "^1.0.2", "safe-regex-test": "^1.1.0" } }, "sha512-upqt1SkGkODW9tsGNG5mtXTXtECizwtS2kA161M+gJPc1xdb/Ax629af6YrTwcOeQHbewrPNlE5Dx7kzvXTizA=="], + + "is-glob": ["is-glob@4.0.3", "", { "dependencies": { "is-extglob": "^2.1.1" } }, "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg=="], + + "is-hexadecimal": ["is-hexadecimal@2.0.1", "", {}, "sha512-DgZQp241c8oO6cA1SbTEWiXeoxV42vlcJxgH+B3hi1AiqqKruZR3ZGF8In3fj4+/y/7rHvlOZLZtgJ/4ttYGZg=="], + + "is-in-ssh": ["is-in-ssh@1.0.0", "", {}, "sha512-jYa6Q9rH90kR1vKB6NM7qqd1mge3Fx4Dhw5TVlK1MUBqhEOuCagrEHMevNuCcbECmXZ0ThXkRm+Ymr51HwEPAw=="], + + "is-inside-container": ["is-inside-container@1.0.0", "", { "dependencies": { "is-docker": "^3.0.0" }, "bin": { "is-inside-container": "cli.js" } }, "sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA=="], + + "is-interactive": ["is-interactive@2.0.0", "", {}, "sha512-qP1vozQRI+BMOPcjFzrjXuQvdak2pHNUMZoeG2eRbiSqyvbEf/wQtEOTOX1guk6E3t36RkaqiSt8A/6YElNxLQ=="], + + "is-map": ["is-map@2.0.3", "", {}, "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw=="], + + "is-negative-zero": ["is-negative-zero@2.0.3", "", {}, "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw=="], + + "is-node-process": ["is-node-process@1.2.0", "", {}, "sha512-Vg4o6/fqPxIjtxgUH5QLJhwZ7gW5diGCVlXpuUfELC62CuxM1iHcRe51f2W1FDy04Ai4KJkagKjx3XaqyfRKXw=="], + + "is-number": ["is-number@7.0.0", "", {}, "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng=="], + + "is-number-object": ["is-number-object@1.1.1", "", { "dependencies": { "call-bound": "^1.0.3", "has-tostringtag": "^1.0.2" } }, "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw=="], + + "is-obj": ["is-obj@3.0.0", "", {}, "sha512-IlsXEHOjtKhpN8r/tRFj2nDyTmHvcfNeu/nrRIcXE17ROeatXchkojffa1SpdqW4cr/Fj6QkEf/Gn4zf6KKvEQ=="], + + "is-plain-obj": ["is-plain-obj@4.1.0", "", {}, "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg=="], + + "is-promise": ["is-promise@4.0.0", "", {}, "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ=="], + + "is-regex": ["is-regex@1.2.1", "", { "dependencies": { "call-bound": "^1.0.2", "gopd": "^1.2.0", "has-tostringtag": "^1.0.2", "hasown": "^2.0.2" } }, "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g=="], + + "is-regexp": ["is-regexp@3.1.0", "", {}, "sha512-rbku49cWloU5bSMI+zaRaXdQHXnthP6DZ/vLnfdSKyL4zUzuWnomtOEiZZOd+ioQ+avFo/qau3KPTc7Fjy1uPA=="], + + "is-set": ["is-set@2.0.3", "", {}, "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg=="], + + "is-shared-array-buffer": ["is-shared-array-buffer@1.0.4", "", { "dependencies": { "call-bound": "^1.0.3" } }, "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A=="], + + "is-stream": ["is-stream@4.0.1", "", {}, "sha512-Dnz92NInDqYckGEUJv689RbRiTSEHCQ7wOVeALbkOz999YpqT46yMRIGtSNl2iCL1waAZSx40+h59NV/EwzV/A=="], + + "is-string": ["is-string@1.1.1", "", { "dependencies": { "call-bound": "^1.0.3", "has-tostringtag": "^1.0.2" } }, "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA=="], + + "is-symbol": ["is-symbol@1.1.1", "", { "dependencies": { "call-bound": "^1.0.2", "has-symbols": "^1.1.0", "safe-regex-test": "^1.1.0" } }, "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w=="], + + "is-typed-array": ["is-typed-array@1.1.15", "", { "dependencies": { "which-typed-array": "^1.1.16" } }, "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ=="], + + "is-unicode-supported": ["is-unicode-supported@2.1.0", "", {}, "sha512-mE00Gnza5EEB3Ds0HfMyllZzbBrmLOX3vfWoj9A9PEnTfratQ/BcaJOuMhnkhjXvb2+FkY3VuHqtAGpTPmglFQ=="], + + "is-weakmap": ["is-weakmap@2.0.2", "", {}, "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w=="], + + "is-weakref": ["is-weakref@1.1.1", "", { "dependencies": { "call-bound": "^1.0.3" } }, "sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew=="], + + "is-weakset": ["is-weakset@2.0.4", "", { "dependencies": { "call-bound": "^1.0.3", "get-intrinsic": "^1.2.6" } }, "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ=="], + + "is-wsl": ["is-wsl@3.1.1", "", { "dependencies": { "is-inside-container": "^1.0.0" } }, "sha512-e6rvdUCiQCAuumZslxRJWR/Doq4VpPR82kqclvcS0efgt430SlGIk05vdCN58+VrzgtIcfNODjozVielycD4Sw=="], + + "isarray": ["isarray@2.0.5", "", {}, "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw=="], + + "isexe": ["isexe@2.0.0", "", {}, "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw=="], + + "israeli-bank-scrapers": ["israeli-bank-scrapers@6.7.4", "", { "dependencies": { "debug": "^4.3.2", "moment": "^2.22.2", "moment-timezone": "^0.5.37", "puppeteer": "^24.40.0" } }, "sha512-M6aEws/6Pxks8/INtDYCbLPDWeoZLStNF9ulTalUg89pAyFemNznl0SkcniIDqIQIZN/5vhtr79HojgvRpxN5w=="], + + "iterator.prototype": ["iterator.prototype@1.1.5", "", { "dependencies": { "define-data-property": "^1.1.4", "es-object-atoms": "^1.0.0", "get-intrinsic": "^1.2.6", "get-proto": "^1.0.0", "has-symbols": "^1.1.0", "set-function-name": "^2.0.2" } }, "sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g=="], + + "jiti": ["jiti@2.7.0", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ=="], + + "jose": ["jose@6.2.3", "", {}, "sha512-YYVDInQKFJfR/xa3ojUTl8c2KoTwiL1R5Wg9YCydwH0x0B9grbzlg5HC7mMjCtUJjbQ/YnGEZIhI5tCgfTb4Hw=="], + + "js-tokens": ["js-tokens@4.0.0", "", {}, "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="], + + "js-yaml": ["js-yaml@4.1.1", "", { "dependencies": { "argparse": "^2.0.1" }, "bin": { "js-yaml": "bin/js-yaml.js" } }, "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA=="], + + "jsesc": ["jsesc@3.1.0", "", { "bin": { "jsesc": "bin/jsesc" } }, "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA=="], + + "json-bigint": ["json-bigint@1.0.0", "", { "dependencies": { "bignumber.js": "^9.0.0" } }, "sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ=="], + + "json-buffer": ["json-buffer@3.0.1", "", {}, "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ=="], + + "json-parse-even-better-errors": ["json-parse-even-better-errors@2.3.1", "", {}, "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w=="], + + "json-schema": ["json-schema@0.4.0", "", {}, "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA=="], + + "json-schema-to-ts": ["json-schema-to-ts@3.1.1", "", { "dependencies": { "@babel/runtime": "^7.18.3", "ts-algebra": "^2.0.0" } }, "sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g=="], + + "json-schema-traverse": ["json-schema-traverse@0.4.1", "", {}, "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg=="], + + "json-schema-typed": ["json-schema-typed@8.0.2", "", {}, "sha512-fQhoXdcvc3V28x7C7BMs4P5+kNlgUURe2jmUT1T//oBRMDrqy1QPelJimwZGo7Hg9VPV3EQV5Bnq4hbFy2vetA=="], + + "json-stable-stringify-without-jsonify": ["json-stable-stringify-without-jsonify@1.0.1", "", {}, "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw=="], + + "json5": ["json5@2.2.3", "", { "bin": { "json5": "lib/cli.js" } }, "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg=="], + + "jsonfile": ["jsonfile@6.2.1", "", { "dependencies": { "universalify": "^2.0.0" }, "optionalDependencies": { "graceful-fs": "^4.1.6" } }, "sha512-zwOTdL3rFQ/lRdBnntKVOX6k5cKJwEc1HdilT71BWEu7J41gXIB2MRp+vxduPSwZJPWBxEzv4yH1wYLJGUHX4Q=="], + + "jsonrepair": ["jsonrepair@3.14.0", "", { "bin": { "jsonrepair": "bin/cli.js" } }, "sha512-tWPGKMZf/8UPim+fcW2EfcQ/d/7aKUrP6IECz9G3Tu6Q5dX0orSleqJ9z6sSw7qrQkjF8/Edo4DvsWBZ8H+HNg=="], + + "jsx-ast-utils": ["jsx-ast-utils@3.3.5", "", { "dependencies": { "array-includes": "^3.1.6", "array.prototype.flat": "^1.3.1", "object.assign": "^4.1.4", "object.values": "^1.1.6" } }, "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ=="], + + "jwa": ["jwa@2.0.1", "", { "dependencies": { "buffer-equal-constant-time": "^1.0.1", "ecdsa-sig-formatter": "1.0.11", "safe-buffer": "^5.0.1" } }, "sha512-hRF04fqJIP8Abbkq5NKGN0Bbr3JxlQ+qhZufXVr0DvujKy93ZCbXZMHDL4EOtodSbCWxOqR8MS1tXA5hwqCXDg=="], + + "jws": ["jws@4.0.1", "", { "dependencies": { "jwa": "^2.0.1", "safe-buffer": "^5.0.1" } }, "sha512-EKI/M/yqPncGUUh44xz0PxSidXFr/+r0pA70+gIYhjv+et7yxM+s29Y+VGDkovRofQem0fs7Uvf4+YmAdyRduA=="], + + "keyv": ["keyv@4.5.4", "", { "dependencies": { "json-buffer": "3.0.1" } }, "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw=="], + + "kleur": ["kleur@4.1.5", "", {}, "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ=="], + + "knip": ["knip@5.88.1", "", { "dependencies": { "@nodelib/fs.walk": "^1.2.3", "fast-glob": "^3.3.3", "formatly": "^0.3.0", "jiti": "^2.6.0", "minimist": "^1.2.8", "oxc-resolver": "^11.19.1", "picocolors": "^1.1.1", "picomatch": "^4.0.1", "smol-toml": "^1.5.2", "strip-json-comments": "5.0.3", "unbash": "^2.2.0", "yaml": "^2.8.2", "zod": "^4.1.11" }, "peerDependencies": { "@types/node": ">=18", "typescript": ">=5.0.4 <7" }, "bin": { "knip": "bin/knip.js", "knip-bun": "bin/knip-bun.js" } }, "sha512-tpy5o7zu1MjawVkLPuahymVJekYY3kYjvzcoInhIchgePxTlo+api90tBv2KfhAIe5uXh+mez1tAfmbv8/TiZg=="], + + "language-subtag-registry": ["language-subtag-registry@0.3.23", "", {}, "sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ=="], + + "language-tags": ["language-tags@1.0.9", "", { "dependencies": { "language-subtag-registry": "^0.3.20" } }, "sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA=="], + + "levn": ["levn@0.4.1", "", { "dependencies": { "prelude-ls": "^1.2.1", "type-check": "~0.4.0" } }, "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ=="], + + "lightningcss": ["lightningcss@1.32.0", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-android-arm64": "1.32.0", "lightningcss-darwin-arm64": "1.32.0", "lightningcss-darwin-x64": "1.32.0", "lightningcss-freebsd-x64": "1.32.0", "lightningcss-linux-arm-gnueabihf": "1.32.0", "lightningcss-linux-arm64-gnu": "1.32.0", "lightningcss-linux-arm64-musl": "1.32.0", "lightningcss-linux-x64-gnu": "1.32.0", "lightningcss-linux-x64-musl": "1.32.0", "lightningcss-win32-arm64-msvc": "1.32.0", "lightningcss-win32-x64-msvc": "1.32.0" } }, "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ=="], + + "lightningcss-android-arm64": ["lightningcss-android-arm64@1.32.0", "", { "os": "android", "cpu": "arm64" }, "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg=="], + + "lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.32.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ=="], + + "lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.32.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w=="], + + "lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.32.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig=="], + + "lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.32.0", "", { "os": "linux", "cpu": "arm" }, "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw=="], + + "lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.32.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ=="], + + "lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.32.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg=="], + + "lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.32.0", "", { "os": "linux", "cpu": "x64" }, "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA=="], + + "lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.32.0", "", { "os": "linux", "cpu": "x64" }, "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg=="], + + "lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.32.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw=="], + + "lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.32.0", "", { "os": "win32", "cpu": "x64" }, "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q=="], + + "lines-and-columns": ["lines-and-columns@1.2.4", "", {}, "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg=="], + + "locate-path": ["locate-path@6.0.0", "", { "dependencies": { "p-locate": "^5.0.0" } }, "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw=="], + + "lodash.merge": ["lodash.merge@4.6.2", "", {}, "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ=="], + + "log-symbols": ["log-symbols@6.0.0", "", { "dependencies": { "chalk": "^5.3.0", "is-unicode-supported": "^1.3.0" } }, "sha512-i24m8rpwhmPIS4zscNzK6MSEhk0DUWa/8iYQWxhffV8jkI4Phvs3F+quL5xvS0gdQR0FyTCMMH33Y78dDTzzIw=="], + + "long": ["long@5.3.2", "", {}, "sha512-mNAgZ1GmyNhD7AuqnTG3/VQ26o760+ZYBPKjPvugO8+nLbYfX6TVpJPseBvopbdY+qpZ/lKUnmEc1LeZYS3QAA=="], + + "longest-streak": ["longest-streak@3.1.0", "", {}, "sha512-9Ri+o0JYgehTaVBBDoMqIl8GXtbWg711O3srftcHhZ0dqnETqLaoIK0x17fUw9rFSlK/0NlsKe0Ahhyl5pXE2g=="], + + "loose-envify": ["loose-envify@1.4.0", "", { "dependencies": { "js-tokens": "^3.0.0 || ^4.0.0" }, "bin": { "loose-envify": "cli.js" } }, "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q=="], + + "lru-cache": ["lru-cache@7.18.3", "", {}, "sha512-jumlc0BIUrS3qJGgIkWZsyfAM7NCWiBcCDhnd+3NNM5KbBmLTgHVfWBcg6W+rLUsIpzpERPsvwUP7CckAQSOoA=="], + + "lucide-react": ["lucide-react@1.16.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-dYwyPzb4MEKpGUmNYk3WKWPnMrHs3FKM+q94kAnJrcDIqqn1hq2xY8scaS2ovsOCM5D51ey2gaRG3PBb1vgoYQ=="], + + "magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="], + + "markdown-table": ["markdown-table@3.0.4", "", {}, "sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw=="], + + "math-intrinsics": ["math-intrinsics@1.1.0", "", {}, "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g=="], + + "mdast-util-find-and-replace": ["mdast-util-find-and-replace@3.0.2", "", { "dependencies": { "@types/mdast": "^4.0.0", "escape-string-regexp": "^5.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-Tmd1Vg/m3Xz43afeNxDIhWRtFZgM2VLyaf4vSTYwudTyeuTneoL3qtWMA5jeLyz/O1vDJmmV4QuScFCA2tBPwg=="], + + "mdast-util-from-markdown": ["mdast-util-from-markdown@2.0.3", "", { "dependencies": { "@types/mdast": "^4.0.0", "@types/unist": "^3.0.0", "decode-named-character-reference": "^1.0.0", "devlop": "^1.0.0", "mdast-util-to-string": "^4.0.0", "micromark": "^4.0.0", "micromark-util-decode-numeric-character-reference": "^2.0.0", "micromark-util-decode-string": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0", "unist-util-stringify-position": "^4.0.0" } }, "sha512-W4mAWTvSlKvf8L6J+VN9yLSqQ9AOAAvHuoDAmPkz4dHf553m5gVj2ejadHJhoJmcmxEnOv6Pa8XJhpxE93kb8Q=="], + + "mdast-util-gfm": ["mdast-util-gfm@3.1.0", "", { "dependencies": { "mdast-util-from-markdown": "^2.0.0", "mdast-util-gfm-autolink-literal": "^2.0.0", "mdast-util-gfm-footnote": "^2.0.0", "mdast-util-gfm-strikethrough": "^2.0.0", "mdast-util-gfm-table": "^2.0.0", "mdast-util-gfm-task-list-item": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-0ulfdQOM3ysHhCJ1p06l0b0VKlhU0wuQs3thxZQagjcjPrlFRqY215uZGHHJan9GEAXd9MbfPjFJz+qMkVR6zQ=="], + + "mdast-util-gfm-autolink-literal": ["mdast-util-gfm-autolink-literal@2.0.1", "", { "dependencies": { "@types/mdast": "^4.0.0", "ccount": "^2.0.0", "devlop": "^1.0.0", "mdast-util-find-and-replace": "^3.0.0", "micromark-util-character": "^2.0.0" } }, "sha512-5HVP2MKaP6L+G6YaxPNjuL0BPrq9orG3TsrZ9YXbA3vDw/ACI4MEsnoDpn6ZNm7GnZgtAcONJyPhOP8tNJQavQ=="], + + "mdast-util-gfm-footnote": ["mdast-util-gfm-footnote@2.1.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "devlop": "^1.1.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0" } }, "sha512-sqpDWlsHn7Ac9GNZQMeUzPQSMzR6Wv0WKRNvQRg0KqHh02fpTz69Qc1QSseNX29bhz1ROIyNyxExfawVKTm1GQ=="], + + "mdast-util-gfm-strikethrough": ["mdast-util-gfm-strikethrough@2.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-mKKb915TF+OC5ptj5bJ7WFRPdYtuHv0yTRxK2tJvi+BDqbkiG7h7u/9SI89nRAYcmap2xHQL9D+QG/6wSrTtXg=="], + + "mdast-util-gfm-table": ["mdast-util-gfm-table@2.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "markdown-table": "^3.0.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-78UEvebzz/rJIxLvE7ZtDd/vIQ0RHv+3Mh5DR96p7cS7HsBhYIICDBCu8csTNWNO6tBWfqXPWekRuj2FNOGOZg=="], + + "mdast-util-gfm-task-list-item": ["mdast-util-gfm-task-list-item@2.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-IrtvNvjxC1o06taBAVJznEnkiHxLFTzgonUdy8hzFVeDun0uTjxxrRGVaNFqkU1wJR3RBPEfsxmU6jDWPofrTQ=="], + + "mdast-util-mdx-expression": ["mdast-util-mdx-expression@2.0.1", "", { "dependencies": { "@types/estree-jsx": "^1.0.0", "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-J6f+9hUp+ldTZqKRSg7Vw5V6MqjATc+3E4gf3CFNcuZNWD8XdyI6zQ8GqH7f8169MM6P7hMBRDVGnn7oHB9kXQ=="], + + "mdast-util-mdx-jsx": ["mdast-util-mdx-jsx@3.2.0", "", { "dependencies": { "@types/estree-jsx": "^1.0.0", "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "@types/unist": "^3.0.0", "ccount": "^2.0.0", "devlop": "^1.1.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0", "parse-entities": "^4.0.0", "stringify-entities": "^4.0.0", "unist-util-stringify-position": "^4.0.0", "vfile-message": "^4.0.0" } }, "sha512-lj/z8v0r6ZtsN/cGNNtemmmfoLAFZnjMbNyLzBafjzikOM+glrjNHPlf6lQDOTccj9n5b0PPihEBbhneMyGs1Q=="], + + "mdast-util-mdxjs-esm": ["mdast-util-mdxjs-esm@2.0.1", "", { "dependencies": { "@types/estree-jsx": "^1.0.0", "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "mdast-util-from-markdown": "^2.0.0", "mdast-util-to-markdown": "^2.0.0" } }, "sha512-EcmOpxsZ96CvlP03NghtH1EsLtr0n9Tm4lPUJUBccV9RwUOneqSycg19n5HGzCf+10LozMRSObtVr3ee1WoHtg=="], + + "mdast-util-phrasing": ["mdast-util-phrasing@4.1.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "unist-util-is": "^6.0.0" } }, "sha512-TqICwyvJJpBwvGAMZjj4J2n0X8QWp21b9l0o7eXyVJ25YNWYbJDVIyD1bZXE6WtV6RmKJVYmQAKWa0zWOABz2w=="], + + "mdast-util-to-hast": ["mdast-util-to-hast@13.2.1", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "@ungap/structured-clone": "^1.0.0", "devlop": "^1.0.0", "micromark-util-sanitize-uri": "^2.0.0", "trim-lines": "^3.0.0", "unist-util-position": "^5.0.0", "unist-util-visit": "^5.0.0", "vfile": "^6.0.0" } }, "sha512-cctsq2wp5vTsLIcaymblUriiTcZd0CwWtCbLvrOzYCDZoWyMNV8sZ7krj09FSnsiJi3WVsHLM4k6Dq/yaPyCXA=="], + + "mdast-util-to-markdown": ["mdast-util-to-markdown@2.1.2", "", { "dependencies": { "@types/mdast": "^4.0.0", "@types/unist": "^3.0.0", "longest-streak": "^3.0.0", "mdast-util-phrasing": "^4.0.0", "mdast-util-to-string": "^4.0.0", "micromark-util-classify-character": "^2.0.0", "micromark-util-decode-string": "^2.0.0", "unist-util-visit": "^5.0.0", "zwitch": "^2.0.0" } }, "sha512-xj68wMTvGXVOKonmog6LwyJKrYXZPvlwabaryTjLh9LuvovB/KAH+kvi8Gjj+7rJjsFi23nkUxRQv1KqSroMqA=="], + + "mdast-util-to-string": ["mdast-util-to-string@4.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0" } }, "sha512-0H44vDimn51F0YwvxSJSm0eCDOJTRlmN0R1yBh4HLj9wiV1Dn0QoXGbvFAWj2hSItVTlCmBF1hqKlIyUBVFLPg=="], + + "media-typer": ["media-typer@1.1.0", "", {}, "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw=="], + + "merge-descriptors": ["merge-descriptors@2.0.0", "", {}, "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g=="], + + "merge-stream": ["merge-stream@2.0.0", "", {}, "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w=="], + + "merge2": ["merge2@1.4.1", "", {}, "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg=="], + + "micromark": ["micromark@4.0.2", "", { "dependencies": { "@types/debug": "^4.0.0", "debug": "^4.0.0", "decode-named-character-reference": "^1.0.0", "devlop": "^1.0.0", "micromark-core-commonmark": "^2.0.0", "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-combine-extensions": "^2.0.0", "micromark-util-decode-numeric-character-reference": "^2.0.0", "micromark-util-encode": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-resolve-all": "^2.0.0", "micromark-util-sanitize-uri": "^2.0.0", "micromark-util-subtokenize": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-zpe98Q6kvavpCr1NPVSCMebCKfD7CA2NqZ+rykeNhONIJBpc1tFKt9hucLGwha3jNTNI8lHpctWJWoimVF4PfA=="], + + "micromark-core-commonmark": ["micromark-core-commonmark@2.0.3", "", { "dependencies": { "decode-named-character-reference": "^1.0.0", "devlop": "^1.0.0", "micromark-factory-destination": "^2.0.0", "micromark-factory-label": "^2.0.0", "micromark-factory-space": "^2.0.0", "micromark-factory-title": "^2.0.0", "micromark-factory-whitespace": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-classify-character": "^2.0.0", "micromark-util-html-tag-name": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-resolve-all": "^2.0.0", "micromark-util-subtokenize": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-RDBrHEMSxVFLg6xvnXmb1Ayr2WzLAWjeSATAoxwKYJV94TeNavgoIdA0a9ytzDSVzBy2YKFK+emCPOEibLeCrg=="], + + "micromark-extension-gfm": ["micromark-extension-gfm@3.0.0", "", { "dependencies": { "micromark-extension-gfm-autolink-literal": "^2.0.0", "micromark-extension-gfm-footnote": "^2.0.0", "micromark-extension-gfm-strikethrough": "^2.0.0", "micromark-extension-gfm-table": "^2.0.0", "micromark-extension-gfm-tagfilter": "^2.0.0", "micromark-extension-gfm-task-list-item": "^2.0.0", "micromark-util-combine-extensions": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-vsKArQsicm7t0z2GugkCKtZehqUm31oeGBV/KVSorWSy8ZlNAv7ytjFhvaryUiCUJYqs+NoE6AFhpQvBTM6Q4w=="], + + "micromark-extension-gfm-autolink-literal": ["micromark-extension-gfm-autolink-literal@2.1.0", "", { "dependencies": { "micromark-util-character": "^2.0.0", "micromark-util-sanitize-uri": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-oOg7knzhicgQ3t4QCjCWgTmfNhvQbDDnJeVu9v81r7NltNCVmhPy1fJRX27pISafdjL+SVc4d3l48Gb6pbRypw=="], + + "micromark-extension-gfm-footnote": ["micromark-extension-gfm-footnote@2.1.0", "", { "dependencies": { "devlop": "^1.0.0", "micromark-core-commonmark": "^2.0.0", "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-normalize-identifier": "^2.0.0", "micromark-util-sanitize-uri": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-/yPhxI1ntnDNsiHtzLKYnE3vf9JZ6cAisqVDauhp4CEHxlb4uoOTxOCJ+9s51bIB8U1N1FJ1RXOKTIlD5B/gqw=="], + + "micromark-extension-gfm-strikethrough": ["micromark-extension-gfm-strikethrough@2.1.0", "", { "dependencies": { "devlop": "^1.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-classify-character": "^2.0.0", "micromark-util-resolve-all": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-ADVjpOOkjz1hhkZLlBiYA9cR2Anf8F4HqZUO6e5eDcPQd0Txw5fxLzzxnEkSkfnD0wziSGiv7sYhk/ktvbf1uw=="], + + "micromark-extension-gfm-table": ["micromark-extension-gfm-table@2.1.1", "", { "dependencies": { "devlop": "^1.0.0", "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-t2OU/dXXioARrC6yWfJ4hqB7rct14e8f7m0cbI5hUmDyyIlwv5vEtooptH8INkbLzOatzKuVbQmAYcbWoyz6Dg=="], + + "micromark-extension-gfm-tagfilter": ["micromark-extension-gfm-tagfilter@2.0.0", "", { "dependencies": { "micromark-util-types": "^2.0.0" } }, "sha512-xHlTOmuCSotIA8TW1mDIM6X2O1SiX5P9IuDtqGonFhEK0qgRI4yeC6vMxEV2dgyr2TiD+2PQ10o+cOhdVAcwfg=="], + + "micromark-extension-gfm-task-list-item": ["micromark-extension-gfm-task-list-item@2.1.0", "", { "dependencies": { "devlop": "^1.0.0", "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-qIBZhqxqI6fjLDYFTBIa4eivDMnP+OZqsNwmQ3xNLE4Cxwc+zfQEfbs6tzAo2Hjq+bh6q5F+Z8/cksrLFYWQQw=="], + + "micromark-factory-destination": ["micromark-factory-destination@2.0.1", "", { "dependencies": { "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-Xe6rDdJlkmbFRExpTOmRj9N3MaWmbAgdpSrBQvCFqhezUn4AHqJHbaEnfbVYYiexVSs//tqOdY/DxhjdCiJnIA=="], + + "micromark-factory-label": ["micromark-factory-label@2.0.1", "", { "dependencies": { "devlop": "^1.0.0", "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-VFMekyQExqIW7xIChcXn4ok29YE3rnuyveW3wZQWWqF4Nv9Wk5rgJ99KzPvHjkmPXF93FXIbBp6YdW3t71/7Vg=="], + + "micromark-factory-space": ["micromark-factory-space@2.0.1", "", { "dependencies": { "micromark-util-character": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-zRkxjtBxxLd2Sc0d+fbnEunsTj46SWXgXciZmHq0kDYGnck/ZSGj9/wULTV95uoeYiK5hRXP2mJ98Uo4cq/LQg=="], + + "micromark-factory-title": ["micromark-factory-title@2.0.1", "", { "dependencies": { "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-5bZ+3CjhAd9eChYTHsjy6TGxpOFSKgKKJPJxr293jTbfry2KDoWkhBb6TcPVB4NmzaPhMs1Frm9AZH7OD4Cjzw=="], + + "micromark-factory-whitespace": ["micromark-factory-whitespace@2.0.1", "", { "dependencies": { "micromark-factory-space": "^2.0.0", "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-Ob0nuZ3PKt/n0hORHyvoD9uZhr+Za8sFoP+OnMcnWK5lngSzALgQYKMr9RJVOWLqQYuyn6ulqGWSXdwf6F80lQ=="], + + "micromark-util-character": ["micromark-util-character@2.1.1", "", { "dependencies": { "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-wv8tdUTJ3thSFFFJKtpYKOYiGP2+v96Hvk4Tu8KpCAsTMs6yi+nVmGh1syvSCsaxz45J6Jbw+9DD6g97+NV67Q=="], + + "micromark-util-chunked": ["micromark-util-chunked@2.0.1", "", { "dependencies": { "micromark-util-symbol": "^2.0.0" } }, "sha512-QUNFEOPELfmvv+4xiNg2sRYeS/P84pTW0TCgP5zc9FpXetHY0ab7SxKyAQCNCc1eK0459uoLI1y5oO5Vc1dbhA=="], + + "micromark-util-classify-character": ["micromark-util-classify-character@2.0.1", "", { "dependencies": { "micromark-util-character": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-K0kHzM6afW/MbeWYWLjoHQv1sgg2Q9EccHEDzSkxiP/EaagNzCm7T/WMKZ3rjMbvIpvBiZgwR3dKMygtA4mG1Q=="], + + "micromark-util-combine-extensions": ["micromark-util-combine-extensions@2.0.1", "", { "dependencies": { "micromark-util-chunked": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-OnAnH8Ujmy59JcyZw8JSbK9cGpdVY44NKgSM7E9Eh7DiLS2E9RNQf0dONaGDzEG9yjEl5hcqeIsj4hfRkLH/Bg=="], + + "micromark-util-decode-numeric-character-reference": ["micromark-util-decode-numeric-character-reference@2.0.2", "", { "dependencies": { "micromark-util-symbol": "^2.0.0" } }, "sha512-ccUbYk6CwVdkmCQMyr64dXz42EfHGkPQlBj5p7YVGzq8I7CtjXZJrubAYezf7Rp+bjPseiROqe7G6foFd+lEuw=="], + + "micromark-util-decode-string": ["micromark-util-decode-string@2.0.1", "", { "dependencies": { "decode-named-character-reference": "^1.0.0", "micromark-util-character": "^2.0.0", "micromark-util-decode-numeric-character-reference": "^2.0.0", "micromark-util-symbol": "^2.0.0" } }, "sha512-nDV/77Fj6eH1ynwscYTOsbK7rR//Uj0bZXBwJZRfaLEJ1iGBR6kIfNmlNqaqJf649EP0F3NWNdeJi03elllNUQ=="], + + "micromark-util-encode": ["micromark-util-encode@2.0.1", "", {}, "sha512-c3cVx2y4KqUnwopcO9b/SCdo2O67LwJJ/UyqGfbigahfegL9myoEFoDYZgkT7f36T0bLrM9hZTAaAyH+PCAXjw=="], + + "micromark-util-html-tag-name": ["micromark-util-html-tag-name@2.0.1", "", {}, "sha512-2cNEiYDhCWKI+Gs9T0Tiysk136SnR13hhO8yW6BGNyhOC4qYFnwF1nKfD3HFAIXA5c45RrIG1ub11GiXeYd1xA=="], + + "micromark-util-normalize-identifier": ["micromark-util-normalize-identifier@2.0.1", "", { "dependencies": { "micromark-util-symbol": "^2.0.0" } }, "sha512-sxPqmo70LyARJs0w2UclACPUUEqltCkJ6PhKdMIDuJ3gSf/Q+/GIe3WKl0Ijb/GyH9lOpUkRAO2wp0GVkLvS9Q=="], + + "micromark-util-resolve-all": ["micromark-util-resolve-all@2.0.1", "", { "dependencies": { "micromark-util-types": "^2.0.0" } }, "sha512-VdQyxFWFT2/FGJgwQnJYbe1jjQoNTS4RjglmSjTUlpUMa95Htx9NHeYW4rGDJzbjvCsl9eLjMQwGeElsqmzcHg=="], + + "micromark-util-sanitize-uri": ["micromark-util-sanitize-uri@2.0.1", "", { "dependencies": { "micromark-util-character": "^2.0.0", "micromark-util-encode": "^2.0.0", "micromark-util-symbol": "^2.0.0" } }, "sha512-9N9IomZ/YuGGZZmQec1MbgxtlgougxTodVwDzzEouPKo3qFWvymFHWcnDi2vzV1ff6kas9ucW+o3yzJK9YB1AQ=="], + + "micromark-util-subtokenize": ["micromark-util-subtokenize@2.1.0", "", { "dependencies": { "devlop": "^1.0.0", "micromark-util-chunked": "^2.0.0", "micromark-util-symbol": "^2.0.0", "micromark-util-types": "^2.0.0" } }, "sha512-XQLu552iSctvnEcgXw6+Sx75GflAPNED1qx7eBJ+wydBb2KCbRZe+NwvIEEMM83uml1+2WSXpBAcp9IUCgCYWA=="], + + "micromark-util-symbol": ["micromark-util-symbol@2.0.1", "", {}, "sha512-vs5t8Apaud9N28kgCrRUdEed4UJ+wWNvicHLPxCa9ENlYuAY31M0ETy5y1vA33YoNPDFTghEbnh6efaE8h4x0Q=="], + + "micromark-util-types": ["micromark-util-types@2.0.2", "", {}, "sha512-Yw0ECSpJoViF1qTU4DC6NwtC4aWGt1EkzaQB8KPPyCRR8z9TWeV0HbEFGTO+ZY1wB22zmxnJqhPyTpOVCpeHTA=="], + + "micromatch": ["micromatch@4.0.8", "", { "dependencies": { "braces": "^3.0.3", "picomatch": "^2.3.1" } }, "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA=="], + + "mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="], + + "mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="], + + "mimic-fn": ["mimic-fn@2.1.0", "", {}, "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg=="], + + "mimic-function": ["mimic-function@5.0.1", "", {}, "sha512-VP79XUPxV2CigYP3jWwAUFSku2aKqBH7uTAapFWCBqutsbmDo96KY5o8uh6U+/YSIn5OxJnXp73beVkpqMIGhA=="], + + "mimic-response": ["mimic-response@3.1.0", "", {}, "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ=="], + + "minimatch": ["minimatch@3.1.5", "", { "dependencies": { "brace-expansion": "^1.1.7" } }, "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w=="], + + "minimist": ["minimist@1.2.8", "", {}, "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA=="], + + "mitt": ["mitt@3.0.1", "", {}, "sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw=="], + + "mkdirp-classic": ["mkdirp-classic@0.5.3", "", {}, "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A=="], + + "moment": ["moment@2.30.1", "", {}, "sha512-uEmtNhbDOrWPFS+hdjFCBfy9f2YoyzRpwcl+DqpC6taX21FzsTLQVbMV/W7PzNSX6x/bhC1zA3c2UQ5NzH6how=="], + + "moment-timezone": ["moment-timezone@0.5.48", "", { "dependencies": { "moment": "^2.29.4" } }, "sha512-f22b8LV1gbTO2ms2j2z13MuPogNoh5UzxL3nzNAYKGraILnbGc9NEE6dyiiiLv46DGRb8A4kg8UKWLjPthxBHw=="], + + "motion-dom": ["motion-dom@12.40.0", "", { "dependencies": { "motion-utils": "^12.39.0" } }, "sha512-HxU3ZaBwNPVQUBQf1xxgq+7JrPNZvjLVxgbpEZL7RrWJnsxOf0/OM+yrHG9ogLQ31Do/r57Oz2gQWPK+6q62mg=="], + + "motion-utils": ["motion-utils@12.39.0", "", {}, "sha512-8nadJAJjTtqRkmRF36FoJTrywK9nnFmnPwnSMyxaOCU7GDjN9RTMJIxx9De8ErM+vpPhMccr/6fo5WciyQLnMQ=="], + + "ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="], + + "msw": ["msw@2.14.6", "", { "dependencies": { "@inquirer/confirm": "^6.0.11", "@mswjs/interceptors": "^0.41.3", "@open-draft/deferred-promise": "^3.0.0", "@types/statuses": "^2.0.6", "cookie": "^1.1.1", "graphql": "^16.13.2", "headers-polyfill": "^5.0.1", "is-node-process": "^1.2.0", "outvariant": "^1.4.3", "path-to-regexp": "^6.3.0", "picocolors": "^1.1.1", "rettime": "^0.11.11", "statuses": "^2.0.2", "strict-event-emitter": "^0.5.1", "tough-cookie": "^6.0.1", "type-fest": "^5.5.0", "until-async": "^3.0.2", "yargs": "^17.7.2" }, "peerDependencies": { "typescript": ">= 4.8.x" }, "optionalPeers": ["typescript"], "bin": { "msw": "cli/index.js" } }, "sha512-ALe+N10S72cyx94cMcy3Zs4HhXCj35sgeAL4c+WTvKi0zWnbd8/h0lcFqv0mb2P+aSgAdD7p9HzvA0DiUPxsyg=="], + + "mute-stream": ["mute-stream@4.0.0", "", {}, "sha512-gSrprq0fJ3EiOErzjdIZrjysVVmJ4uu1QWfCDss5LypA5OXvrMje5Ym5z6V6RLyJ2eF87lasX7t6a0AnFvZblg=="], + + "nanoid": ["nanoid@3.3.12", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ=="], + + "napi-build-utils": ["napi-build-utils@2.0.0", "", {}, "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA=="], + + "napi-postinstall": ["napi-postinstall@0.3.4", "", { "bin": { "napi-postinstall": "lib/cli.js" } }, "sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ=="], + + "natural-compare": ["natural-compare@1.4.0", "", {}, "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw=="], + + "negotiator": ["negotiator@1.0.0", "", {}, "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg=="], + + "netmask": ["netmask@2.1.1", "", {}, "sha512-eonl3sLUha+S1GzTPxychyhnUzKyeQkZ7jLjKrBagJgPla13F+uQ71HgpFefyHgqrjEbCPkDArxYsjY8/+gLKA=="], + + "next": ["next@16.2.6", "", { "dependencies": { "@next/env": "16.2.6", "@swc/helpers": "0.5.15", "baseline-browser-mapping": "^2.9.19", "caniuse-lite": "^1.0.30001579", "postcss": "8.4.31", "styled-jsx": "5.1.6" }, "optionalDependencies": { "@next/swc-darwin-arm64": "16.2.6", "@next/swc-darwin-x64": "16.2.6", "@next/swc-linux-arm64-gnu": "16.2.6", "@next/swc-linux-arm64-musl": "16.2.6", "@next/swc-linux-x64-gnu": "16.2.6", "@next/swc-linux-x64-musl": "16.2.6", "@next/swc-win32-arm64-msvc": "16.2.6", "@next/swc-win32-x64-msvc": "16.2.6", "sharp": "^0.34.5" }, "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" }, "optionalPeers": ["@opentelemetry/api", "@playwright/test", "babel-plugin-react-compiler", "sass"], "bin": { "next": "dist/bin/next" } }, "sha512-qOVgKJg1+At15NpeUP+eJgCHvTCgXsogweq87Ri/Ix7PkqQHg4sdaXmSFqKlgaIXE4kW0g25LE68W87UANlHtw=="], + + "next-intl": ["next-intl@4.12.0", "", { "dependencies": { "@formatjs/intl-localematcher": "^0.8.1", "@parcel/watcher": "^2.4.1", "@swc/core": "^1.15.2", "icu-minify": "^4.12.0", "negotiator": "^1.0.0", "next-intl-swc-plugin-extractor": "^4.12.0", "po-parser": "^2.1.1", "use-intl": "^4.12.0" }, "peerDependencies": { "next": "^12.0.0 || ^13.0.0 || ^14.0.0 || ^15.0.0 || ^16.0.0", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || >=19.0.0-rc <19.0.0 || ^19.0.0" } }, "sha512-v8KpppWG0yLLlChJ3Of6uoPew9LeRDBAtY6vpJmF7YJmBZlHEzzoEL4w1g1dAU+VleEPNoXNm9hg1eEsKWV5hw=="], + + "next-intl-swc-plugin-extractor": ["next-intl-swc-plugin-extractor@4.12.0", "", {}, "sha512-jUxVEu1Nryjt4YgaDktSys7ioOgQfcNPF/SF2dbPNxbVb6U+P1INRgHeCVN+EC59H2rnTFIQwbddmOCrUWFr3g=="], + + "next-themes": ["next-themes@0.4.6", "", { "peerDependencies": { "react": "^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc", "react-dom": "^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc" } }, "sha512-pZvgD5L0IEvX5/9GWyHMf3m8BKiVQwsCMHfoFosXtXBMnaS0ZnIJ9ST4b4NqLVKDEm8QBxoNNGNaBv2JNF6XNA=="], + + "node-abi": ["node-abi@3.92.0", "", { "dependencies": { "semver": "^7.3.5" } }, "sha512-KdHvFWZjEKDf0cakgFjebl371GPsISX2oZHcuyKqM7DtogIsHrqKeLTo8wBHxaXRAQlY2PsPlZmfo+9ZCxEREQ=="], + + "node-addon-api": ["node-addon-api@7.1.1", "", {}, "sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ=="], + + "node-domexception": ["node-domexception@1.0.0", "", {}, "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ=="], + + "node-exports-info": ["node-exports-info@1.6.0", "", { "dependencies": { "array.prototype.flatmap": "^1.3.3", "es-errors": "^1.3.0", "object.entries": "^1.1.9", "semver": "^6.3.1" } }, "sha512-pyFS63ptit/P5WqUkt+UUfe+4oevH+bFeIiPPdfb0pFeYEu/1ELnJu5l+5EcTKYL5M7zaAa7S8ddywgXypqKCw=="], + + "node-fetch": ["node-fetch@3.3.2", "", { "dependencies": { "data-uri-to-buffer": "^4.0.0", "fetch-blob": "^3.1.4", "formdata-polyfill": "^4.0.10" } }, "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA=="], + + "node-releases": ["node-releases@2.0.46", "", {}, "sha512-GYVXHE2KnrzAfsAjl4uP++evGFCrAU1jta4ubEjIG7YWt/64Gqv66a30yKwWczVjA6j3bM4nBwH7Pk1JmDHaxQ=="], + + "npm-run-path": ["npm-run-path@6.0.0", "", { "dependencies": { "path-key": "^4.0.0", "unicorn-magic": "^0.3.0" } }, "sha512-9qny7Z9DsQU8Ou39ERsPU4OZQlSTP47ShQzuKZ6PRXpYLtIFgl/DEBYEXKlvcEa+9tHVcK8CF81Y2V72qaZhWA=="], + + "object-assign": ["object-assign@4.1.1", "", {}, "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg=="], + + "object-inspect": ["object-inspect@1.13.4", "", {}, "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew=="], + + "object-keys": ["object-keys@1.1.1", "", {}, "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA=="], + + "object-treeify": ["object-treeify@1.1.33", "", {}, "sha512-EFVjAYfzWqWsBMRHPMAXLCDIJnpMhdWAqR7xG6M6a2cs6PMFpl/+Z20w9zDW4vkxOFfddegBKq9Rehd0bxWE7A=="], + + "object.assign": ["object.assign@4.1.7", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "define-properties": "^1.2.1", "es-object-atoms": "^1.0.0", "has-symbols": "^1.1.0", "object-keys": "^1.1.1" } }, "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw=="], + + "object.entries": ["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" } }, "sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw=="], + + "object.fromentries": ["object.fromentries@2.0.8", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-abstract": "^1.23.2", "es-object-atoms": "^1.0.0" } }, "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ=="], + + "object.groupby": ["object.groupby@1.0.3", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-abstract": "^1.23.2" } }, "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ=="], + + "object.values": ["object.values@1.2.1", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "define-properties": "^1.2.1", "es-object-atoms": "^1.0.0" } }, "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA=="], + + "ollama": ["ollama@0.6.3", "", { "dependencies": { "whatwg-fetch": "^3.6.20" } }, "sha512-KEWEhIqE5wtfzEIZbDCLH51VFZ6Z3ZSa6sIOg/E/tBV8S51flyqBOXi+bRxlOYKDf8i327zG9eSTb8IJxvm3Zg=="], + + "on-finished": ["on-finished@2.4.1", "", { "dependencies": { "ee-first": "1.1.1" } }, "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg=="], + + "once": ["once@1.4.0", "", { "dependencies": { "wrappy": "1" } }, "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w=="], + + "onetime": ["onetime@5.1.2", "", { "dependencies": { "mimic-fn": "^2.1.0" } }, "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg=="], + + "open": ["open@11.0.0", "", { "dependencies": { "default-browser": "^5.4.0", "define-lazy-prop": "^3.0.0", "is-in-ssh": "^1.0.0", "is-inside-container": "^1.0.0", "powershell-utils": "^0.1.0", "wsl-utils": "^0.3.0" } }, "sha512-smsWv2LzFjP03xmvFoJ331ss6h+jixfA4UUV/Bsiyuu4YJPfN+FIQGOIiv4w9/+MoHkfkJ22UIaQWRVFRfH6Vw=="], + + "optionator": ["optionator@0.9.4", "", { "dependencies": { "deep-is": "^0.1.3", "fast-levenshtein": "^2.0.6", "levn": "^0.4.1", "prelude-ls": "^1.2.1", "type-check": "^0.4.0", "word-wrap": "^1.2.5" } }, "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g=="], + + "ora": ["ora@8.2.0", "", { "dependencies": { "chalk": "^5.3.0", "cli-cursor": "^5.0.0", "cli-spinners": "^2.9.2", "is-interactive": "^2.0.0", "is-unicode-supported": "^2.0.0", "log-symbols": "^6.0.0", "stdin-discarder": "^0.2.2", "string-width": "^7.2.0", "strip-ansi": "^7.1.0" } }, "sha512-weP+BZ8MVNnlCm8c0Qdc1WSWq4Qn7I+9CJGm7Qali6g44e/PUzbjNqJX5NJ9ljlNMosfJvg1fKEGILklK9cwnw=="], + + "outvariant": ["outvariant@1.4.3", "", {}, "sha512-+Sl2UErvtsoajRDKCE5/dBz4DIvHXQQnAxtQTF04OJxY0+DyZXSo5P5Bb7XYWOh81syohlYL24hbDwxedPUJCA=="], + + "own-keys": ["own-keys@1.0.1", "", { "dependencies": { "get-intrinsic": "^1.2.6", "object-keys": "^1.1.1", "safe-push-apply": "^1.0.0" } }, "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg=="], + + "oxc-resolver": ["oxc-resolver@11.19.1", "", { "optionalDependencies": { "@oxc-resolver/binding-android-arm-eabi": "11.19.1", "@oxc-resolver/binding-android-arm64": "11.19.1", "@oxc-resolver/binding-darwin-arm64": "11.19.1", "@oxc-resolver/binding-darwin-x64": "11.19.1", "@oxc-resolver/binding-freebsd-x64": "11.19.1", "@oxc-resolver/binding-linux-arm-gnueabihf": "11.19.1", "@oxc-resolver/binding-linux-arm-musleabihf": "11.19.1", "@oxc-resolver/binding-linux-arm64-gnu": "11.19.1", "@oxc-resolver/binding-linux-arm64-musl": "11.19.1", "@oxc-resolver/binding-linux-ppc64-gnu": "11.19.1", "@oxc-resolver/binding-linux-riscv64-gnu": "11.19.1", "@oxc-resolver/binding-linux-riscv64-musl": "11.19.1", "@oxc-resolver/binding-linux-s390x-gnu": "11.19.1", "@oxc-resolver/binding-linux-x64-gnu": "11.19.1", "@oxc-resolver/binding-linux-x64-musl": "11.19.1", "@oxc-resolver/binding-openharmony-arm64": "11.19.1", "@oxc-resolver/binding-wasm32-wasi": "11.19.1", "@oxc-resolver/binding-win32-arm64-msvc": "11.19.1", "@oxc-resolver/binding-win32-ia32-msvc": "11.19.1", "@oxc-resolver/binding-win32-x64-msvc": "11.19.1" } }, "sha512-qE/CIg/spwrTBFt5aKmwe3ifeDdLfA2NESN30E42X/lII5ClF8V7Wt6WIJhcGZjp0/Q+nQ+9vgxGk//xZNX2hg=="], + + "p-limit": ["p-limit@3.1.0", "", { "dependencies": { "yocto-queue": "^0.1.0" } }, "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ=="], + + "p-locate": ["p-locate@5.0.0", "", { "dependencies": { "p-limit": "^3.0.2" } }, "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw=="], + + "p-retry": ["p-retry@4.6.2", "", { "dependencies": { "@types/retry": "0.12.0", "retry": "^0.13.1" } }, "sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ=="], + + "pac-proxy-agent": ["pac-proxy-agent@7.2.0", "", { "dependencies": { "@tootallnate/quickjs-emscripten": "^0.23.0", "agent-base": "^7.1.2", "debug": "^4.3.4", "get-uri": "^6.0.1", "http-proxy-agent": "^7.0.0", "https-proxy-agent": "^7.0.6", "pac-resolver": "^7.0.1", "socks-proxy-agent": "^8.0.5" } }, "sha512-TEB8ESquiLMc0lV8vcd5Ql/JAKAoyzHFXaStwjkzpOpC5Yv+pIzLfHvjTSdf3vpa2bMiUQrg9i6276yn8666aA=="], + + "pac-resolver": ["pac-resolver@7.0.1", "", { "dependencies": { "degenerator": "^5.0.0", "netmask": "^2.0.2" } }, "sha512-5NPgf87AT2STgwa2ntRMr45jTKrYBGkVU36yT0ig/n/GMAa3oPqhZfIQ2kMEimReg0+t9kZViDVZ83qfVUlckg=="], + + "parent-module": ["parent-module@1.0.1", "", { "dependencies": { "callsites": "^3.0.0" } }, "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g=="], + + "parse-entities": ["parse-entities@4.0.2", "", { "dependencies": { "@types/unist": "^2.0.0", "character-entities-legacy": "^3.0.0", "character-reference-invalid": "^2.0.0", "decode-named-character-reference": "^1.0.0", "is-alphanumerical": "^2.0.0", "is-decimal": "^2.0.0", "is-hexadecimal": "^2.0.0" } }, "sha512-GG2AQYWoLgL877gQIKeRPGO1xF9+eG1ujIb5soS5gPvLQ1y2o8FL90w2QWNdf9I361Mpp7726c+lj3U0qK1uGw=="], + + "parse-json": ["parse-json@5.2.0", "", { "dependencies": { "@babel/code-frame": "^7.0.0", "error-ex": "^1.3.1", "json-parse-even-better-errors": "^2.3.0", "lines-and-columns": "^1.1.6" } }, "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg=="], + + "parse-ms": ["parse-ms@4.0.0", "", {}, "sha512-TXfryirbmq34y8QBwgqCVLi+8oA3oWx2eAnSn62ITyEhEYaWRlVZ2DvMM9eZbMs/RfxPu/PK/aBLyGj4IrqMHw=="], + + "parseurl": ["parseurl@1.3.3", "", {}, "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="], + + "path-browserify": ["path-browserify@1.0.1", "", {}, "sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g=="], + + "path-exists": ["path-exists@4.0.0", "", {}, "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w=="], + + "path-key": ["path-key@3.1.1", "", {}, "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q=="], + + "path-parse": ["path-parse@1.0.7", "", {}, "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw=="], + + "path-to-regexp": ["path-to-regexp@6.3.0", "", {}, "sha512-Yhpw4T9C6hPpgPeA28us07OJeqZ5EzQTkbfwuhsUg0c237RomFoETJgmp2sa3F/41gfLE6G5cqcYwznmeEeOlQ=="], + + "pend": ["pend@1.2.0", "", {}, "sha512-F3asv42UuXchdzt+xXqfW1OGlVBe+mxa2mqI0pg5yAHZPvFmY3Y6drSf/GQ1A86WgWEN9Kzh/WrgKa6iGcHXLg=="], + + "picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="], + + "picomatch": ["picomatch@4.0.4", "", {}, "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A=="], + + "pkce-challenge": ["pkce-challenge@5.0.1", "", {}, "sha512-wQ0b/W4Fr01qtpHlqSqspcj3EhBvimsdh0KlHhH8HRZnMsEa0ea2fTULOXOS9ccQr3om+GcGRk4e+isrZWV8qQ=="], + + "po-parser": ["po-parser@2.1.1", "", {}, "sha512-ECF4zHLbUItpUgE3OTtLKlPjeBN+fKEczj2zYjDfCGOzicNs0GK3Vg2IoAYwx7LH/XYw43fZQP6xnZ4TkNxSLQ=="], + + "possible-typed-array-names": ["possible-typed-array-names@1.1.0", "", {}, "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg=="], + + "postcss": ["postcss@8.5.15", "", { "dependencies": { "nanoid": "^3.3.12", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-FfR8sjd4em2T6fb3I2MwAJU7HWVMr9zba+enmQeeWFfCbm+UOC/0X4DS8XtpUTMwWMGbjKYP7xjfNekzyGmB3A=="], + + "postcss-selector-parser": ["postcss-selector-parser@7.1.1", "", { "dependencies": { "cssesc": "^3.0.0", "util-deprecate": "^1.0.2" } }, "sha512-orRsuYpJVw8LdAwqqLykBj9ecS5/cRHlI5+nvTo8LcCKmzDmqVORXtOIYEEQuL9D4BxtA1lm5isAqzQZCoQ6Eg=="], + + "powershell-utils": ["powershell-utils@0.1.0", "", {}, "sha512-dM0jVuXJPsDN6DvRpea484tCUaMiXWjuCn++HGTqUWzGDjv5tZkEZldAJ/UMlqRYGFrD/etByo4/xOuC/snX2A=="], + + "prebuild-install": ["prebuild-install@7.1.3", "", { "dependencies": { "detect-libc": "^2.0.0", "expand-template": "^2.0.3", "github-from-package": "0.0.0", "minimist": "^1.2.3", "mkdirp-classic": "^0.5.3", "napi-build-utils": "^2.0.0", "node-abi": "^3.3.0", "pump": "^3.0.0", "rc": "^1.2.7", "simple-get": "^4.0.0", "tar-fs": "^2.0.0", "tunnel-agent": "^0.6.0" }, "bin": { "prebuild-install": "bin.js" } }, "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug=="], + + "prelude-ls": ["prelude-ls@1.2.1", "", {}, "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g=="], + + "pretty-ms": ["pretty-ms@9.3.0", "", { "dependencies": { "parse-ms": "^4.0.0" } }, "sha512-gjVS5hOP+M3wMm5nmNOucbIrqudzs9v/57bWRHQWLYklXqoXKrVfYW2W9+glfGsqtPgpiz5WwyEEB+ksXIx3gQ=="], + + "progress": ["progress@2.0.3", "", {}, "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA=="], + + "prompts": ["prompts@2.4.2", "", { "dependencies": { "kleur": "^3.0.3", "sisteransi": "^1.0.5" } }, "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q=="], + + "prop-types": ["prop-types@15.8.1", "", { "dependencies": { "loose-envify": "^1.4.0", "object-assign": "^4.1.1", "react-is": "^16.13.1" } }, "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg=="], + + "property-information": ["property-information@7.1.0", "", {}, "sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ=="], + + "protobufjs": ["protobufjs@7.6.1", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.2", "@protobufjs/base64": "^1.1.2", "@protobufjs/codegen": "^2.0.5", "@protobufjs/eventemitter": "^1.1.1", "@protobufjs/fetch": "^1.1.1", "@protobufjs/float": "^1.0.2", "@protobufjs/inquire": "^1.1.2", "@protobufjs/path": "^1.1.2", "@protobufjs/pool": "^1.1.0", "@protobufjs/utf8": "^1.1.1", "@types/node": ">=13.7.0", "long": "^5.3.2" } }, "sha512-4K0myLaWL5EteuSAro91EGFgcfVgxb64Jx+7oDAY6GOkXD4M69yuSEljNcInGVCA5sOPxmZ/EqDLj2x0Q0+Ygg=="], + + "proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="], + + "proxy-agent": ["proxy-agent@6.5.0", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "^4.3.4", "http-proxy-agent": "^7.0.1", "https-proxy-agent": "^7.0.6", "lru-cache": "^7.14.1", "pac-proxy-agent": "^7.1.0", "proxy-from-env": "^1.1.0", "socks-proxy-agent": "^8.0.5" } }, "sha512-TmatMXdr2KlRiA2CyDu8GqR8EjahTG3aY3nXjdzFyoZbmB8hrBsTyMezhULIXKnC0jpfjlmiZ3+EaCzoInSu/A=="], + + "proxy-from-env": ["proxy-from-env@1.1.0", "", {}, "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="], + + "pump": ["pump@3.0.4", "", { "dependencies": { "end-of-stream": "^1.1.0", "once": "^1.3.1" } }, "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA=="], + + "punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="], + + "puppeteer": ["puppeteer@24.43.1", "", { "dependencies": { "@puppeteer/browsers": "2.13.2", "chromium-bidi": "14.0.0", "cosmiconfig": "^9.0.0", "devtools-protocol": "0.0.1608973", "puppeteer-core": "24.43.1", "typed-query-selector": "^2.12.2" }, "bin": { "puppeteer": "lib/cjs/puppeteer/node/cli.js" } }, "sha512-/FSOViCrqRdb1HDocpsM9Z1giA71gTQPUt3SpHGVRALKAy/rJr1fLFYZW9F23qPxqVxTHQnbh/5B5opJST3kAw=="], + + "puppeteer-core": ["puppeteer-core@24.43.1", "", { "dependencies": { "@puppeteer/browsers": "2.13.2", "chromium-bidi": "14.0.0", "debug": "^4.4.3", "devtools-protocol": "0.0.1608973", "typed-query-selector": "^2.12.2", "webdriver-bidi-protocol": "0.4.1", "ws": "^8.20.0" } }, "sha512-T5ScUMAsmhdNbgDR41AGESYeS6V9MSgetkSnVhhW+gXvzC42VesKCn5ld87gAZDJ6vLHL9GkRvY9WtQWSnwFbw=="], + + "qs": ["qs@6.15.2", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-Rzq0KEyX/w/tEybncDgdkZrJgVUsUMk3xjh3t5bv3S1HTAtg+uOYt72+ZfwiQwKdysThkTBdL/rTi6HDmX9Ddw=="], + + "queue-microtask": ["queue-microtask@1.2.3", "", {}, "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A=="], + + "range-parser": ["range-parser@1.2.1", "", {}, "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="], + + "raw-body": ["raw-body@3.0.2", "", { "dependencies": { "bytes": "~3.1.2", "http-errors": "~2.0.1", "iconv-lite": "~0.7.0", "unpipe": "~1.0.0" } }, "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA=="], + + "rc": ["rc@1.2.8", "", { "dependencies": { "deep-extend": "^0.6.0", "ini": "~1.3.0", "minimist": "^1.2.0", "strip-json-comments": "~2.0.1" }, "bin": { "rc": "./cli.js" } }, "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw=="], + + "react": ["react@19.2.4", "", {}, "sha512-9nfp2hYpCwOjAN+8TZFGhtWEwgvWHXqESH8qT89AT/lWklpLON22Lc8pEtnpsZz7VmawabSU0gCjnj8aC0euHQ=="], + + "react-dom": ["react-dom@19.2.4", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.4" } }, "sha512-AXJdLo8kgMbimY95O2aKQqsz2iWi9jMgKJhRBAxECE4IFxfcazB2LmzloIoibJI3C12IlY20+KFaLv+71bUJeQ=="], + + "react-is": ["react-is@16.13.1", "", {}, "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ=="], + + "react-markdown": ["react-markdown@10.1.0", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "hast-util-to-jsx-runtime": "^2.0.0", "html-url-attributes": "^3.0.0", "mdast-util-to-hast": "^13.0.0", "remark-parse": "^11.0.0", "remark-rehype": "^11.0.0", "unified": "^11.0.0", "unist-util-visit": "^5.0.0", "vfile": "^6.0.0" }, "peerDependencies": { "@types/react": ">=18", "react": ">=18" } }, "sha512-qKxVopLT/TyA6BX3Ue5NwabOsAzm0Q7kAPwq6L+wWDwisYs7R8vZ0nRXqq6rkueboxpkjvLGU9fWifiX/ZZFxQ=="], + + "react-redux": ["react-redux@9.3.0", "", { "dependencies": { "@types/use-sync-external-store": "^0.0.6", "use-sync-external-store": "^1.4.0" }, "peerDependencies": { "@types/react": "^18.2.25 || ^19", "react": "^18.0 || ^19", "redux": "^5.0.0" }, "optionalPeers": ["@types/react", "redux"] }, "sha512-KQopgqFo/p/fgmAs5qz6p5RWaNAzq40WAu7fJIXnQpYxFPbJYtsJPWvGeF2rOBaY/kEuV77AVsX8TsQzKm+A/g=="], + + "readable-stream": ["readable-stream@3.6.2", "", { "dependencies": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } }, "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA=="], + + "recast": ["recast@0.23.11", "", { "dependencies": { "ast-types": "^0.16.1", "esprima": "~4.0.0", "source-map": "~0.6.1", "tiny-invariant": "^1.3.3", "tslib": "^2.0.1" } }, "sha512-YTUo+Flmw4ZXiWfQKGcwwc11KnoRAYgzAE2E7mXKCjSviTKShtxBsN6YUUBB2gtaBzKzeKunxhUwNHQuRryhWA=="], + + "recharts": ["recharts@3.8.1", "", { "dependencies": { "@reduxjs/toolkit": "^1.9.0 || 2.x.x", "clsx": "^2.1.1", "decimal.js-light": "^2.5.1", "es-toolkit": "^1.39.3", "eventemitter3": "^5.0.1", "immer": "^10.1.1", "react-redux": "8.x.x || 9.x.x", "reselect": "5.1.1", "tiny-invariant": "^1.3.3", "use-sync-external-store": "^1.2.2", "victory-vendor": "^37.0.2" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-is": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-mwzmO1s9sFL0TduUpwndxCUNoXsBw3u3E/0+A+cLcrSfQitSG62L32N69GhqUrrT5qKcAE3pCGVINC6pqkBBQg=="], + + "redux": ["redux@5.0.1", "", {}, "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w=="], + + "redux-thunk": ["redux-thunk@3.1.0", "", { "peerDependencies": { "redux": "^5.0.0" } }, "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw=="], + + "reflect.getprototypeof": ["reflect.getprototypeof@1.0.10", "", { "dependencies": { "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-abstract": "^1.23.9", "es-errors": "^1.3.0", "es-object-atoms": "^1.0.0", "get-intrinsic": "^1.2.7", "get-proto": "^1.0.1", "which-builtin-type": "^1.2.1" } }, "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw=="], + + "regexp.prototype.flags": ["regexp.prototype.flags@1.5.4", "", { "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" } }, "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA=="], + + "remark-gfm": ["remark-gfm@4.0.1", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-gfm": "^3.0.0", "micromark-extension-gfm": "^3.0.0", "remark-parse": "^11.0.0", "remark-stringify": "^11.0.0", "unified": "^11.0.0" } }, "sha512-1quofZ2RQ9EWdeN34S79+KExV1764+wCUGop5CPL1WGdD0ocPpu91lzPGbwWMECpEpd42kJGQwzRfyov9j4yNg=="], + + "remark-parse": ["remark-parse@11.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-from-markdown": "^2.0.0", "micromark-util-types": "^2.0.0", "unified": "^11.0.0" } }, "sha512-FCxlKLNGknS5ba/1lmpYijMUzX2esxW5xQqjWxw2eHFfS2MSdaHVINFmhjo+qN1WhZhNimq0dZATN9pH0IDrpA=="], + + "remark-rehype": ["remark-rehype@11.1.2", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "mdast-util-to-hast": "^13.0.0", "unified": "^11.0.0", "vfile": "^6.0.0" } }, "sha512-Dh7l57ianaEoIpzbp0PC9UKAdCSVklD8E5Rpw7ETfbTl3FqcOOgq5q2LVDhgGCkaBv7p24JXikPdvhhmHvKMsw=="], + + "remark-stringify": ["remark-stringify@11.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-to-markdown": "^2.0.0", "unified": "^11.0.0" } }, "sha512-1OSmLd3awB/t8qdoEOMazZkNsfVTeY4fTsgzcQFdXNq8ToTN4ZGwrMnlda4K6smTFKD+GRV6O48i6Z4iKgPPpw=="], + + "require-directory": ["require-directory@2.1.1", "", {}, "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q=="], + + "require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="], + + "reselect": ["reselect@5.1.1", "", {}, "sha512-K/BG6eIky/SBpzfHZv/dd+9JBFiS4SWV7FIujVyJRux6e45+73RaUHXLmIR1f7WOMaQ0U1km6qwklRQxpJJY0w=="], + + "resolve": ["resolve@2.0.0-next.7", "", { "dependencies": { "es-errors": "^1.3.0", "is-core-module": "^2.16.2", "node-exports-info": "^1.6.0", "object-keys": "^1.1.1", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" }, "bin": { "resolve": "bin/resolve" } }, "sha512-tqt+NBWwyaMgw3zDsnygx4CByWjQEJHOPMdslYhppaQSJUtL/D4JO9CcBBlhPoI8lz9oJIDXkwXfhF4aWqP8xQ=="], + + "resolve-from": ["resolve-from@4.0.0", "", {}, "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g=="], + + "resolve-pkg-maps": ["resolve-pkg-maps@1.0.0", "", {}, "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw=="], + + "restore-cursor": ["restore-cursor@5.1.0", "", { "dependencies": { "onetime": "^7.0.0", "signal-exit": "^4.1.0" } }, "sha512-oMA2dcrw6u0YfxJQXm342bFKX/E4sG9rbTzO9ptUcR/e8A33cHuvStiYOwH7fszkZlZ1z/ta9AAoPk2F4qIOHA=="], + + "retry": ["retry@0.13.1", "", {}, "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg=="], + + "rettime": ["rettime@0.11.11", "", {}, "sha512-ILJRqVWBCTlg9r42fFgwVZx1gnFAcQF8mRoMkbgQfIrjEDf9nbBFDFx00oloOa+Q869FUtaYDXZvEfnecQSCoQ=="], + + "reusify": ["reusify@1.1.0", "", {}, "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw=="], + + "router": ["router@2.2.0", "", { "dependencies": { "debug": "^4.4.0", "depd": "^2.0.0", "is-promise": "^4.0.0", "parseurl": "^1.3.3", "path-to-regexp": "^8.0.0" } }, "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ=="], + + "run-applescript": ["run-applescript@7.1.0", "", {}, "sha512-DPe5pVFaAsinSaV6QjQ6gdiedWDcRCbUuiQfQa2wmWV7+xC9bGulGI8+TdRmoFkAPaBXk8CrAbnlY2ISniJ47Q=="], + + "run-parallel": ["run-parallel@1.2.0", "", { "dependencies": { "queue-microtask": "^1.2.2" } }, "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA=="], + + "safe-array-concat": ["safe-array-concat@1.1.4", "", { "dependencies": { "call-bind": "^1.0.9", "call-bound": "^1.0.4", "get-intrinsic": "^1.3.0", "has-symbols": "^1.1.0", "isarray": "^2.0.5" } }, "sha512-wtZlHyOje6OZTGqAoaDKxFkgRtkF9CnHAVnCHKfuj200wAgL+bSJhdsCD2l0Qx/2ekEXjPWcyKkfGb5CPboslg=="], + + "safe-buffer": ["safe-buffer@5.2.1", "", {}, "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="], + + "safe-push-apply": ["safe-push-apply@1.0.0", "", { "dependencies": { "es-errors": "^1.3.0", "isarray": "^2.0.5" } }, "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA=="], + + "safe-regex-test": ["safe-regex-test@1.1.0", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "is-regex": "^1.2.1" } }, "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw=="], + + "safer-buffer": ["safer-buffer@2.1.2", "", {}, "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="], + + "scheduler": ["scheduler@0.27.0", "", {}, "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q=="], + + "semver": ["semver@6.3.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA=="], + + "send": ["send@1.2.1", "", { "dependencies": { "debug": "^4.4.3", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "fresh": "^2.0.0", "http-errors": "^2.0.1", "mime-types": "^3.0.2", "ms": "^2.1.3", "on-finished": "^2.4.1", "range-parser": "^1.2.1", "statuses": "^2.0.2" } }, "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ=="], + + "serve-static": ["serve-static@2.2.1", "", { "dependencies": { "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "parseurl": "^1.3.3", "send": "^1.2.0" } }, "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw=="], + + "server-only": ["server-only@0.0.1", "", {}, "sha512-qepMx2JxAa5jjfzxG79yPPq+8BuFToHd1hm7kI+Z4zAq1ftQiP7HcxMhDDItrbtwVeLg/cY2JnKnrcFkmiswNA=="], + + "set-cookie-parser": ["set-cookie-parser@3.1.0", "", {}, "sha512-kjnC1DXBHcxaOaOXBHBeRtltsDG2nUiUni+jP92M9gYdW12rsmx92UsfpH7o5tDRs7I1ZZPSQJQGv3UaRfCiuw=="], + + "set-function-length": ["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.2.4", "gopd": "^1.0.1", "has-property-descriptors": "^1.0.2" } }, "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg=="], + + "set-function-name": ["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" } }, "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ=="], + + "set-proto": ["set-proto@1.0.0", "", { "dependencies": { "dunder-proto": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.0.0" } }, "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw=="], + + "setprototypeof": ["setprototypeof@1.2.0", "", {}, "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="], + + "shadcn": ["shadcn@4.8.1", "", { "dependencies": { "@babel/core": "^7.28.0", "@babel/parser": "^7.28.0", "@babel/plugin-transform-typescript": "^7.28.0", "@babel/preset-typescript": "^7.27.1", "@dotenvx/dotenvx": "^1.48.4", "@modelcontextprotocol/sdk": "^1.26.0", "@types/validate-npm-package-name": "^4.0.2", "browserslist": "^4.26.2", "commander": "^14.0.0", "cosmiconfig": "^9.0.0", "dedent": "^1.6.0", "deepmerge": "^4.3.1", "diff": "^8.0.2", "execa": "^9.6.0", "fast-glob": "^3.3.3", "fs-extra": "^11.3.1", "fuzzysort": "^3.1.0", "https-proxy-agent": "^7.0.6", "kleur": "^4.1.5", "msw": "^2.10.4", "node-fetch": "^3.3.2", "open": "^11.0.0", "ora": "^8.2.0", "postcss": "^8.5.6", "postcss-selector-parser": "^7.1.0", "prompts": "^2.4.2", "recast": "^0.23.11", "stringify-object": "^5.0.0", "tailwind-merge": "^3.0.1", "ts-morph": "^26.0.0", "tsconfig-paths": "^4.2.0", "validate-npm-package-name": "^7.0.1", "zod": "^3.24.1", "zod-to-json-schema": "^3.24.6" }, "bin": { "shadcn": "dist/index.js" } }, "sha512-B2J63uYwqfLJPNzZk2AAlonXG1aJhwv8xmXlUag8OZdV61CQB6570nkXLjU1u9XN1qXm0LpJhSFQ0ZhlOruA4Q=="], + + "sharp": ["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" } }, "sha512-Ou9I5Ft9WNcCbXrU9cMgPBcCK8LiwLqcbywW3t4oDV37n1pzpuNLsYiAV8eODnjbtQlSDwZ2cUEeQz4E54Hltg=="], + + "shebang-command": ["shebang-command@2.0.0", "", { "dependencies": { "shebang-regex": "^3.0.0" } }, "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA=="], + + "shebang-regex": ["shebang-regex@3.0.0", "", {}, "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A=="], + + "side-channel": ["side-channel@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3", "side-channel-list": "^1.0.0", "side-channel-map": "^1.0.1", "side-channel-weakmap": "^1.0.2" } }, "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw=="], + + "side-channel-list": ["side-channel-list@1.0.1", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.4" } }, "sha512-mjn/0bi/oUURjc5Xl7IaWi/OJJJumuoJFQJfDDyO46+hBWsfaVM65TBHq2eoZBhzl9EchxOijpkbRC8SVBQU0w=="], + + "side-channel-map": ["side-channel-map@1.0.1", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3" } }, "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA=="], + + "side-channel-weakmap": ["side-channel-weakmap@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3", "side-channel-map": "^1.0.1" } }, "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A=="], + + "signal-exit": ["signal-exit@4.1.0", "", {}, "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw=="], + + "simple-concat": ["simple-concat@1.0.1", "", {}, "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q=="], + + "simple-get": ["simple-get@4.0.1", "", { "dependencies": { "decompress-response": "^6.0.0", "once": "^1.3.1", "simple-concat": "^1.0.0" } }, "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA=="], + + "sisteransi": ["sisteransi@1.0.5", "", {}, "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg=="], + + "smart-buffer": ["smart-buffer@4.2.0", "", {}, "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg=="], + + "smol-toml": ["smol-toml@1.6.1", "", {}, "sha512-dWUG8F5sIIARXih1DTaQAX4SsiTXhInKf1buxdY9DIg4ZYPZK5nGM1VRIYmEbDbsHt7USo99xSLFu5Q1IqTmsg=="], + + "socks": ["socks@2.8.9", "", { "dependencies": { "ip-address": "^10.1.1", "smart-buffer": "^4.2.0" } }, "sha512-LJhUYUvItdQ0LkJTmPeaEObWXAqFyfmP85x0tch/ez9cahmhlBBLbIqDFnvBnUJGagb0JbIQrkBs1wJ+yRYpEw=="], + + "socks-proxy-agent": ["socks-proxy-agent@8.0.5", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "^4.3.4", "socks": "^2.8.3" } }, "sha512-HehCEsotFqbPW9sJ8WVYB6UbmIMv7kUUORIF2Nncq4VQvBfNBLibW9YZR5dlYCSUhwcD628pRllm7n+E+YTzJw=="], + + "sonner": ["sonner@2.0.7", "", { "peerDependencies": { "react": "^18.0.0 || ^19.0.0 || ^19.0.0-rc", "react-dom": "^18.0.0 || ^19.0.0 || ^19.0.0-rc" } }, "sha512-W6ZN4p58k8aDKA4XPcx2hpIQXBRAgyiWVkYhT7CvK6D3iAu7xjvVyhQHg2/iaKJZ1XVJ4r7XuwGL+WGEK37i9w=="], + + "source-map": ["source-map@0.6.1", "", {}, "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g=="], + + "source-map-js": ["source-map-js@1.2.1", "", {}, "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA=="], + + "source-map-support": ["source-map-support@0.5.21", "", { "dependencies": { "buffer-from": "^1.0.0", "source-map": "^0.6.0" } }, "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w=="], + + "space-separated-tokens": ["space-separated-tokens@2.0.2", "", {}, "sha512-PEGlAwrG8yXGXRjW32fGbg66JAlOAwbObuqVoJpv/mRgoWDQfgH1wDPvtzWyUSNAXBGSk8h755YDbbcEy3SH2Q=="], + + "stable-hash": ["stable-hash@0.0.5", "", {}, "sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA=="], + + "standardwebhooks": ["standardwebhooks@1.0.0", "", { "dependencies": { "@stablelib/base64": "^1.0.0", "fast-sha256": "^1.3.0" } }, "sha512-BbHGOQK9olHPMvQNHWul6MYlrRTAOKn03rOe4A8O3CLWhNf4YHBqq2HJKKC+sfqpxiBY52pNeesD6jIiLDz8jg=="], + + "statuses": ["statuses@2.0.2", "", {}, "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw=="], + + "stdin-discarder": ["stdin-discarder@0.2.2", "", {}, "sha512-UhDfHmA92YAlNnCfhmq0VeNL5bDbiZGg7sZ2IvPsXubGkiNa9EC+tUTsjBRsYUAz87btI6/1wf4XoVvQ3uRnmQ=="], + + "stop-iteration-iterator": ["stop-iteration-iterator@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "internal-slot": "^1.1.0" } }, "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ=="], + + "streamx": ["streamx@2.26.0", "", { "dependencies": { "events-universal": "^1.0.0", "fast-fifo": "^1.3.2", "text-decoder": "^1.1.0" } }, "sha512-VvNG1K72Po/xwJzxZFnZ++Tbrv4lwSptsbkFuzXCJAYZvCK5nnxsvXU6ajqkv7chyiI1Y0YXq2Jh8Iy8Y7NF/A=="], + + "strict-event-emitter": ["strict-event-emitter@0.5.1", "", {}, "sha512-vMgjE/GGEPEFnhFub6pa4FmJBRBVOLpIII2hvCZ8Kzb7K0hlHo7mQv6xYrBvCL2LtAIBwFUK8wvuJgTVSQ5MFQ=="], + + "string-width": ["string-width@7.2.0", "", { "dependencies": { "emoji-regex": "^10.3.0", "get-east-asian-width": "^1.0.0", "strip-ansi": "^7.1.0" } }, "sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ=="], + + "string.prototype.includes": ["string.prototype.includes@2.0.1", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-abstract": "^1.23.3" } }, "sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg=="], + + "string.prototype.matchall": ["string.prototype.matchall@4.0.12", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "define-properties": "^1.2.1", "es-abstract": "^1.23.6", "es-errors": "^1.3.0", "es-object-atoms": "^1.0.0", "get-intrinsic": "^1.2.6", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "internal-slot": "^1.1.0", "regexp.prototype.flags": "^1.5.3", "set-function-name": "^2.0.2", "side-channel": "^1.1.0" } }, "sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA=="], + + "string.prototype.repeat": ["string.prototype.repeat@1.0.0", "", { "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.5" } }, "sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w=="], + + "string.prototype.trim": ["string.prototype.trim@1.2.10", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.2", "define-data-property": "^1.1.4", "define-properties": "^1.2.1", "es-abstract": "^1.23.5", "es-object-atoms": "^1.0.0", "has-property-descriptors": "^1.0.2" } }, "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA=="], + + "string.prototype.trimend": ["string.prototype.trimend@1.0.9", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.2", "define-properties": "^1.2.1", "es-object-atoms": "^1.0.0" } }, "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ=="], + + "string.prototype.trimstart": ["string.prototype.trimstart@1.0.8", "", { "dependencies": { "call-bind": "^1.0.7", "define-properties": "^1.2.1", "es-object-atoms": "^1.0.0" } }, "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg=="], + + "string_decoder": ["string_decoder@1.3.0", "", { "dependencies": { "safe-buffer": "~5.2.0" } }, "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA=="], + + "stringify-entities": ["stringify-entities@4.0.4", "", { "dependencies": { "character-entities-html4": "^2.0.0", "character-entities-legacy": "^3.0.0" } }, "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg=="], + + "stringify-object": ["stringify-object@5.0.0", "", { "dependencies": { "get-own-enumerable-keys": "^1.0.0", "is-obj": "^3.0.0", "is-regexp": "^3.1.0" } }, "sha512-zaJYxz2FtcMb4f+g60KsRNFOpVMUyuJgA51Zi5Z1DOTC3S59+OQiVOzE9GZt0x72uBGWKsQIuBKeF9iusmKFsg=="], + + "strip-ansi": ["strip-ansi@7.2.0", "", { "dependencies": { "ansi-regex": "^6.2.2" } }, "sha512-yDPMNjp4WyfYBkHnjIRLfca1i6KMyGCtsVgoKe/z1+6vukgaENdgGBZt+ZmKPc4gavvEZ5OgHfHdrazhgNyG7w=="], + + "strip-bom": ["strip-bom@3.0.0", "", {}, "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA=="], + + "strip-final-newline": ["strip-final-newline@4.0.0", "", {}, "sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw=="], + + "strip-json-comments": ["strip-json-comments@5.0.3", "", {}, "sha512-1tB5mhVo7U+ETBKNf92xT4hrQa3pm0MZ0PQvuDnWgAAGHDsfp4lPSpiS6psrSiet87wyGPh9ft6wmhOMQ0hDiw=="], + + "style-to-js": ["style-to-js@1.1.21", "", { "dependencies": { "style-to-object": "1.0.14" } }, "sha512-RjQetxJrrUJLQPHbLku6U/ocGtzyjbJMP9lCNK7Ag0CNh690nSH8woqWH9u16nMjYBAok+i7JO1NP2pOy8IsPQ=="], + + "style-to-object": ["style-to-object@1.0.14", "", { "dependencies": { "inline-style-parser": "0.2.7" } }, "sha512-LIN7rULI0jBscWQYaSswptyderlarFkjQ+t79nzty8tcIAceVomEVlLzH5VP4Cmsv6MtKhs7qaAiwlcp+Mgaxw=="], + + "styled-jsx": ["styled-jsx@5.1.6", "", { "dependencies": { "client-only": "0.0.1" }, "peerDependencies": { "react": ">= 16.8.0 || 17.x.x || ^18.0.0-0 || ^19.0.0-0" } }, "sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA=="], + + "supports-color": ["supports-color@7.2.0", "", { "dependencies": { "has-flag": "^4.0.0" } }, "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw=="], + + "supports-preserve-symlinks-flag": ["supports-preserve-symlinks-flag@1.0.0", "", {}, "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w=="], + + "swr": ["swr@2.4.1", "", { "dependencies": { "dequal": "^2.0.3", "use-sync-external-store": "^1.6.0" }, "peerDependencies": { "react": "^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-2CC6CiKQtEwaEeNiqWTAw9PGykW8SR5zZX8MZk6TeAvEAnVS7Visz8WzphqgtQ8v2xz/4Q5K+j+SeMaKXeeQIA=="], + + "tagged-tag": ["tagged-tag@1.0.0", "", {}, "sha512-yEFYrVhod+hdNyx7g5Bnkkb0G6si8HJurOoOEgC8B/O0uXLHlaey/65KRv6cuWBNhBgHKAROVpc7QyYqE5gFng=="], + + "tailwind-merge": ["tailwind-merge@3.6.0", "", {}, "sha512-uxL7qAVQriqRQPAyK3pj66VqskWqoZ37PW94jwOTwNfq/z9oyu1V+eqrZqtR2+fCiXdYOZe/Modt8GtvqNzu+w=="], + + "tailwindcss": ["tailwindcss@4.3.0", "", {}, "sha512-y6nxMGB1nMW9R6k96e5gdIFzcfL/gTJRNaqGes1YvkLnPVXzWgbqFF2yLC0T8G774n24cx3Pe8XrKoniCOAH+Q=="], + + "tapable": ["tapable@2.3.3", "", {}, "sha512-uxc/zpqFg6x7C8vOE7lh6Lbda8eEL9zmVm/PLeTPBRhh1xCgdWaQ+J1CUieGpIfm2HdtsUpRv+HshiasBMcc6A=="], + + "tar-fs": ["tar-fs@3.1.2", "", { "dependencies": { "pump": "^3.0.0", "tar-stream": "^3.1.5" }, "optionalDependencies": { "bare-fs": "^4.0.1", "bare-path": "^3.0.0" } }, "sha512-QGxxTxxyleAdyM3kpFs14ymbYmNFrfY+pHj7Z8FgtbZ7w2//VAgLMac7sT6nRpIHjppXO2AwwEOg0bPFVRcmXw=="], + + "tar-stream": ["tar-stream@3.2.0", "", { "dependencies": { "b4a": "^1.6.4", "bare-fs": "^4.5.5", "fast-fifo": "^1.2.0", "streamx": "^2.15.0" } }, "sha512-ojzvCvVaNp6aOTFmG7jaRD0meowIAuPc3cMMhSgKiVWws1GyHbGd/xvnyuRKcKlMpt3qvxx6r0hreCNITP9hIg=="], + + "teex": ["teex@1.0.1", "", { "dependencies": { "streamx": "^2.12.5" } }, "sha512-eYE6iEI62Ni1H8oIa7KlDU6uQBtqr4Eajni3wX7rpfXD8ysFx8z0+dri+KWEPWpBsxXfxu58x/0jvTVT1ekOSg=="], + + "text-decoder": ["text-decoder@1.2.7", "", { "dependencies": { "b4a": "^1.6.4" } }, "sha512-vlLytXkeP4xvEq2otHeJfSQIRyWxo/oZGEbXrtEEF9Hnmrdly59sUbzZ/QgyWuLYHctCHxFF4tRQZNQ9k60ExQ=="], + + "throttleit": ["throttleit@2.1.0", "", {}, "sha512-nt6AMGKW1p/70DF/hGBdJB57B8Tspmbp5gfJ8ilhLnt7kkr2ye7hzD6NVG8GGErk2HWF34igrL2CXmNIkzKqKw=="], + + "tiny-invariant": ["tiny-invariant@1.3.3", "", {}, "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg=="], + + "tinyglobby": ["tinyglobby@0.2.16", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.4" } }, "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg=="], + + "tldts": ["tldts@7.4.0", "", { "dependencies": { "tldts-core": "^7.4.0" }, "bin": { "tldts": "bin/cli.js" } }, "sha512-yHBe+zVfzNZ3QfTPW/Z6KK1G2t340gFjMHqI/4KKSt/abzYydzuCnpqdaF5gCCABby+9Yfbj59oR5F2Fd5CBzg=="], + + "tldts-core": ["tldts-core@7.4.0", "", {}, "sha512-/mb9kRld+x1sIMXxWNOAp5m6C+D4GrAORWlJkOJ5dElvxdN1eutz/o7qHLp9gFvDF4Y3/L2xeScoxz6AbEo8rQ=="], + + "to-regex-range": ["to-regex-range@5.0.1", "", { "dependencies": { "is-number": "^7.0.0" } }, "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ=="], + + "toidentifier": ["toidentifier@1.0.1", "", {}, "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA=="], + + "tough-cookie": ["tough-cookie@6.0.1", "", { "dependencies": { "tldts": "^7.0.5" } }, "sha512-LktZQb3IeoUWB9lqR5EWTHgW/VTITCXg4D21M+lvybRVdylLrRMnqaIONLVb5mav8vM19m44HIcGq4qASeu2Qw=="], + + "trim-lines": ["trim-lines@3.0.1", "", {}, "sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg=="], + + "trough": ["trough@2.2.0", "", {}, "sha512-tmMpK00BjZiUyVyvrBK7knerNgmgvcV/KLVyuma/SC+TQN167GrMRciANTz09+k3zW8L8t60jWO1GpfkZdjTaw=="], + + "ts-algebra": ["ts-algebra@2.0.0", "", {}, "sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw=="], + + "ts-api-utils": ["ts-api-utils@2.5.0", "", { "peerDependencies": { "typescript": ">=4.8.4" } }, "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA=="], + + "ts-morph": ["ts-morph@26.0.0", "", { "dependencies": { "@ts-morph/common": "~0.27.0", "code-block-writer": "^13.0.3" } }, "sha512-ztMO++owQnz8c/gIENcM9XfCEzgoGphTv+nKpYNM1bgsdOVC/jRZuEBf6N+mLLDNg68Kl+GgUZfOySaRiG1/Ug=="], + + "tsconfig-paths": ["tsconfig-paths@4.2.0", "", { "dependencies": { "json5": "^2.2.2", "minimist": "^1.2.6", "strip-bom": "^3.0.0" } }, "sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg=="], + + "tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="], + + "tsx": ["tsx@4.22.4", "", { "dependencies": { "esbuild": "~0.28.0" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "bin": { "tsx": "dist/cli.mjs" } }, "sha512-X8EX+XV4QR5xCsrgxaED954zTDfY8KqlDtskKEL0cHhyS/P8b4IFOvGDQpsC9Q1XnLq915wEfwwY/zzskCtmhg=="], + + "tunnel-agent": ["tunnel-agent@0.6.0", "", { "dependencies": { "safe-buffer": "^5.0.1" } }, "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w=="], + + "tw-animate-css": ["tw-animate-css@1.4.0", "", {}, "sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ=="], + + "type-check": ["type-check@0.4.0", "", { "dependencies": { "prelude-ls": "^1.2.1" } }, "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew=="], + + "type-fest": ["type-fest@5.6.0", "", { "dependencies": { "tagged-tag": "^1.0.0" } }, "sha512-8ZiHFm91orbSAe2PSAiSVBVko18pbhbiB3U9GglSzF/zCGkR+rxpHx6sEMCUm4kxY4LjDIUGgCfUMtwfZfjfUA=="], + + "type-is": ["type-is@2.1.0", "", { "dependencies": { "content-type": "^2.0.0", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-faYHw0anBbc/kWF3zFTEnxSFOAGUX9GFbOBthvDdLsIlEoWOFOtS0zgCiQYwIskL9iGXZL3kAXD8OoZ4GmMATA=="], + + "typed-array-buffer": ["typed-array-buffer@1.0.3", "", { "dependencies": { "call-bound": "^1.0.3", "es-errors": "^1.3.0", "is-typed-array": "^1.1.14" } }, "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw=="], + + "typed-array-byte-length": ["typed-array-byte-length@1.0.3", "", { "dependencies": { "call-bind": "^1.0.8", "for-each": "^0.3.3", "gopd": "^1.2.0", "has-proto": "^1.2.0", "is-typed-array": "^1.1.14" } }, "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg=="], + + "typed-array-byte-offset": ["typed-array-byte-offset@1.0.4", "", { "dependencies": { "available-typed-arrays": "^1.0.7", "call-bind": "^1.0.8", "for-each": "^0.3.3", "gopd": "^1.2.0", "has-proto": "^1.2.0", "is-typed-array": "^1.1.15", "reflect.getprototypeof": "^1.0.9" } }, "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ=="], + + "typed-array-length": ["typed-array-length@1.0.7", "", { "dependencies": { "call-bind": "^1.0.7", "for-each": "^0.3.3", "gopd": "^1.0.1", "is-typed-array": "^1.1.13", "possible-typed-array-names": "^1.0.0", "reflect.getprototypeof": "^1.0.6" } }, "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg=="], + + "typed-query-selector": ["typed-query-selector@2.12.2", "", {}, "sha512-EOPFbyIub4ngnEdqi2yOcNeDLaX/0jcE1JoAXQDDMIthap7FoN795lc/SHfIq2d416VufXpM8z/lD+WRm2gfOQ=="], + + "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], + + "typescript-eslint": ["typescript-eslint@8.60.0", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.60.0", "@typescript-eslint/parser": "8.60.0", "@typescript-eslint/typescript-estree": "8.60.0", "@typescript-eslint/utils": "8.60.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.1.0" } }, "sha512-9f65qWLZdAW9m1JaxBDUHcqRUfL8bkxxXL7XxEfI+F09q56PkBvIfCjLF3yInsDM/BBmwkqmCQdCZe/RYlIWEw=="], + + "unbash": ["unbash@2.2.0", "", {}, "sha512-X2wH19RAPZE3+ldGicOkoj/SIA83OIxcJ6Cuaw23hf8Xc6fQpvZXY0SftE2JgS0QhYLUG4uwodSI3R53keyh7w=="], + + "unbox-primitive": ["unbox-primitive@1.1.0", "", { "dependencies": { "call-bound": "^1.0.3", "has-bigints": "^1.0.2", "has-symbols": "^1.1.0", "which-boxed-primitive": "^1.1.1" } }, "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw=="], + + "undici-types": ["undici-types@6.21.0", "", {}, "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ=="], + + "unicorn-magic": ["unicorn-magic@0.3.0", "", {}, "sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA=="], + + "unified": ["unified@11.0.5", "", { "dependencies": { "@types/unist": "^3.0.0", "bail": "^2.0.0", "devlop": "^1.0.0", "extend": "^3.0.0", "is-plain-obj": "^4.0.0", "trough": "^2.0.0", "vfile": "^6.0.0" } }, "sha512-xKvGhPWw3k84Qjh8bI3ZeJjqnyadK+GEFtazSfZv/rKeTkTjOJho6mFqh2SM96iIcZokxiOpg78GazTSg8+KHA=="], + + "unist-util-is": ["unist-util-is@6.0.1", "", { "dependencies": { "@types/unist": "^3.0.0" } }, "sha512-LsiILbtBETkDz8I9p1dQ0uyRUWuaQzd/cuEeS1hoRSyW5E5XGmTzlwY1OrNzzakGowI9Dr/I8HVaw4hTtnxy8g=="], + + "unist-util-position": ["unist-util-position@5.0.0", "", { "dependencies": { "@types/unist": "^3.0.0" } }, "sha512-fucsC7HjXvkB5R3kTCO7kUjRdrS0BJt3M/FPxmHMBOm8JQi2BsHAHFsy27E0EolP8rp0NzXsJ+jNPyDWvOJZPA=="], + + "unist-util-stringify-position": ["unist-util-stringify-position@4.0.0", "", { "dependencies": { "@types/unist": "^3.0.0" } }, "sha512-0ASV06AAoKCDkS2+xw5RXJywruurpbC4JZSm7nr7MOt1ojAzvyyaO+UxZf18j8FCF6kmzCZKcAgN/yu2gm2XgQ=="], + + "unist-util-visit": ["unist-util-visit@5.1.0", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0", "unist-util-visit-parents": "^6.0.0" } }, "sha512-m+vIdyeCOpdr/QeQCu2EzxX/ohgS8KbnPDgFni4dQsfSCtpz8UqDyY5GjRru8PDKuYn7Fq19j1CQ+nJSsGKOzg=="], + + "unist-util-visit-parents": ["unist-util-visit-parents@6.0.2", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-is": "^6.0.0" } }, "sha512-goh1s1TBrqSqukSc8wrjwWhL0hiJxgA8m4kFxGlQ+8FYQ3C/m11FcTs4YYem7V664AhHVvgoQLk890Ssdsr2IQ=="], + + "universalify": ["universalify@2.0.1", "", {}, "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw=="], + + "unpipe": ["unpipe@1.0.0", "", {}, "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ=="], + + "unrs-resolver": ["unrs-resolver@1.12.2", "", { "dependencies": { "napi-postinstall": "^0.3.4" }, "optionalDependencies": { "@unrs/resolver-binding-android-arm-eabi": "1.12.2", "@unrs/resolver-binding-android-arm64": "1.12.2", "@unrs/resolver-binding-darwin-arm64": "1.12.2", "@unrs/resolver-binding-darwin-x64": "1.12.2", "@unrs/resolver-binding-freebsd-x64": "1.12.2", "@unrs/resolver-binding-linux-arm-gnueabihf": "1.12.2", "@unrs/resolver-binding-linux-arm-musleabihf": "1.12.2", "@unrs/resolver-binding-linux-arm64-gnu": "1.12.2", "@unrs/resolver-binding-linux-arm64-musl": "1.12.2", "@unrs/resolver-binding-linux-loong64-gnu": "1.12.2", "@unrs/resolver-binding-linux-loong64-musl": "1.12.2", "@unrs/resolver-binding-linux-ppc64-gnu": "1.12.2", "@unrs/resolver-binding-linux-riscv64-gnu": "1.12.2", "@unrs/resolver-binding-linux-riscv64-musl": "1.12.2", "@unrs/resolver-binding-linux-s390x-gnu": "1.12.2", "@unrs/resolver-binding-linux-x64-gnu": "1.12.2", "@unrs/resolver-binding-linux-x64-musl": "1.12.2", "@unrs/resolver-binding-openharmony-arm64": "1.12.2", "@unrs/resolver-binding-wasm32-wasi": "1.12.2", "@unrs/resolver-binding-win32-arm64-msvc": "1.12.2", "@unrs/resolver-binding-win32-ia32-msvc": "1.12.2", "@unrs/resolver-binding-win32-x64-msvc": "1.12.2" } }, "sha512-dmlRxBJJayXjqTwC+JtF1HhJmgf3ftQ3YejFcZrf4+KKtJv0qDsK1pjqaaVjG7wJ5NJ6UVP1OqRMQ71Z4C3rxQ=="], + + "until-async": ["until-async@3.0.2", "", {}, "sha512-IiSk4HlzAMqTUseHHe3VhIGyuFmN90zMTpD3Z3y8jeQbzLIq500MVM7Jq2vUAnTKAFPJrqwkzr6PoTcPhGcOiw=="], + + "update-browserslist-db": ["update-browserslist-db@1.2.3", "", { "dependencies": { "escalade": "^3.2.0", "picocolors": "^1.1.1" }, "peerDependencies": { "browserslist": ">= 4.21.0" }, "bin": { "update-browserslist-db": "cli.js" } }, "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w=="], + + "uri-js": ["uri-js@4.4.1", "", { "dependencies": { "punycode": "^2.1.0" } }, "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg=="], + + "use-intl": ["use-intl@4.12.0", "", { "dependencies": { "@formatjs/fast-memoize": "^3.1.0", "@schummar/icu-type-parser": "1.21.5", "icu-minify": "^4.12.0", "intl-messageformat": "^11.1.0" }, "peerDependencies": { "react": "^17.0.0 || ^18.0.0 || >=19.0.0-rc <19.0.0 || ^19.0.0" } }, "sha512-r+qVb7UI1+kiOhjYsmsNUCY+jrnjVopwGeFlmMyQj4YInlwZzgMeMSv9n8MqnWWy77HL5BVM8K2WgX50SbtcpA=="], + + "use-sync-external-store": ["use-sync-external-store@1.6.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w=="], + + "util-deprecate": ["util-deprecate@1.0.2", "", {}, "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="], + + "validate-npm-package-name": ["validate-npm-package-name@7.0.2", "", {}, "sha512-hVDIBwsRruT73PbK7uP5ebUt+ezEtCmzZz3F59BSr2F6OVFnJ/6h8liuvdLrQ88Xmnk6/+xGGuq+pG9WwTuy3A=="], + + "vary": ["vary@1.1.2", "", {}, "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg=="], + + "vfile": ["vfile@6.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "vfile-message": "^4.0.0" } }, "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q=="], + + "vfile-message": ["vfile-message@4.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-stringify-position": "^4.0.0" } }, "sha512-QTHzsGd1EhbZs4AsQ20JX1rC3cOlt/IWJruk893DfLRr57lcnOeMaWG4K0JrRta4mIJZKth2Au3mM3u03/JWKw=="], + + "victory-vendor": ["victory-vendor@37.3.6", "", { "dependencies": { "@types/d3-array": "^3.0.3", "@types/d3-ease": "^3.0.0", "@types/d3-interpolate": "^3.0.1", "@types/d3-scale": "^4.0.2", "@types/d3-shape": "^3.1.0", "@types/d3-time": "^3.0.0", "@types/d3-timer": "^3.0.0", "d3-array": "^3.1.6", "d3-ease": "^3.0.1", "d3-interpolate": "^3.0.1", "d3-scale": "^4.0.2", "d3-shape": "^3.1.0", "d3-time": "^3.0.0", "d3-timer": "^3.0.1" } }, "sha512-SbPDPdDBYp+5MJHhBCAyI7wKM3d5ivekigc2Dk2s7pgbZ9wIgIBYGVw4zGHBml/qTFbexrofXW6Gu4noGxrOwQ=="], + + "walk-up-path": ["walk-up-path@4.0.0", "", {}, "sha512-3hu+tD8YzSLGuFYtPRb48vdhKMi0KQV5sn+uWr8+7dMEq/2G/dtLrdDinkLjqq5TIbIBjYJ4Ax/n3YiaW7QM8A=="], + + "web-streams-polyfill": ["web-streams-polyfill@3.3.3", "", {}, "sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw=="], + + "webdriver-bidi-protocol": ["webdriver-bidi-protocol@0.4.1", "", {}, "sha512-ARrjNjtWRRs2w4Tk7nqrf2gBI0QXWuOmMCx2hU+1jUt6d00MjMxURrhxhGbrsoiZKJrhTSTzbIrc554iKI10qw=="], + + "whatwg-fetch": ["whatwg-fetch@3.6.20", "", {}, "sha512-EqhiFU6daOA8kpjOWTL0olhVOF3i7OrFzSYiGsEMB8GcXS+RrzauAERX65xMeNWVqxA6HXH2m69Z9LaKKdisfg=="], + + "which": ["which@2.0.2", "", { "dependencies": { "isexe": "^2.0.0" }, "bin": { "node-which": "./bin/node-which" } }, "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA=="], + + "which-boxed-primitive": ["which-boxed-primitive@1.1.1", "", { "dependencies": { "is-bigint": "^1.1.0", "is-boolean-object": "^1.2.1", "is-number-object": "^1.1.1", "is-string": "^1.1.1", "is-symbol": "^1.1.1" } }, "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA=="], + + "which-builtin-type": ["which-builtin-type@1.2.1", "", { "dependencies": { "call-bound": "^1.0.2", "function.prototype.name": "^1.1.6", "has-tostringtag": "^1.0.2", "is-async-function": "^2.0.0", "is-date-object": "^1.1.0", "is-finalizationregistry": "^1.1.0", "is-generator-function": "^1.0.10", "is-regex": "^1.2.1", "is-weakref": "^1.0.2", "isarray": "^2.0.5", "which-boxed-primitive": "^1.1.0", "which-collection": "^1.0.2", "which-typed-array": "^1.1.16" } }, "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q=="], + + "which-collection": ["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.3" } }, "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw=="], + + "which-typed-array": ["which-typed-array@1.1.21", "", { "dependencies": { "available-typed-arrays": "^1.0.7", "call-bind": "^1.0.9", "call-bound": "^1.0.4", "for-each": "^0.3.5", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-tostringtag": "^1.0.2" } }, "sha512-zbRA8cVm6io/d5W8uIe2hblzN76/Wm3v/yiythQvr+dpBWeqhPSWIDNj4zOyHi4zKbMK6DN34Xsr9jPHJERAEw=="], + + "word-wrap": ["word-wrap@1.2.5", "", {}, "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA=="], + + "wrap-ansi": ["wrap-ansi@7.0.0", "", { "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q=="], + + "wrappy": ["wrappy@1.0.2", "", {}, "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="], + + "ws": ["ws@8.21.0", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g=="], + + "wsl-utils": ["wsl-utils@0.3.1", "", { "dependencies": { "is-wsl": "^3.1.0", "powershell-utils": "^0.1.0" } }, "sha512-g/eziiSUNBSsdDJtCLB8bdYEUMj4jR7AGeUo96p/3dTafgjHhpF4RiCFPiRILwjQoDXx5MqkBr4fwWtR3Ky4Wg=="], + + "y18n": ["y18n@5.0.8", "", {}, "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA=="], + + "yallist": ["yallist@3.1.1", "", {}, "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="], + + "yaml": ["yaml@2.9.0", "", { "bin": { "yaml": "bin.mjs" } }, "sha512-2AvhNX3mb8zd6Zy7INTtSpl1F15HW6Wnqj0srWlkKLcpYl/gMIMJiyuGq2KeI2YFxUPjdlB+3Lc10seMLtL4cA=="], + + "yargs": ["yargs@17.7.2", "", { "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", "yargs-parser": "^21.1.1" } }, "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w=="], + + "yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="], + + "yauzl": ["yauzl@2.10.0", "", { "dependencies": { "buffer-crc32": "~0.2.3", "fd-slicer": "~1.1.0" } }, "sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g=="], + + "yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="], + + "yocto-spinner": ["yocto-spinner@1.2.0", "", { "dependencies": { "yoctocolors": "^2.1.1" } }, "sha512-Yw0hUB6UA3o4YUgKy3oSe9a4cxoaZ9sBfYDw+JSxo6Id0KoJGoxzPA24qqUXYKBWABs/zDSGTz9kww7t3F0XGw=="], + + "yoctocolors": ["yoctocolors@2.1.2", "", {}, "sha512-CzhO+pFNo8ajLM2d2IW/R93ipy99LWjtwblvC1RsoSUMZgyLbYFr221TnSNT7GjGdYui6P459mw9JH/g/zW2ug=="], + + "zod": ["zod@4.4.3", "", {}, "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ=="], + + "zod-to-json-schema": ["zod-to-json-schema@3.25.2", "", { "peerDependencies": { "zod": "^3.25.28 || ^4" } }, "sha512-O/PgfnpT1xKSDeQYSCfRI5Gy3hPf91mKVDuYLUHZJMiDFptvP41MSnWofm8dnCm0256ZNfZIM7DSzuSMAFnjHA=="], + + "zod-validation-error": ["zod-validation-error@4.0.2", "", { "peerDependencies": { "zod": "^3.25.0 || ^4.0.0" } }, "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ=="], + + "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="], + + "@babel/helper-compilation-targets/lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="], + + "@base-ui/utils/reselect": ["reselect@5.2.0", "", {}, "sha512-AgZ3UOZm3YndfrJ4OYjgrT7bmCm/1iqkjvEfH/oYjzh6PD2qw4QuT3jjnXIrpdt4MTpMXclMT3lXbmRY+XRakw=="], + + "@dotenvx/dotenvx/commander": ["commander@11.1.0", "", {}, "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ=="], + + "@dotenvx/dotenvx/execa": ["execa@5.1.1", "", { "dependencies": { "cross-spawn": "^7.0.3", "get-stream": "^6.0.0", "human-signals": "^2.1.0", "is-stream": "^2.0.0", "merge-stream": "^2.0.0", "npm-run-path": "^4.0.1", "onetime": "^5.1.2", "signal-exit": "^3.0.3", "strip-final-newline": "^2.0.0" } }, "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg=="], + + "@dotenvx/dotenvx/which": ["which@4.0.0", "", { "dependencies": { "isexe": "^3.1.1" }, "bin": { "node-which": "bin/which.js" } }, "sha512-GlaYyEb07DPxYCKhKzplCWBJtvxZcZMrL+4UkrTSJHHPyZU4mYYTv3qaOe77H7EODLSSopAUFAc6W8U4yqvscg=="], + + "@esbuild-kit/core-utils/esbuild": ["esbuild@0.18.20", "", { "optionalDependencies": { "@esbuild/android-arm": "0.18.20", "@esbuild/android-arm64": "0.18.20", "@esbuild/android-x64": "0.18.20", "@esbuild/darwin-arm64": "0.18.20", "@esbuild/darwin-x64": "0.18.20", "@esbuild/freebsd-arm64": "0.18.20", "@esbuild/freebsd-x64": "0.18.20", "@esbuild/linux-arm": "0.18.20", "@esbuild/linux-arm64": "0.18.20", "@esbuild/linux-ia32": "0.18.20", "@esbuild/linux-loong64": "0.18.20", "@esbuild/linux-mips64el": "0.18.20", "@esbuild/linux-ppc64": "0.18.20", "@esbuild/linux-riscv64": "0.18.20", "@esbuild/linux-s390x": "0.18.20", "@esbuild/linux-x64": "0.18.20", "@esbuild/netbsd-x64": "0.18.20", "@esbuild/openbsd-x64": "0.18.20", "@esbuild/sunos-x64": "0.18.20", "@esbuild/win32-arm64": "0.18.20", "@esbuild/win32-ia32": "0.18.20", "@esbuild/win32-x64": "0.18.20" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-ceqxoedUrcayh7Y7ZX6NdbbDzGROiyVBgC4PriJThBKSVPWnnFHZAkfI1lJT8QFkOwH4qOS2SJkS4wvpGl8BpA=="], + + "@eslint-community/eslint-utils/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="], + + "@eslint/eslintrc/globals": ["globals@14.0.0", "", {}, "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ=="], + + "@eslint/eslintrc/strip-json-comments": ["strip-json-comments@3.1.1", "", {}, "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig=="], + + "@modelcontextprotocol/sdk/ajv": ["ajv@8.20.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", "require-from-string": "^2.0.2" } }, "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA=="], + + "@mswjs/interceptors/@open-draft/deferred-promise": ["@open-draft/deferred-promise@2.2.0", "", {}, "sha512-CecwLWx3rhxVQF6V4bAgPS5t+So2sTbPgAzafKkVizyi7tlwpcFpdFqq+wqF2OwNBmqFuu6tOyouTuxgpMfzmA=="], + + "@next/eslint-plugin-next/fast-glob": ["fast-glob@3.3.1", "", { "dependencies": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", "glob-parent": "^5.1.2", "merge2": "^1.3.0", "micromatch": "^4.0.4" } }, "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg=="], + + "@puppeteer/browsers/semver": ["semver@7.8.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg=="], + + "@reduxjs/toolkit/immer": ["immer@11.1.8", "", {}, "sha512-/tbkHMW7y10Lx6i1crLjD4/OhNkRG+Fo7byZHtah0547nIeXYcpIXaUh0IAQY6gO5459qpGGYapcEOHtFXkIuA=="], + + "@reduxjs/toolkit/reselect": ["reselect@5.2.0", "", {}, "sha512-AgZ3UOZm3YndfrJ4OYjgrT7bmCm/1iqkjvEfH/oYjzh6PD2qw4QuT3jjnXIrpdt4MTpMXclMT3lXbmRY+XRakw=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.10.0", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.1", "tslib": "^2.4.0" }, "bundled": true }, "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.10.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.2.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w=="], + + "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.4", "", { "dependencies": { "@tybys/wasm-util": "^0.10.1" }, "peerDependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1" }, "bundled": true }, "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow=="], + + "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.10.2", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg=="], + + "@tailwindcss/oxide-wasm32-wasi/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="], + + "@ts-morph/common/minimatch": ["minimatch@10.2.5", "", { "dependencies": { "brace-expansion": "^5.0.5" } }, "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg=="], + + "@typescript-eslint/eslint-plugin/ignore": ["ignore@7.0.5", "", {}, "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg=="], + + "@typescript-eslint/typescript-estree/minimatch": ["minimatch@10.2.5", "", { "dependencies": { "brace-expansion": "^5.0.5" } }, "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg=="], + + "@typescript-eslint/typescript-estree/semver": ["semver@7.8.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg=="], + + "@typescript-eslint/visitor-keys/eslint-visitor-keys": ["eslint-visitor-keys@5.0.1", "", {}, "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA=="], + + "ajv-formats/ajv": ["ajv@8.20.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", "require-from-string": "^2.0.2" } }, "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA=="], + + "chromium-bidi/zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="], + + "cliui/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + + "cliui/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "degenerator/ast-types": ["ast-types@0.13.4", "", { "dependencies": { "tslib": "^2.0.1" } }, "sha512-x1FCFnFifvYDDzTaLII71vG5uvDwgtmDTEVWAxrgeiR8VjMONcCXJx7E+USjDtHlwFmt9MysbqgF9b9Vjr6w+w=="], + + "enquirer/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "eslint-import-resolver-node/debug": ["debug@3.2.7", "", { "dependencies": { "ms": "^2.1.1" } }, "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ=="], + + "eslint-module-utils/debug": ["debug@3.2.7", "", { "dependencies": { "ms": "^2.1.1" } }, "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ=="], + + "eslint-plugin-import/debug": ["debug@3.2.7", "", { "dependencies": { "ms": "^2.1.1" } }, "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ=="], + + "eslint-plugin-import/tsconfig-paths": ["tsconfig-paths@3.15.0", "", { "dependencies": { "@types/json5": "^0.0.29", "json5": "^1.0.2", "minimist": "^1.2.6", "strip-bom": "^3.0.0" } }, "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg=="], + + "express/cookie": ["cookie@0.7.2", "", {}, "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w=="], + + "extract-zip/get-stream": ["get-stream@5.2.0", "", { "dependencies": { "pump": "^3.0.0" } }, "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA=="], + + "fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="], + + "get-uri/data-uri-to-buffer": ["data-uri-to-buffer@6.0.2", "", {}, "sha512-7hvf7/GW8e86rW0ptuwS3OcBGDjIi6SZva7hCyWC0yYry2cOPmLIjXAUHI6DK2HsnwJd9ifmt57i8eV2n4YNpw=="], + + "is-bun-module/semver": ["semver@7.8.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg=="], + + "log-symbols/chalk": ["chalk@5.6.2", "", {}, "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA=="], + + "log-symbols/is-unicode-supported": ["is-unicode-supported@1.3.0", "", {}, "sha512-43r2mRvz+8JRIKnWJ+3j8JtjRKZ6GmjzfaE/qiBJnikNnYv/6bagRJ1kUhNk8R5EX/GkobD+r+sfxCPJsiKBLQ=="], + + "mdast-util-find-and-replace/escape-string-regexp": ["escape-string-regexp@5.0.0", "", {}, "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw=="], + + "micromatch/picomatch": ["picomatch@2.3.2", "", {}, "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA=="], + + "node-abi/semver": ["semver@7.8.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg=="], + + "npm-run-path/path-key": ["path-key@4.0.0", "", {}, "sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ=="], + + "ora/chalk": ["chalk@5.6.2", "", {}, "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA=="], + + "parse-entities/@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="], + + "prebuild-install/tar-fs": ["tar-fs@2.1.4", "", { "dependencies": { "chownr": "^1.1.1", "mkdirp-classic": "^0.5.2", "pump": "^3.0.0", "tar-stream": "^2.1.4" } }, "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ=="], + + "prompts/kleur": ["kleur@3.0.3", "", {}, "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w=="], + + "rc/strip-json-comments": ["strip-json-comments@2.0.1", "", {}, "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ=="], + + "restore-cursor/onetime": ["onetime@7.0.0", "", { "dependencies": { "mimic-function": "^5.0.0" } }, "sha512-VXJjc87FScF88uafS3JllDgvAm+c/Slfz06lorj2uAY34rlUu0Nt+v8wreiImcrgAjjIHp1rXpTDlLOGw29WwQ=="], + + "router/path-to-regexp": ["path-to-regexp@8.4.2", "", {}, "sha512-qRcuIdP69NPm4qbACK+aDogI5CBDMi1jKe0ry5rSQJz8JVLsC7jV8XpiJjGRLLol3N+R5ihGYcrPLTno6pAdBA=="], + + "shadcn/zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="], + + "sharp/semver": ["semver@7.8.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-rkVq3IXh+4FDGch+KwzX3aV9W3kO54GyEgpvBzSyctDA6Xtd7RJQV1xmXbeQp5v7+VzLOfVqiutSE6GICgPFvg=="], + + "string-width/emoji-regex": ["emoji-regex@10.6.0", "", {}, "sha512-toUI84YS5YmxW219erniWD0CIVOo46xGKColeNQRgOzDorgBi1v4D71/OFzgD9GO2UGKIv1C3Sp8DAn0+j5w7A=="], + + "tsx/esbuild": ["esbuild@0.28.0", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.28.0", "@esbuild/android-arm": "0.28.0", "@esbuild/android-arm64": "0.28.0", "@esbuild/android-x64": "0.28.0", "@esbuild/darwin-arm64": "0.28.0", "@esbuild/darwin-x64": "0.28.0", "@esbuild/freebsd-arm64": "0.28.0", "@esbuild/freebsd-x64": "0.28.0", "@esbuild/linux-arm": "0.28.0", "@esbuild/linux-arm64": "0.28.0", "@esbuild/linux-ia32": "0.28.0", "@esbuild/linux-loong64": "0.28.0", "@esbuild/linux-mips64el": "0.28.0", "@esbuild/linux-ppc64": "0.28.0", "@esbuild/linux-riscv64": "0.28.0", "@esbuild/linux-s390x": "0.28.0", "@esbuild/linux-x64": "0.28.0", "@esbuild/netbsd-arm64": "0.28.0", "@esbuild/netbsd-x64": "0.28.0", "@esbuild/openbsd-arm64": "0.28.0", "@esbuild/openbsd-x64": "0.28.0", "@esbuild/openharmony-arm64": "0.28.0", "@esbuild/sunos-x64": "0.28.0", "@esbuild/win32-arm64": "0.28.0", "@esbuild/win32-ia32": "0.28.0", "@esbuild/win32-x64": "0.28.0" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-sNR9MHpXSUV/XB4zmsFKN+QgVG82Cc7+/aaxJ8Adi8hyOac+EXptIp45QBPaVyX3N70664wRbTcLTOemCAnyqw=="], + + "type-is/content-type": ["content-type@2.0.0", "", {}, "sha512-j/O/d7GcZCyNl7/hwZAb606rzqkyvaDctLmckbxLzHvFBzTJHuGEdodATcP3yIRoDrLHkIATJuvzbFlp/ki2cQ=="], + + "wrap-ansi/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + + "wrap-ansi/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "yargs/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + + "@dotenvx/dotenvx/execa/get-stream": ["get-stream@6.0.1", "", {}, "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg=="], + + "@dotenvx/dotenvx/execa/human-signals": ["human-signals@2.1.0", "", {}, "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw=="], + + "@dotenvx/dotenvx/execa/is-stream": ["is-stream@2.0.1", "", {}, "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg=="], + + "@dotenvx/dotenvx/execa/npm-run-path": ["npm-run-path@4.0.1", "", { "dependencies": { "path-key": "^3.0.0" } }, "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw=="], + + "@dotenvx/dotenvx/execa/signal-exit": ["signal-exit@3.0.7", "", {}, "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ=="], + + "@dotenvx/dotenvx/execa/strip-final-newline": ["strip-final-newline@2.0.0", "", {}, "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA=="], + + "@dotenvx/dotenvx/which/isexe": ["isexe@3.1.5", "", {}, "sha512-6B3tLtFqtQS4ekarvLVMZ+X+VlvQekbe4taUkf/rhVO3d/h0M2rfARm/pXLcPEsjjMsFgrFgSrhQIxcSVrBz8w=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.18.20", "", { "os": "android", "cpu": "arm" }, "sha512-fyi7TDI/ijKKNZTUJAQqiG5T7YjJXgnzkURqmGj13C6dCqckZBLdl4h7bkhHt/t0WP+zO9/zwroDvANaOqO5Sw=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.18.20", "", { "os": "android", "cpu": "arm64" }, "sha512-Nz4rJcchGDtENV0eMKUNa6L12zz2zBDXuhj/Vjh18zGqB44Bi7MBMSXjgunJgjRhCmKOjnPuZp4Mb6OKqtMHLQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.18.20", "", { "os": "android", "cpu": "x64" }, "sha512-8GDdlePJA8D6zlZYJV/jnrRAi6rOiNaCC/JclcXpB+KIuvfBN4owLtgzY2bsxnx666XjJx2kDPUmnTtR8qKQUg=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.18.20", "", { "os": "darwin", "cpu": "arm64" }, "sha512-bxRHW5kHU38zS2lPTPOyuyTm+S+eobPUnTNkdJEfAddYgEcll4xkT8DB9d2008DtTbl7uJag2HuE5NZAZgnNEA=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.18.20", "", { "os": "darwin", "cpu": "x64" }, "sha512-pc5gxlMDxzm513qPGbCbDukOdsGtKhfxD1zJKXjCCcU7ju50O7MeAZ8c4krSJcOIJGFR+qx21yMMVYwiQvyTyQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.18.20", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-yqDQHy4QHevpMAaxhhIwYPMv1NECwOvIpGCZkECn8w2WFHXjEwrBn3CeNIYsibZ/iZEUemj++M26W3cNR5h+Tw=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.18.20", "", { "os": "freebsd", "cpu": "x64" }, "sha512-tgWRPPuQsd3RmBZwarGVHZQvtzfEBOreNuxEMKFcd5DaDn2PbBxfwLcj4+aenoh7ctXcbXmOQIn8HI6mCSw5MQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.18.20", "", { "os": "linux", "cpu": "arm" }, "sha512-/5bHkMWnq1EgKr1V+Ybz3s1hWXok7mDFUMQ4cG10AfW3wL02PSZi5kFpYKrptDsgb2WAJIvRcDm+qIvXf/apvg=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.18.20", "", { "os": "linux", "cpu": "arm64" }, "sha512-2YbscF+UL7SQAVIpnWvYwM+3LskyDmPhe31pE7/aoTMFKKzIc9lLbyGUpmmb8a8AixOL61sQ/mFh3jEjHYFvdA=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.18.20", "", { "os": "linux", "cpu": "ia32" }, "sha512-P4etWwq6IsReT0E1KHU40bOnzMHoH73aXp96Fs8TIT6z9Hu8G6+0SHSw9i2isWrD2nbx2qo5yUqACgdfVGx7TA=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.18.20", "", { "os": "linux", "cpu": "none" }, "sha512-nXW8nqBTrOpDLPgPY9uV+/1DjxoQ7DoB2N8eocyq8I9XuqJ7BiAMDMf9n1xZM9TgW0J8zrquIb/A7s3BJv7rjg=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.18.20", "", { "os": "linux", "cpu": "none" }, "sha512-d5NeaXZcHp8PzYy5VnXV3VSd2D328Zb+9dEq5HE6bw6+N86JVPExrA6O68OPwobntbNJ0pzCpUFZTo3w0GyetQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.18.20", "", { "os": "linux", "cpu": "ppc64" }, "sha512-WHPyeScRNcmANnLQkq6AfyXRFr5D6N2sKgkFo2FqguP44Nw2eyDlbTdZwd9GYk98DZG9QItIiTlFLHJHjxP3FA=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.18.20", "", { "os": "linux", "cpu": "none" }, "sha512-WSxo6h5ecI5XH34KC7w5veNnKkju3zBRLEQNY7mv5mtBmrP/MjNBCAlsM2u5hDBlS3NGcTQpoBvRzqBcRtpq1A=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.18.20", "", { "os": "linux", "cpu": "s390x" }, "sha512-+8231GMs3mAEth6Ja1iK0a1sQ3ohfcpzpRLH8uuc5/KVDFneH6jtAJLFGafpzpMRO6DzJ6AvXKze9LfFMrIHVQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.18.20", "", { "os": "linux", "cpu": "x64" }, "sha512-UYqiqemphJcNsFEskc73jQ7B9jgwjWrSayxawS6UVFZGWrAAtkzjxSqnoclCXxWtfwLdzU+vTpcNYhpn43uP1w=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.18.20", "", { "os": "none", "cpu": "x64" }, "sha512-iO1c++VP6xUBUmltHZoMtCUdPlnPGdBom6IrO4gyKPFFVBKioIImVooR5I83nTew5UOYrk3gIJhbZh8X44y06A=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.18.20", "", { "os": "openbsd", "cpu": "x64" }, "sha512-e5e4YSsuQfX4cxcygw/UCPIEP6wbIL+se3sxPdCiMbFLBWu0eiZOJ7WoD+ptCLrmjZBK1Wk7I6D/I3NglUGOxg=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.18.20", "", { "os": "sunos", "cpu": "x64" }, "sha512-kDbFRFp0YpTQVVrqUd5FTYmWo45zGaXe0X8E1G/LKFC0v8x0vWrhOWSLITcCn63lmZIxfOMXtCfti/RxN/0wnQ=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.18.20", "", { "os": "win32", "cpu": "arm64" }, "sha512-ddYFR6ItYgoaq4v4JmQQaAI5s7npztfV4Ag6NrhiaW0RrnOXqBkgwZLofVTlq1daVTQNhtI5oieTvkRPfZrePg=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.18.20", "", { "os": "win32", "cpu": "ia32" }, "sha512-Wv7QBi3ID/rROT08SABTS7eV4hX26sVduqDOTe1MvGMjNd3EjOz4b7zeexIR62GTIEKrfJXKL9LFxTYgkyeu7g=="], + + "@esbuild-kit/core-utils/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.18.20", "", { "os": "win32", "cpu": "x64" }, "sha512-kTdfRcSiDfQca/y9QIkng02avJ+NCaQvrMejlsB3RRv5sE9rRoeBPISaZpKxHELzRxZyLvNts1P27W3wV+8geQ=="], + + "@modelcontextprotocol/sdk/ajv/json-schema-traverse": ["json-schema-traverse@1.0.0", "", {}, "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="], + + "@next/eslint-plugin-next/fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="], + + "@ts-morph/common/minimatch/brace-expansion": ["brace-expansion@5.0.6", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g=="], + + "@typescript-eslint/typescript-estree/minimatch/brace-expansion": ["brace-expansion@5.0.6", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g=="], + + "ajv-formats/ajv/json-schema-traverse": ["json-schema-traverse@1.0.0", "", {}, "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="], + + "cliui/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="], + + "cliui/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + + "enquirer/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + + "eslint-plugin-import/tsconfig-paths/json5": ["json5@1.0.2", "", { "dependencies": { "minimist": "^1.2.0" }, "bin": { "json5": "lib/cli.js" } }, "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA=="], + + "prebuild-install/tar-fs/tar-stream": ["tar-stream@2.2.0", "", { "dependencies": { "bl": "^4.0.3", "end-of-stream": "^1.4.1", "fs-constants": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^3.1.1" } }, "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ=="], + + "tsx/esbuild/@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.28.0", "", { "os": "aix", "cpu": "ppc64" }, "sha512-lhRUCeuOyJQURhTxl4WkpFTjIsbDayJHih5kZC1giwE+MhIzAb7mEsQMqMf18rHLsrb5qI1tafG20mLxEWcWlA=="], + + "tsx/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.28.0", "", { "os": "android", "cpu": "arm" }, "sha512-wqh0ByljabXLKHeWXYLqoJ5jKC4XBaw6Hk08OfMrCRd2nP2ZQ5eleDZC41XHyCNgktBGYMbqnrJKq/K/lzPMSQ=="], + + "tsx/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.28.0", "", { "os": "android", "cpu": "arm64" }, "sha512-+WzIXQOSaGs33tLEgYPYe/yQHf0WTU0X42Jca3y8NWMbUVhp7rUnw+vAsRC/QiDrdD31IszMrZy+qwPOPjd+rw=="], + + "tsx/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.28.0", "", { "os": "android", "cpu": "x64" }, "sha512-+VJggoaKhk2VNNqVL7f6S189UzShHC/mR9EE8rDdSkdpN0KflSwWY/gWjDrNxxisg8Fp1ZCD9jLMo4m0OUfeUA=="], + + "tsx/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.28.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-0T+A9WZm+bZ84nZBtk1ckYsOvyA3x7e2Acj1KdVfV4/2tdG4fzUp91YHx+GArWLtwqp77pBXVCPn2We7Letr0Q=="], + + "tsx/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.28.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-fyzLm/DLDl/84OCfp2f/XQ4flmORsjU7VKt8HLjvIXChJoFFOIL6pLJPH4Yhd1n1gGFF9mPwtlN5Wf82DZs+LQ=="], + + "tsx/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.28.0", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-l9GeW5UZBT9k9brBYI+0WDffcRxgHQD8ShN2Ur4xWq/NFzUKm3k5lsH4PdaRgb2w7mI9u61nr2gI2mLI27Nh3Q=="], + + "tsx/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.28.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-BXoQai/A0wPO6Es3yFJ7APCiKGc1tdAEOgeTNy3SsB491S3aHn4S4r3e976eUnPdU+NbdtmBuLncYir2tMU9Nw=="], + + "tsx/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.28.0", "", { "os": "linux", "cpu": "arm" }, "sha512-CjaaREJagqJp7iTaNQjjidaNbCKYcd4IDkzbwwxtSvjI7NZm79qiHc8HqciMddQ6CKvJT6aBd8lO9kN/ZudLlw=="], + + "tsx/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.28.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-RVyzfb3FWsGA55n6WY0MEIEPURL1FcbhFE6BffZEMEekfCzCIMtB5yyDcFnVbTnwk+CLAgTujmV/Lgvih56W+A=="], + + "tsx/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.28.0", "", { "os": "linux", "cpu": "ia32" }, "sha512-KBnSTt1kxl9x70q+ydterVdl+Cn0H18ngRMRCEQfrbqdUuntQQ0LoMZv47uB97NljZFzY6HcfqEZ2SAyIUTQBQ=="], + + "tsx/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.28.0", "", { "os": "linux", "cpu": "none" }, "sha512-zpSlUce1mnxzgBADvxKXX5sl8aYQHo2ezvMNI8I0lbblJtp8V4odlm3Yzlj7gPyt3T8ReksE6bK+pT3WD+aJRg=="], + + "tsx/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.28.0", "", { "os": "linux", "cpu": "none" }, "sha512-2jIfP6mmjkdmeTlsX/9vmdmhBmKADrWqN7zcdtHIeNSCH1SqIoNI63cYsjQR8J+wGa4Y5izRcSHSm8K3QWmk3w=="], + + "tsx/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.28.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-bc0FE9wWeC0WBm49IQMPSPILRocGTQt3j5KPCA8os6VprfuJ7KD+5PzESSrJ6GmPIPJK965ZJHTUlSA6GNYEhg=="], + + "tsx/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.28.0", "", { "os": "linux", "cpu": "none" }, "sha512-SQPZOwoTTT/HXFXQJG/vBX8sOFagGqvZyXcgLA3NhIqcBv1BJU1d46c0rGcrij2B56Z2rNiSLaZOYW5cUk7yLQ=="], + + "tsx/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.28.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-SCfR0HN8CEEjnYnySJTd2cw0k9OHB/YFzt5zgJEwa+wL/T/raGWYMBqwDNAC6dqFKmJYZoQBRfHjgwLHGSrn3Q=="], + + "tsx/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.28.0", "", { "os": "linux", "cpu": "x64" }, "sha512-us0dSb9iFxIi8srnpl931Nvs65it/Jd2a2K3qs7fz2WfGPHqzfzZTfec7oxZJRNPXPnNYZtanmRc4AL/JwVzHQ=="], + + "tsx/esbuild/@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.28.0", "", { "os": "none", "cpu": "arm64" }, "sha512-CR/RYotgtCKwtftMwJlUU7xCVNg3lMYZ0RzTmAHSfLCXw3NtZtNpswLEj/Kkf6kEL3Gw+BpOekRX0BYCtklhUw=="], + + "tsx/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.28.0", "", { "os": "none", "cpu": "x64" }, "sha512-nU1yhmYutL+fQ71Kxnhg8uEOdC0pwEW9entHykTgEbna2pw2dkbFSMeqjjyHZoCmt8SBkOSvV+yNmm94aUrrqw=="], + + "tsx/esbuild/@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.28.0", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-cXb5vApOsRsxsEl4mcZ1XY3D4DzcoMxR/nnc4IyqYs0rTI8ZKmW6kyyg+11Z8yvgMfAEldKzP7AdP64HnSC/6g=="], + + "tsx/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.28.0", "", { "os": "openbsd", "cpu": "x64" }, "sha512-8wZM2qqtv9UP3mzy7HiGYNH/zjTA355mpeuA+859TyR+e+Tc08IHYpLJuMsfpDJwoLo1ikIJI8jC3GFjnRClzA=="], + + "tsx/esbuild/@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.28.0", "", { "os": "none", "cpu": "arm64" }, "sha512-FLGfyizszcef5C3YtoyQDACyg95+dndv79i2EekILBofh5wpCa1KuBqOWKrEHZg3zrL3t5ouE5jgr94vA+Wb2w=="], + + "tsx/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.28.0", "", { "os": "sunos", "cpu": "x64" }, "sha512-1ZgjUoEdHZZl/YlV76TSCz9Hqj9h9YmMGAgAPYd+q4SicWNX3G5GCyx9uhQWSLcbvPW8Ni7lj4gDa1T40akdlw=="], + + "tsx/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.28.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-Q9StnDmQ/enxnpxCCLSg0oo4+34B9TdXpuyPeTedN/6+iXBJ4J+zwfQI28u/Jl40nOYAxGoNi7mFP40RUtkmUA=="], + + "tsx/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.28.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-zF3ag/gfiCe6U2iczcRzSYJKH1DCI+ByzSENHlM2FcDbEeo5Zd2C86Aq0tKUYAJJ1obRP84ymxIAksZUcdztHA=="], + + "tsx/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.28.0", "", { "os": "win32", "cpu": "x64" }, "sha512-pEl1bO9mfAmIC+tW5btTmrKaujg3zGtUmWNdCw/xs70FBjwAL3o9OEKNHvNmnyylD6ubxUERiEhdsL0xBQ9efw=="], + + "wrap-ansi/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="], + + "wrap-ansi/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + + "yargs/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="], + + "yargs/string-width/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "@ts-morph/common/minimatch/brace-expansion/balanced-match": ["balanced-match@4.0.4", "", {}, "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA=="], + + "@typescript-eslint/typescript-estree/minimatch/brace-expansion/balanced-match": ["balanced-match@4.0.4", "", {}, "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA=="], + + "yargs/string-width/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + } +} diff --git a/design.md b/design.md new file mode 100644 index 0000000..299e7b9 --- /dev/null +++ b/design.md @@ -0,0 +1,192 @@ +# Spent design system + +The single source of truth for how Spent looks and feels. Every screen, component, +and pull request should follow this. When something here conflicts with a component, +the component is wrong, fix the component. + +The goal: a **calm, cozy, trustworthy** finance app. Warm "buttercream" surfaces, +soft rounded cards, a serif display face for headlines, tabular numerals for money. +Nothing should look rough. + +All tokens live in [`src/app/globals.css`](src/app/globals.css) under `@theme inline` +and `:root` / `.dark`. Tailwind v4 exposes them as utilities (`bg-card`, +`text-muted-foreground`, `rounded-2xl`, `text-status-over`, ...). **Never hardcode a +hex / rgb / oklch value in a component** unless it is genuinely dynamic data (a +per-category color from the DB, a bank brand color). Use the token. + +--- + +## 1. Color + +Always use the semantic token, never a raw color. Light and dark are defined once in +`globals.css`; using the token means dark mode and future re-tints "just work". + +| Token (utility) | Use for | +| -------------------------- | ----------------------------------------------------------- | +| `bg-background` / `text-foreground` | App canvas + primary text | +| `bg-card` / `text-card-foreground` | Card and panel surfaces | +| `bg-popover` | Dialogs, sheets, dropdowns, tooltips | +| `bg-muted` / `text-muted-foreground`| Subtle fills, secondary/meta text | +| `bg-secondary` / `bg-accent`| Hover fills, quiet buttons, chips | +| `bg-primary` / `text-primary` | Primary actions, brand green, active nav | +| `bg-destructive` | Destructive actions only | +| `border-border` / `border-input` | All borders and input outlines | +| `ring-ring` | Focus rings (see Accessibility) | + +### Status colors (pace / health) + +Four semantic status tokens, registered as color utilities. Use `text-status-*`, +`bg-status-*`, `border-status-*` (with opacity modifiers like `bg-status-over/10`): + +| Token | Meaning | +| -------------------- | ------------------------------------ | +| `status-on-track` | Good / income / under budget / OK | +| `status-plenty-left` | Comfortably under | +| `status-heads-up` | Warning / approaching limit / review | +| `status-over` | Over budget / expense / error | + +Soft tinted backgrounds (banners, badges) use `color-mix` against the token so they +adapt per theme, e.g. `color-mix(in oklch, var(--status-over) 12%, var(--card))`. +Prefer the helpers in [`src/lib/colors.ts`](src/lib/colors.ts) / status utilities +over re-deriving the mix inline. + +### Charts & categories + +`chart-1`..`chart-5` are the pastel chart palette. Category swatches come from the +database (`category.color`) and are the one place inline color is correct. When a +category color is missing, fall back to `var(--muted-foreground)`, never a hardcoded +beige/grey hex. + +--- + +## 2. Typography + +Three families, mapped in `globals.css`: + +- **`font-serif`** (Fraunces) — display headings, hero numbers, card titles. Cozy, editorial. +- **`font-sans`** (Inter) — all body text, labels, controls. The default. +- **`font-mono`** (Geist Mono) — only where needed; numbers use `tabular-nums`, not mono. + +Rules: + +- **Money and any aligned number gets `tabular-nums`** so columns line up and values + don't jitter as they change. +- Page title: `font-serif text-2xl leading-none tracking-tight` (see `PageHeader`). +- Hero amount: `font-serif` at `text-4xl`/`text-5xl`. +- **Eyebrow / section label**: the small uppercase muted label is one component, + `CardLabel` ([`src/components/ui/card-label.tsx`](src/components/ui/card-label.tsx)): + `text-xs font-semibold uppercase tracking-[0.08em] text-muted-foreground`. + Do not hand-roll `text-[10px]/[11px]` uppercase spans, use ``. +- De-emphasized text is `text-muted-foreground`, not `text-foreground/60`. + +--- + +## 3. Radius + +Driven by `--radius: 0.875rem`. The scale (`rounded-sm`..`rounded-4xl`) is in `@theme`. + +- **Cards / panels / sheets / dialogs: `rounded-2xl`.** This is the canonical surface radius. +- Buttons, inputs, dropdown items, badges-as-rect: `rounded-md` / `rounded-lg`. +- Pills, status chips, fully-round toggles: `rounded-full`. + +Do not mix `rounded-xl` / `rounded-3xl` for cards. One surface radius. + +--- + +## 4. Surfaces & elevation + +The canonical card is: + +``` +rounded-2xl border border-border bg-card +``` + +A **border**, not a ring, not a shadow. The `Card` primitive +([`src/components/ui/card.tsx`](src/components/ui/card.tsx)), +`CardShell` (home), `HeroCard`, KPI cards, settings sections, and chat surfaces all +resolve to this. Reserve `shadow`/`ring` for transient layers (open dropdowns, +popovers, the dragged thumb), not resting cards. + +Padding: cards use `p-5 md:p-6`; hero/feature cards `p-6 md:p-8`. + +--- + +## 5. Spacing & layout + +- **Page padding**: `p-4 md:p-6 lg:p-8`. +- **Section gap** within a page: `space-y-6` (or `gap-4 md:gap-5 lg:gap-6` in the home grid). +- Content max width for reading/forms: `max-w-4xl` (settings), `max-w-3xl` (chat column). +- Every flex child that can shrink and truncate needs `min-w-0`. +- The home grid is 12 columns (`grid-cols-12`) with responsive `col-span-*`. + +--- + +## 6. Components + +- **PageHeader** ([`app-shell.tsx`](src/components/layout/app-shell.tsx)) — sticky, + `backdrop-blur`, serif title, optional meta + actions. Every top-level screen uses it, + including chat. Actions wrap on mobile. +- **Card / CardShell** — section container, section 4. +- **CardLabel** — the eyebrow, section 2. +- **ProgressBar** ([`src/components/ui/progress-bar.tsx`](src/components/ui/progress-bar.tsx)) + — the budget/pace meter. Clamps 0-100, optional elapsed marker, `tone` from status. +- **Button** ([`button.tsx`](src/components/ui/button.tsx)) — variants + `default | outline | secondary | ghost | destructive | link`; sizes + `default | xs | sm | lg | icon*`. Icon-only buttons MUST have `aria-label`. +- **Segmented filter / Tabs** — use the `Tabs` primitive (or shared segmented pill) for + view toggles; do not hand-build divergent active states. +- **Status pill / dot** — derive color from the status token, never amber/emerald literals. + +--- + +## 7. Iconography + +- Icons come from **`lucide-react`**, default size `size-4` (16px). No hand-drawn inline SVGs. +- Spinners are `Loader2` with `animate-spin`. +- Decorative icons get `aria-hidden`; meaningful icon-only controls get `aria-label`. + +--- + +## 8. Motion + +Keyframes live in `globals.css` (`fadeIn`, `countIn`, `pop`, `checkPop`, `dotPulse`). +Framer Motion is used for the setup wizard step transitions. Keep durations short +(150-280ms) and easing gentle. Respect `prefers-reduced-motion` where feasible. + +--- + +## 9. RTL & internationalization (hard rules) + +Spent ships English (LTR) and Hebrew (RTL); `` flips. Therefore: + +- **Use logical properties, never physical.** `ms-/me-` not `ml-/mr-`; `ps-/pe-` not + `pl-/pr-`; `start-/end-` not `left-/right-`; `text-start/text-end` not + `text-left/text-right`; `border-s/border-e`; `rounded-s/rounded-e`. The only + physical exception is true visual centering (`left-1/2 -translate-x-1/2`). +- Directional icons (chevrons, arrows) flip with `rtl:rotate-180`. +- Popover/sheet/menu sides use logical `inline-start`/`inline-end`. +- **All user-facing copy goes through `next-intl`** (`useTranslations` / + `getTranslations`). No hardcoded strings in components. Keep `en.json` and `he.json` + at full key parity (`bun run i18n:check`). +- Money via `formatCurrency(amount, currency, locale)`; dates via the `formatters.ts` + helpers. Respect `chargedCurrency`, do not assume ILS per-transaction. + +--- + +## 10. Accessibility + +- Visible focus: `focus-visible:ring-2 focus-visible:ring-ring` (built into Button). +- Every icon-only control has an accessible name; decorative glyphs are `aria-hidden`. +- Active nav exposes `aria-current="page"`. +- Inputs have associated labels; validation errors link via `aria-describedby`. +- Don't encode meaning in color alone, pair status color with text or an icon. +- Touch targets are comfortable (~40px+) on coarse pointers. + +--- + +## 11. Conventions + +- No em dashes anywhere (copy, comments, commits). Use commas, parentheses, or "to". +- `import "server-only"` at the top of every `src/server/` file. +- Comments only where the "why" isn't obvious. +- The full gate is `bun run ci` (format, typecheck, i18n, knip, react-doctor, test). diff --git a/docs/superpowers/plans/2026-05-17-docs-upgrade.md b/docs/superpowers/plans/2026-05-17-docs-upgrade.md index 33c51bd..3ac49c2 100644 --- a/docs/superpowers/plans/2026-05-17-docs-upgrade.md +++ b/docs/superpowers/plans/2026-05-17-docs-upgrade.md @@ -1777,7 +1777,7 @@ A Next.js app + a SQLite file + a headless Chromium tab that drives your bank's ## Status -**Beta.** Built by [Shay Avivi](https://github.com/Shaya16) for personal use first and then released. MIT licensed. Issues and PRs welcome on [GitHub](https://github.com/Shaya16/Spent). +**Beta.** Built by [Shay Avivi](https://github.com/Shaya16) for personal use first and then released. MIT licensed. Issues and PRs welcome on [GitHub](https://github.com/alon710/Spent). Things that work today: Isracard, Hapoalim, Max, Visa Cal, Leumi, Mizrahi, Discount, FIBI group, Yahav, One Zero (with OTP), Beyahad Bishvilha. Light + dark themes. Hebrew + English UI. @@ -1851,11 +1851,11 @@ brew install node@20 Open **Terminal** (in *Applications → Utilities*) and clone the repo: ```sh -git clone https://github.com/Shaya16/Spent.git ~/Applications/Spent +git clone https://github.com/alon710/Spent.git ~/Applications/Spent cd ~/Applications/Spent ``` -No Git? Download the [latest source as a zip](https://github.com/Shaya16/Spent/releases) and unzip it to `~/Applications/Spent`. Then in Terminal: `cd ~/Applications/Spent`. +No Git? Download the [latest source as a zip](https://github.com/alon710/Spent/releases) and unzip it to `~/Applications/Spent`. Then in Terminal: `cd ~/Applications/Spent`. @@ -1987,11 +1987,11 @@ When the installer asks *"Automatically install the necessary tools"* near the e Open **PowerShell** (as Administrator) and clone: ```powershell -git clone https://github.com/Shaya16/Spent.git $env:USERPROFILE\Spent +git clone https://github.com/alon710/Spent.git $env:USERPROFILE\Spent cd $env:USERPROFILE\Spent ``` -No Git? Download the [latest source zip](https://github.com/Shaya16/Spent/releases), right-click → *Extract All* into `%USERPROFILE%\Spent`, then `cd` there. +No Git? Download the [latest source zip](https://github.com/alon710/Spent/releases), right-click → *Extract All* into `%USERPROFILE%\Spent`, then `cd` there. @@ -2132,7 +2132,7 @@ sudo dnf install nodejs ```sh -git clone https://github.com/Shaya16/Spent.git ~/.local/share/spent +git clone https://github.com/alon710/Spent.git ~/.local/share/spent cd ~/.local/share/spent ``` @@ -2745,7 +2745,7 @@ The transaction list mixes scripts naturally: Hebrew merchants like "ארומה" - Charts (Recharts) don't fully mirror — the y-axis stays on the left. - Date inputs in some browsers still pop their picker in LTR. -If any of these block you, please file an issue on [GitHub](https://github.com/Shaya16/Spent/issues). +If any of these block you, please file an issue on [GitHub](https://github.com/alon710/Spent/issues). ``` @@ -2923,7 +2923,7 @@ Or copy the `.db`, `.db-wal`, and `.db-shm` files together while the service is ## Still stuck? -Open an issue on [GitHub](https://github.com/Shaya16/Spent/issues) with: +Open an issue on [GitHub](https://github.com/alon710/Spent/issues) with: - Your operating system and version - What you were doing when it broke @@ -3023,7 +3023,7 @@ In practice, this is rare for read-only scraping that happens infrequently. But If you find a security issue, please **don't** open a public issue. Email the maintainer (address in `SECURITY.md` in the repo). We'll respond within 72 hours. -For non-security bugs, [GitHub Issues](https://github.com/Shaya16/Spent/issues) is the right place. +For non-security bugs, [GitHub Issues](https://github.com/alon710/Spent/issues) is the right place. ## Auditing the code yourself @@ -3119,11 +3119,11 @@ You are responsible for the security of your own machine. ## Open source license -Spent is licensed under the **MIT License**. Use, modify, and redistribute it freely, subject to the license terms. See the [LICENSE](https://github.com/Shaya16/Spent/blob/main/LICENSE) file in the repository. +Spent is licensed under the **MIT License**. Use, modify, and redistribute it freely, subject to the license terms. See the [LICENSE](https://github.com/alon710/Spent/blob/main/LICENSE) file in the repository. ## Questions -If anything on this page is unclear, please open an issue at [github.com/Shaya16/Spent/issues](https://github.com/Shaya16/Spent/issues) so we can clarify it for everyone. +If anything on this page is unclear, please open an issue at [github.com/alon710/Spent/issues](https://github.com/alon710/Spent/issues) so we can clarify it for everyone. **By installing Spent and connecting a bank account, you confirm that you have read this disclaimer and accept the terms above.** diff --git a/docs/transaction-deduplication-design.md b/docs/transaction-deduplication-design.md new file mode 100644 index 0000000..2624ee2 --- /dev/null +++ b/docs/transaction-deduplication-design.md @@ -0,0 +1,3043 @@ +# Transaction Deduplication via Financial Events: Design for Spent + +## Summary + +The same real-world money event (an ATM withdrawal, a credit-card bill payment, an +internal transfer, a loan or investment payment) shows up as two or more rows once you +aggregate multiple accounts. Today Spent flattens this with a single `kind` flag +(`expense` / `income` / `transfer`) set by keyword rules in `detectKind` +(`src/server/lib/transfers.ts`) and a greedy 1:1 pairer in `findInternalTransferPairs` +(`src/server/lib/internal-transfers.ts`). That keeps transfers out of spend, but the +relationship between the rows is invisible, unauditable, has no confidence score, no +review/undo, and no link between a credit-card bill and the purchases that compose it. + +This design adds a first-class **Financial Event** layer above the existing +`transactions` table. A Financial Event groups the N rows that represent one real-world +event, records its type, a chosen **canonical transaction** for reporting, a confidence +score, and human-readable match reasons. It keeps the existing exact dedup +(`dedup_hash` / `dedup_sequence`) and the `kind` column intact and backward compatible, +generalizing the current keyword-flipping into a transparent, tunable matching engine. + +Core principles that run through every section: + +- **Two layers, kept separate.** Layer 1 is intra-account exact dedup (already shipped, + unchanged). Layer 2 is the new cross-account event grouping. +- **Group, never delete.** Rows are preserved; an event points at them and picks one + canonical row for reporting. Everything is reversible. +- **Count each expense exactly once.** Reporting reads a single "spendable" projection + so transfers, CC bill payments, and the bill-vs-purchases double count are excluded + once and only once. +- **Never silently hide real money.** Auto-merge requires a hard anchor (opposite-sign, + equal-amount, cross-account). Fuzzy description similarity can refine or suggest but + can never, on its own, flip a real expense to a transfer. When uncertain, suggest for + review instead of auto-hiding. + +This document was produced by a multi-agent research + design + adversarial-review +workflow. Each section below was independently drafted and then adversarially verified +against the real codebase. + +## Table of Contents + +0. [Unified Data Model (Authoritative)](#unified-data-model-authoritative) (read first; governs all naming) +1. [Recommended Domain Model](#recommended-domain-model) +2. [Deduplication Architecture](#deduplication-architecture) +3. [Matching Algorithms and Confidence Scoring](#matching-algorithms-and-confidence-scoring) +4. [Database Schema Suggestions (SQLite)](#database-schema-suggestions-sqlite) +5. [Edge Cases and Failure Scenarios](#edge-cases-and-failure-scenarios) +6. [UX Recommendations for Merged Transactions](#ux-recommendations-for-merged-transactions) +7. [Scalable Implementation Strategy](#scalable-implementation-strategy) +8. [Appendix: Industry Practices](#appendix-industry-practices) + + +--- + +## Unified Data Model (Authoritative) + +This section is the single source of truth for the event-grouping data model. Wherever the detailed sections below (Domain Model, Database Schema, Deduplication Architecture, Matching Algorithms) use a different table name, column name, role, enum value, or structure, the names and structures defined here win and those sections should be read with the substitutions in the Naming normalization map at the end. The design layers a first-class "financial event" object over the existing `transactions` table without touching the `kind` column, the exact `dedup_hash` / `dedup_sequence` dedup, or any reporting query; on upgrade day it produces byte-for-byte identical totals. + +### Decisions + +| Concern | Decision | Rationale | +|---|---|---| +| 1. Membership table name | `event_members` | It is the spelling used by two of four sections (Architecture, Matching) and reads naturally as "members of an event"; `event_memberships`, `financial_event_members`, and `event_transactions` are rejected as longer or less used variants. | +| 2. Membership table vs columns on `transactions` | A `event_members` TABLE plus a denormalized `transactions.event_id` pointer for the single "primary" event | A single `transactions.event_id` column cannot represent the N:1 case where one card purchase is both its own expense and a linked member of a bill, so a table is required; the denormalized pointer is kept only as a fast filter for the hot analytics path and the candidate query. | +| 3. Canonical column name | `canonical_transaction_id` | The user requirement language says "Canonical transaction chosen for reporting"; the events DDL and the `spendable_transactions` view both use this exact name, so the view's correlated subquery resolves. | +| 4. Role vocabulary | Simple set: `debit`, `credit`, `bill_payment`, `purchase`, `fee`, `reversal` | A short, unambiguous set keyed to the leg's function; the rich set is mapped onto it (`source_debit`->`debit`, `dest_credit`->`credit`, `underlying_purchase`->`purchase`, `fee_leg`->`fee`, `reversal_of`->`reversal`, `canonical` is dropped as a role because canonicity lives on the event, not the member). | +| 5. event_type enum | `internal_transfer`, `credit_card_payment` (1:1 bank-debit <-> card-credit), `credit_card_statement` (N:1 one bill <-> N purchases), `atm_withdrawal`, `loan_repayment`, `investment_transfer`, `refund_reversal`, `fee`, `duplicate` | The 1:1 leg-pair and the N:1 bill-to-purchases link are genuinely different cardinalities with different projection rules, so they are two distinct types (`credit_card_payment` vs `credit_card_statement`), not one; `cc_statement` / `cc_bill_payment` are folded into these two clear names. | +| 6. `match_rules` name collision | Split into `match_settings` (per-event-type thresholds: epsilon, day_window, min_score, auto_score, require_keyword) and `match_rules` (user pattern overrides: description_pattern, set_kind, set_event_type, hide, priority) | They are two unrelated concerns (tuning the matcher vs forcing a classification), so they get two tables; "thresholds" become `match_settings` and the user override engine keeps the `match_rules` name. | +| 7. FK from event tables to `transactions(id)` | Tables freshly created in this migration (`financial_events.canonical_transaction_id`, `event_members.transaction_id`) DO declare inline FKs with `ON DELETE` actions; the column added to the existing `transactions` table (`event_id`) does NOT, because SQLite rejects `ALTER TABLE ADD COLUMN ... REFERENCES`. | Inline FKs are legal and desirable on new tables and pass `foreign_key_check`; the one column on the pre-existing table cannot carry an inline FK without a full table recreate, so its integrity is enforced in app code. | +| 8. Provenance fields | Keep `prior_kind` on `event_members` (lossless undo) and `confidence` + `reasons` (JSON) on `financial_events`; unify `source` to `heuristic`, `rule`, `user`, `ai` (the `auto` value maps to `heuristic`) | The richer source set distinguishes which automatic mechanism fired (matcher heuristic vs a saved `match_rules` override vs an LLM suggestion), which the review UI needs; `auto` collapses to `heuristic`. | + +### Canonical SQLite DDL + +The migration sorts strictly after `021_reclassify_credit_card_transfers.sql` (migrations are applied in lexicographic filename order, and `020_*` / `021_*` duplicate prefixes already exist). Name it `022_financial_events.sql`. Row-pairing backfill stays in idempotent TypeScript so the runner's closing `foreign_key_check` cannot trip on a dangling reference; only DDL and workspace-scoped seeds live in the SQL. + +```sql +-- 022_financial_events.sql +-- First-class, auditable, reversible event-grouping layer over transactions. +-- Additive only: no table dropped, no column removed, no type changed. The kind +-- column and the dedup_hash / dedup_sequence dedup are left intact and stay the +-- analytics source of truth. Pair-reconstruction runs in TypeScript +-- (src/server/db/backfill/022_events.ts), not here, so the runner's closing +-- foreign_key_check never sees a half-written reference. + +-- 1. The event: one real-world money movement composed of N transaction rows. +CREATE TABLE financial_events ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + + event_type TEXT NOT NULL CHECK (event_type IN ( + 'internal_transfer', -- account A debit <-> account B credit (1:1) + 'credit_card_payment', -- bank debit <-> card-side credit (1:1 legs) + 'credit_card_statement', -- one bank bill <-> N card purchases (N:1) + 'atm_withdrawal', -- cash leaving a tracked account (single leg) + 'loan_repayment', -- checking -> loan account + 'investment_transfer', -- checking -> brokerage account + 'refund_reversal', -- a credit reversing an earlier purchase + 'fee', -- a standalone or attached fee row + 'duplicate')), -- pending<->posted or re-pull artifact, linked not collapsed + + -- The one member projected by per-event reporting, or NULL when the event + -- contributes nothing to spend. NULL for internal_transfer, credit_card_payment, + -- credit_card_statement, refund_reversal, and duplicate (none are spend). The + -- name is canonical_transaction_id everywhere, including the spendable view. + canonical_transaction_id INTEGER REFERENCES transactions(id) ON DELETE SET NULL, + + status TEXT NOT NULL DEFAULT 'suggested' + CHECK (status IN ('suggested','confirmed','rejected')), + + -- Provenance: heuristic = the matcher, rule = a match_rules override fired, + -- user = a manual action, ai = an LLM suggestion. 'auto' from the schema + -- section maps to 'heuristic'. + source TEXT NOT NULL DEFAULT 'heuristic' + CHECK (source IN ('heuristic','rule','user','ai')), + + -- Fellegi-Sunter P(match) in [0,1]. A score, not money, so REAL is correct. + confidence REAL NOT NULL DEFAULT 1.0 + CHECK (confidence >= 0 AND confidence <= 1), + + -- JSON array of human-readable reasons plus the comparison vector. JSON1 is + -- compiled into better-sqlite3's bundled SQLite, so json_valid works. + reasons TEXT CHECK (reasons IS NULL OR json_valid(reasons)), + + -- Idempotency key, the events-layer analogue of UNIQUE(dedup_hash,dedup_sequence). + -- Derived from the sorted member (dedup_hash:dedup_sequence) pairs, NOT from + -- row ids (ids are not stable across a re-pull, dedup_hash is). A re-sync over + -- an overlapping window re-derives the same key and upserts in place. + event_key TEXT NOT NULL, + + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + + UNIQUE (workspace_id, event_key) +); + +CREATE INDEX idx_events_workspace_status ON financial_events(workspace_id, status); +CREATE INDEX idx_events_workspace_type ON financial_events(workspace_id, event_type); +CREATE INDEX idx_events_canonical ON financial_events(canonical_transaction_id); +-- Review queue: open, auto-detected, low confidence first. +CREATE INDEX idx_events_review + ON financial_events(workspace_id, status, confidence) + WHERE status = 'suggested' AND source = 'heuristic'; + +-- 2. Membership: a transaction <-> event link carrying a role. This is a TABLE, +-- not a column, because one card purchase can be its own expense AND a member +-- of a credit_card_statement event at the same time. Inline FKs are legal +-- here because the table is freshly created. +CREATE TABLE event_members ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + event_id INTEGER NOT NULL REFERENCES financial_events(id) ON DELETE CASCADE, + transaction_id INTEGER NOT NULL REFERENCES transactions(id) ON DELETE CASCADE, + + role TEXT NOT NULL CHECK (role IN ( + 'debit', -- money leaving an account (was source_debit / outflow / transfer_out) + 'credit', -- money arriving in an account (was dest_credit / inflow / transfer_in) + 'bill_payment', -- the single bank debit that settles a card statement + 'purchase', -- an individual card purchase the statement settles (was underlying_purchase / bill_item) + 'fee', -- an associated fee leg (FX fee, ATM fee) (was fee_leg) + 'reversal')), -- a credit reversing an earlier purchase (was reversal_of) + + -- The kind this leg held before the event excluded it, for lossless undo. + -- NULL means the event did NOT change this leg (a 'purchase' leg in a + -- credit_card_statement keeps kind='expense', so its prior_kind is NULL). + prior_kind TEXT CHECK (prior_kind IS NULL OR prior_kind IN ('expense','income','transfer')), + + -- Per-leg score, distinct from the event-level confidence, so one weak leg can + -- be flagged while the event as a whole is confirmed. + match_confidence REAL + CHECK (match_confidence IS NULL OR (match_confidence >= 0 AND match_confidence <= 1)), + + created_at TEXT NOT NULL DEFAULT (datetime('now')), + + UNIQUE (workspace_id, transaction_id, event_id) +); + +CREATE INDEX idx_members_event ON event_members(event_id); +CREATE INDEX idx_members_txn ON event_members(workspace_id, transaction_id); + +-- A transaction may be at most ONE *grouping* leg (the leg that determines its +-- kind). 'purchase' is the deliberate exception so a card purchase can be linked +-- to its statement WITHOUT losing its own kind='expense' projection. This is the +-- partial unique index that resolves contradiction 2. +CREATE UNIQUE INDEX idx_members_one_grouping_leg + ON event_members(workspace_id, transaction_id) + WHERE role != 'purchase'; + +-- 3. Denormalized primary-event pointer on transactions. No inline REFERENCES: +-- SQLite rejects FK on ADD COLUMN. Integrity is enforced in app code, and the +-- unmerge path nulls event_id on members before deleting an event. It mirrors +-- the GROUPING membership only (never a 'purchase' link), so a purchase's +-- event_id stays NULL and it remains spendable. +ALTER TABLE transactions ADD COLUMN event_id INTEGER; + +ALTER TABLE transactions ADD COLUMN event_role TEXT + CHECK (event_role IS NULL OR event_role IN + ('debit','credit','bill_payment','purchase','fee','reversal')); + +ALTER TABLE transactions ADD COLUMN match_confidence REAL + CHECK (match_confidence IS NULL OR (match_confidence >= 0 AND match_confidence <= 1)); + +-- Partial index keeps the analytics hot path untouched: on a fresh install almost +-- no rows have an event. +CREATE INDEX idx_transactions_event + ON transactions(event_id) + WHERE event_id IS NOT NULL; + +-- 4. BLOCKING SUPPORT for candidate generation. SQLite cannot index a bare +-- expression like ABS(charged_amount) referenced from a join unless that exact +-- expression is the index key, so we materialize a generated column and index +-- THAT. STORED so the blocking index is a plain b-tree the planner uses for an +-- equality probe; abs/round are deterministic, which STORED generated columns +-- require. charged_currency is included so a 100 USD debit never blocks with a +-- 100 ILS credit. +ALTER TABLE transactions + ADD COLUMN charged_abs_amount REAL + GENERATED ALWAYS AS (ROUND(ABS(charged_amount), 2)) STORED; + +CREATE INDEX idx_transactions_block + ON transactions(workspace_id, charged_currency, charged_abs_amount, date); + +-- 5. Per-event-type matcher thresholds (tunable without a code edit). +CREATE TABLE match_settings ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + event_type TEXT NOT NULL CHECK (event_type IN ( + 'internal_transfer','credit_card_payment','credit_card_statement', + 'atm_withdrawal','loan_repayment','investment_transfer', + 'refund_reversal','fee','duplicate')), + -- Max allowed |abs(a) - abs(b)|. Matches DEFAULT_EPSILON = 0.01. + epsilon REAL NOT NULL DEFAULT 0.01, + -- Max date gap in days. Matches DEFAULT_DAY_WINDOW = 2. + day_window INTEGER NOT NULL DEFAULT 2, + -- Suggest in [min_score, auto_score); auto-confirm at or above auto_score. + min_score REAL NOT NULL DEFAULT 0.80, + auto_score REAL NOT NULL DEFAULT 0.97, + -- Require a description keyword on at least one side. + require_keyword INTEGER NOT NULL DEFAULT 1, + enabled INTEGER NOT NULL DEFAULT 1, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + UNIQUE (workspace_id, event_type) +); + +-- Seed defaults per EXISTING workspace so behavior is unchanged on upgrade. +-- New workspaces created after this migration must read with a hard-coded +-- fallback so detection is never silently disabled. +INSERT INTO match_settings (workspace_id, event_type, epsilon, day_window, min_score, auto_score, require_keyword) +SELECT w.id, et.event_type, et.epsilon, et.day_window, et.min_score, et.auto_score, et.require_keyword +FROM workspaces w +CROSS JOIN ( + SELECT 'internal_transfer' AS event_type, 0.01 AS epsilon, 2 AS day_window, 0.80 AS min_score, 0.97 AS auto_score, 1 AS require_keyword + UNION ALL SELECT 'credit_card_payment', 0.01, 5, 0.80, 0.97, 0 + UNION ALL SELECT 'credit_card_statement', 1.00, 38, 0.80, 0.97, 0 + UNION ALL SELECT 'atm_withdrawal', 0.01, 2, 0.80, 0.97, 1 + UNION ALL SELECT 'loan_repayment', 0.01, 5, 0.80, 0.97, 1 + UNION ALL SELECT 'investment_transfer', 0.01, 5, 0.80, 0.97, 1 + UNION ALL SELECT 'refund_reversal', 0.01, 90, 0.80, 0.97, 0 + UNION ALL SELECT 'fee', 0.01, 2, 0.80, 0.97, 0 + UNION ALL SELECT 'duplicate', 0.00, 10, 0.80, 0.97, 0 +) et; + +-- 6. User pattern overrides (sticky corrections). Matched against the RAW +-- description (the stable bank original), not a cleaned merchant name. +CREATE TABLE match_rules ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + description_pattern TEXT, -- substring or /regex/ against raw description + provider TEXT, + amount_min REAL, -- on ABS(charged_amount) + amount_max REAL, -- on ABS(charged_amount) + set_kind TEXT CHECK (set_kind IS NULL OR set_kind IN ('expense','income','transfer')), + set_event_type TEXT CHECK (set_event_type IS NULL OR set_event_type IN ( + 'internal_transfer','credit_card_payment','credit_card_statement', + 'atm_withdrawal','loan_repayment','investment_transfer','fee')), + set_category_id INTEGER REFERENCES categories(id) ON DELETE SET NULL, + hide INTEGER NOT NULL DEFAULT 0, -- maps to transactions.is_excluded (migration 020) + priority INTEGER NOT NULL DEFAULT 100, -- lower runs first + enabled INTEGER NOT NULL DEFAULT 1, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')) +); + +CREATE INDEX idx_rules_workspace ON match_rules(workspace_id, enabled, priority); + +-- 7. The canonical "what counts as spend exactly once" definition. A row is +-- spendable when it is a completed expense that is EITHER unattached to any +-- grouping event OR is the canonical row of its event. Because the day-one +-- backfill leaves every event with canonical_transaction_id = NULL, the +-- t.id = (NULL) branch is never true, so every currently-excluded leg stays +-- excluded and every currently-counted expense (event_id IS NULL) stays +-- counted: zero reporting drift on upgrade. Uses the canonical column name. +CREATE VIEW spendable_transactions AS +SELECT t.* +FROM transactions t +WHERE t.status = 'completed' + AND t.kind = 'expense' + AND ( + t.event_id IS NULL + OR t.id = (SELECT fe.canonical_transaction_id + FROM financial_events fe + WHERE fe.id = t.event_id) + ); +``` + +The day-window slack still must be applied in application code (`julianday` arithmetic), and the blocking probe must scan the row's own amount bucket plus its neighbors before applying the precise `ABS(ABS(a) - ABS(b)) <= epsilon` test, because rounding is a hard bucket boundary. The generated `charged_abs_amount` column makes each bucket probe an O(log n) point lookup rather than a scan, which is what keeps detection off the O(n^2) path. + +### Canonical TypeScript types and enums + +```typescript +// src/lib/types.ts (additive) + +export type EventType = + | "internal_transfer" + | "credit_card_payment" // 1:1 bank debit <-> card-side credit + | "credit_card_statement" // N:1 one bill <-> N purchases + | "atm_withdrawal" + | "loan_repayment" + | "investment_transfer" + | "refund_reversal" + | "fee" + | "duplicate"; + +export type EventRole = + | "debit" // money leaving an account + | "credit" // money arriving in an account + | "bill_payment" // the single bank debit that settles a card statement + | "purchase" // an individual card purchase a statement settles (additive, keeps kind='expense') + | "fee" // an associated fee leg + | "reversal"; // a credit reversing an earlier purchase + +export type EventStatus = "suggested" | "confirmed" | "rejected"; + +export type EventSource = "heuristic" | "rule" | "user" | "ai"; + +export type TransactionKind = "expense" | "income" | "transfer"; + +export interface FinancialEvent { + id: number; + workspaceId: number; + eventType: EventType; + canonicalTransactionId: number | null; + status: EventStatus; + source: EventSource; + confidence: number; + reasons: string[] | null; + eventKey: string; + createdAt: string; + updatedAt: string; +} + +export interface EventMember { + id: number; + workspaceId: number; + eventId: number; + transactionId: number; + role: EventRole; + priorKind: TransactionKind | null; + matchConfidence: number | null; + createdAt: string; +} + +export interface MatchSettings { + id: number; + workspaceId: number; + eventType: EventType; + epsilon: number; + dayWindow: number; + minScore: number; + autoScore: number; + requireKeyword: boolean; + enabled: boolean; + createdAt: string; + updatedAt: string; +} + +export interface MatchRule { + id: number; + workspaceId: number; + descriptionPattern: string | null; + provider: string | null; + amountMin: number | null; + amountMax: number | null; + setKind: TransactionKind | null; + setEventType: Exclude | null; + setCategoryId: number | null; + hide: boolean; + priority: number; + enabled: boolean; + createdAt: string; + updatedAt: string; +} +``` + +### Naming normalization map + +| Variant used in a section below | Canonical name | +|---|---| +| `event_memberships`, `financial_event_members`, `event_transactions` (table) | `event_members` | +| `representative_txn_id`, `canonical_txn_id` (column) | `canonical_transaction_id` | +| `match_reasons`, `reasons_json` (column) | `reasons` | +| `state` (on the event) | `status` | +| `match_rules` used for thresholds (epsilon/day_window/min_score/auto_score/require_keyword) | `match_settings` | +| `match_rules` used for user pattern overrides | `match_rules` (unchanged) | +| role `source_debit`, `outflow`, `transfer_out` | role `debit` | +| role `dest_credit`, `inflow`, `transfer_in` | role `credit` | +| role `underlying_purchase`, `bill_item` | role `purchase` | +| role `fee_leg` | role `fee` | +| role `reversal_of` | role `reversal` | +| role `bill` (cc_statement member) | role `bill_payment` | +| role `canonical` (as a member role) | dropped; canonicity lives on `financial_events.canonical_transaction_id` | +| event_type `cc_statement`, `cc_bill_payment` (N:1) | `credit_card_statement` | +| event_type `credit_card_payment` (1:1 leg pair) | `credit_card_payment` (unchanged) | +| `source` value `auto` | `heuristic` | +| `transactions.match_amount_cents`, `norm_description` (matcher scratch columns) | not in this authoritative core; blocking uses the generated `charged_abs_amount` column | +| `match_candidates` (staging table) | retained as the sections define it; orthogonal to this core model | +| migration file `022_event_matching.sql` | `022_financial_events.sql` | + +--- + +## Recommended Domain Model + +This section introduces a first-class **Financial Event** layer that sits *above* the existing `transactions` table. It does not replace `kind`, the existing exact dedup (`dedup_hash` / `dedup_sequence`), or any reporting query. It generalizes the ad-hoc `kind`-flipping done today in `detectKind` (`src/server/lib/transfers.ts`) and `findInternalTransferPairs` (`src/server/lib/internal-transfers.ts`) into an auditable, reversible grouping with persisted scores and reasons. + +The guiding principle, taken from how Monarch, Copilot, YNAB and Simplifi all ship: **count each real-world spend exactly once, at purchase time, and never on the settling money-movement.** Where those products achieve this purely by tagging each leg with an excluded category, Spent keeps that per-leg signal (the `kind` column) *and* adds the explicit grouping object the aggregators lack, so the relationship between a credit-card bill and its underlying purchases is visible, auditable, and undoable. + +### 0. Constraints this design must respect (verified against the codebase) + +These are load-bearing facts that shaped every decision below. + +- **Migrations are ordered lexicographically by full filename and tracked by filename**, not by an integer counter. `migrate.ts` does `readdirSync(...).filter(.sql).sort()` and records each applied file in `_migrations(name)`. The tree already contains *duplicate numeric prefixes* (`020_excluded.sql` + `020_multiple_bank_credentials.sql`, `021_chat_sessions.sql` + `021_reclassify_credit_card_transfers.sql`). The current lexicographic tail is `021_reclassify_credit_card_transfers.sql`. The new file must sort strictly after it; use a descriptive, unambiguous name: **`022_financial_events.sql`**. Do not rely on "the counter is 021"; rely on the sort. +- **`detectKind` is NOT a general bank classifier.** `isBankProvider` only returns true for `hapoalim` and `leumi` (`BANK_PROVIDERS_SET`). Every other provider (`mizrahi`, `discount`, `oneZero`, all card providers) falls through to the `expense` branch at insert time. So today's insert-time CC-payment and income detection only fires for two banks. The event layer must therefore treat `detectKind` as a *partial, two-bank* first pass and do the heavier lifting in the post-insert grouping pass, which is provider-agnostic. +- **`is_excluded` already exists** on `transactions` (migration `020_excluded.sql`), as does the `excluded_merchants` table, and `mapTransactionRow` already reads `is_excluded`. The new `match_rules.hide` action maps onto that column. No new exclusion column is introduced. +- **`bank_credentials` already carries `label` (plaintext, <=128 chars) and `requires_manual_two_factor`** (migrations `020_multiple_bank_credentials.sql`, `019_two_factor.sql`). The `accounts.label` falls back to `bank_credentials.label`. +- **Pairing today is O(debits x credits) inside the sync window** (`findInternalTransferPairs` nested loop). Generalizing it without bucketing would reproduce that quadratic. The scorer below buckets candidates by `(currency, ROUND(ABS(charged_amount),2))` first, so comparison is near-linear in practice. +- **`ALTER TABLE ... ADD COLUMN REFERENCES ...` is rejected by SQLite.** That is exactly why migration 013 recreated tables. So `transactions.account_id` cannot be added with an inline `REFERENCES` clause; it is added as a plain `INTEGER` column and the relationship is enforced in application code (better-sqlite3 has `foreign_keys` ON, and an inline FK on ADD COLUMN would throw). The `accounts` table itself can declare its FKs normally because it is freshly created. + +### Entities at a glance + +``` +Account (NEW) one row per real-world account behind transactions +Transaction (EXISTING) unchanged columns; one new nullable account_id (no inline FK) +FinancialEvent (NEW) a logical money-movement grouping N transactions +EventMembership (NEW) transaction <-> event link, carries a role +MatchCandidate (NEW) a proposed (not yet confirmed) grouping + score + reasons +MatchRule (NEW) a persisted user/heuristic rule that classifies or groups +``` + +A row's `kind` answers *"how does this single row score in spend/income analytics?"*; a `FinancialEvent` answers *"what real-world thing happened, and which rows are its legs?"*. A transfer / CC-payment / ATM move becomes an event whose canonical projection is excluded from spend. A plain expense or income row needs no event at all (see roles below). + +### 1. Account + +Today an account is implicit: the tuple `(workspace_id, provider, account_number)`, optionally narrowed by `credential_id`. We materialize it so events can reference stable account identities, and so an `AccountType` can drive type-specific event rules (a debit on a `credit_card` account behaves differently from one on `checking`). Stable account identity also defends against credential re-link churn (the largest documented duplicate source in Copilot/Quicken). + +```sql +-- 022_financial_events.sql (additive) +PRAGMA foreign_keys = OFF; -- migrate.ts already toggles this; left here for clarity + +CREATE TABLE accounts ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + credential_id INTEGER REFERENCES bank_credentials(id) ON DELETE SET NULL, + provider TEXT NOT NULL, + account_number TEXT NOT NULL, + -- checking|credit_card|loan|brokerage|wallet|cash|unknown + type TEXT NOT NULL DEFAULT 'unknown' + CHECK(type IN ('checking','credit_card','loan','brokerage','wallet','cash','unknown')), + label TEXT, -- display name; UI falls back to bank_credentials.label + currency TEXT, -- dominant charged_currency, informational only + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + -- stable identity within a workspace; keyed on (provider, account_number), + -- NOT the rotating credential id, so it survives credential re-link. + UNIQUE(workspace_id, provider, account_number) +); +CREATE INDEX idx_accounts_workspace ON accounts(workspace_id); + +-- No inline REFERENCES: SQLite rejects FK on ADD COLUMN. The link is enforced +-- in application code; account_id is nullable so the migration cannot fail and +-- legacy rows degrade gracefully. +ALTER TABLE transactions ADD COLUMN account_id INTEGER; +CREATE INDEX idx_transactions_account ON transactions(account_id); + +-- Backfill one accounts row per distinct (workspace_id, provider, account_number), +-- seeding type from the provider's BankKind in BANK_PROVIDERS (card -> credit_card, +-- bank -> checking). oneZero and per-account overrides are applied by the app later. +INSERT INTO accounts (workspace_id, provider, account_number, credential_id, type) +SELECT t.workspace_id, t.provider, t.account_number, + (SELECT bc.id FROM bank_credentials bc + WHERE bc.workspace_id = t.workspace_id AND bc.provider = t.provider + LIMIT 1), + 'unknown' -- refined to checking/credit_card by a one-time app backfill from BANK_PROVIDERS +FROM (SELECT DISTINCT workspace_id, provider, account_number FROM transactions) t; + +UPDATE transactions +SET account_id = ( + SELECT a.id FROM accounts a + WHERE a.workspace_id = transactions.workspace_id + AND a.provider = transactions.provider + AND a.account_number = transactions.account_number +); +``` + +The `type` seed is left `'unknown'` in pure SQL because `BANK_PROVIDERS` lives in TypeScript (`src/lib/types.ts`), not SQL; a one-time app-level backfill maps each provider's `BankKind` to `checking`/`credit_card`. Pairing falls back to the existing `credential_id`/`account_number` logic in `isDifferentAccount` whenever `account_id` is null, so the absence of accounts never breaks matching. + +```typescript +export type AccountType = + | "checking" | "credit_card" | "loan" + | "brokerage" | "wallet" | "cash" | "unknown"; + +export interface Account { + id: number; + workspaceId: number; + credentialId: number | null; + provider: string; + accountNumber: string; + type: AccountType; + label: string | null; + currency: string | null; + createdAt: string; + updatedAt: string; +} +``` + +### 2. FinancialEvent, EventMembership, and roles + +```sql +CREATE TABLE financial_events ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + type TEXT NOT NULL CHECK(type IN ( + 'internal_transfer','credit_card_payment','atm_withdrawal', + 'loan_repayment','investment_transfer','refund_reversal','fee')), + -- member transaction used for any per-event projection (a "transfers" report, + -- a fee row). NULL for money-movement events that must never be summed. + -- No inline FK to transactions on this column is needed because the table is + -- new; it is declared inline and valid. + canonical_transaction_id INTEGER REFERENCES transactions(id) ON DELETE SET NULL, + -- Fellegi-Sunter P(match) in [0,1]. + confidence REAL NOT NULL DEFAULT 1.0 CHECK(confidence >= 0 AND confidence <= 1), + -- 'heuristic' (transfers.ts logic), 'rule' (MatchRule), 'user' (manual), + -- 'ai' (LLM-suggested). Mirrors transactions.category_source semantics. + source TEXT NOT NULL CHECK(source IN ('heuristic','rule','user','ai')), + -- 'confirmed' once auto-applied/accepted; 'suggested' awaiting review; + -- 'rejected' tombstone kept for audit + to suppress re-suggestion. + state TEXT NOT NULL DEFAULT 'confirmed' + CHECK(state IN ('suggested','confirmed','rejected')), + reasons TEXT, -- JSON array of human-readable reasons + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')) +); +CREATE INDEX idx_events_workspace ON financial_events(workspace_id); +CREATE INDEX idx_events_type ON financial_events(workspace_id, type); +CREATE INDEX idx_events_canonical ON financial_events(canonical_transaction_id); + +CREATE TABLE event_memberships ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + event_id INTEGER NOT NULL REFERENCES financial_events(id) ON DELETE CASCADE, + transaction_id INTEGER NOT NULL REFERENCES transactions(id) ON DELETE CASCADE, + role TEXT NOT NULL CHECK(role IN ( + 'source_debit','dest_credit','bill_payment', + 'underlying_purchase','fee_leg','reversal_of','canonical')), + created_at TEXT NOT NULL DEFAULT (datetime('now')), + -- A transaction belongs to at most ONE GROUPING event (transfer/cc-payment/ + -- loan/investment/refund). See the membership-uniqueness note below for why + -- 'underlying_purchase' is the deliberate exception, handled by a partial + -- unique index rather than a column-level UNIQUE. + UNIQUE(workspace_id, transaction_id, event_id) +); +CREATE INDEX idx_memberships_event ON event_memberships(event_id); +CREATE INDEX idx_memberships_txn ON event_memberships(workspace_id, transaction_id); + +-- A transaction may participate as at most ONE *grouping* leg (the legs that +-- determine its kind). 'underlying_purchase' is excluded so a card purchase can +-- be audited as part of a credit_card_payment event WITHOUT losing its own +-- kind='expense' projection. This is the fix for the over-clustering concern and +-- for the "a purchase is both its own expense and a member of the bill" tension. +CREATE UNIQUE INDEX idx_memberships_one_grouping_leg + ON event_memberships(workspace_id, transaction_id) + WHERE role != 'underlying_purchase'; +``` + +Why the membership rule changed from a blanket `UNIQUE(workspace_id, transaction_id)`: the draft's blanket constraint is self-contradictory for credit-card payments. A card purchase must (a) keep `kind='expense'` and count in spend, and (b) be attachable to the `credit_card_payment` event for auditing. Under a blanket unique constraint it could be a member of *either* its own event *or* the bill, never linked to the bill without losing its expense projection. The partial unique index resolves this: exactly one *grouping* leg per transaction (so transfers cannot chain or double-assign), but `underlying_purchase` links are additive annotations that never touch the purchase's `kind`. There is no separate `actual_expense`/`actual_income` event type at all; a plain expense is just `kind='expense'` with zero memberships. This removes a whole class of "degenerate event" rows and the over-clustering risk the entity-resolution literature warns about. + +```typescript +export type EventType = + | "internal_transfer" | "credit_card_payment" | "atm_withdrawal" + | "loan_repayment" | "investment_transfer" | "refund_reversal" | "fee"; + +export type EventRole = + | "source_debit" // money leaving an account (e.g. checking -X) + | "dest_credit" // money arriving in an account (e.g. card +X) + | "bill_payment" // the single bank debit that pays a CC statement + | "underlying_purchase" // an individual card purchase the bill settles (additive) + | "fee_leg" // an associated fee row (FX fee, ATM fee) + | "reversal_of" // a credit that reverses an earlier debit + | "canonical"; // the member a per-event report projects + +export type EventSource = "heuristic" | "rule" | "user" | "ai"; +export type EventState = "suggested" | "confirmed" | "rejected"; + +export interface FinancialEvent { + id: number; + workspaceId: number; + type: EventType; + canonicalTransactionId: number | null; + confidence: number; + source: EventSource; + state: EventState; + reasons: string[] | null; + createdAt: string; + updatedAt: string; +} + +export interface EventMembership { + id: number; + workspaceId: number; + eventId: number; + transactionId: number; + role: EventRole; + createdAt: string; +} +``` + +### 3. Event taxonomy + +"Canonical" is the row a *per-event* view projects (e.g. a "Transfers" or "CC Payments" report). For spend/income, money-movement events have no projection at all: every grouping leg is set to `kind='transfer'` and disappears from spend AND income via the existing `kind` filters. This is the schema-level guarantee a transfer can never be summed. + +| Event type | Cardinality | Members and roles | Canonical | Spend/income effect | +|---|---|---|---|---| +| **internal_transfer** | 2 legs (3 with a fee) | `source_debit` (acct A) + `dest_credit` (acct B); optional `fee_leg` | the debit, for a transfers view | both legs `kind='transfer'`, excluded from spend AND income; a `fee_leg` stays `kind='expense'` | +| **credit_card_payment** | 1 + 0..1 + 0..N | `bill_payment` (bank debit) + optional `dest_credit` (card-side credit, if the card account is tracked) + 0..N `underlying_purchase` (card line items, additive) | the `bill_payment`, for a payments view | bill + card-side credit `kind='transfer'` (excluded); underlying purchases keep `kind='expense'` and count once | +| **atm_withdrawal** | 1 (2 with fee) | `source_debit` (+ optional `fee_leg`) | the withdrawal | per `treatAtmAsTransfers`: filed under "Cash & ATM" as spend, or flipped `kind='transfer'` and excluded | +| **loan_repayment** | 2 | `source_debit` (checking) + `dest_credit` (loan acct, if tracked) | the debit | both legs `kind='transfer'`; an interest portion, if separable, may be a `fee_leg` expense | +| **investment_transfer** | 2 | `source_debit` (checking) + `dest_credit` (brokerage) | the debit | both `kind='transfer'`, excluded | +| **refund_reversal** | 2 (or 1 standalone) | `reversal_of` (the credit) + the original `underlying_purchase` it cancels | none | see "Refund handling" below; nets the original expense toward zero | +| **fee** | 1 (or `fee_leg` of a parent) | the fee row as `canonical` or `fee_leg` | the fee row | counts as expense, never excluded | + +Cardinality notes that drive the schema: + +- **The two-leg events** generalize today's `findInternalTransferPairs`. Today the paired rows are merely flipped to `kind='transfer'`; now they become an event with `source_debit`/`dest_credit` roles, a stored `confidence`, and `reasons`. The cross-account, opposite-sign, equal-`ABS(charged_amount)`, date-window logic in `internal-transfers.ts` becomes the candidate generator, with bucketing added (see scorer). +- **`credit_card_payment` is the one true one-to-many link** and is the gap the brief calls out. The bank `bill_payment` debit equals the *sum* of many card `underlying_purchase` rows, so it can never be paired 1:1 by amount. Two caveats the implementer must respect: (1) the sum is over `charged_amount` (the ILS-charged figure), not `original_amount`; purchases in foreign `original_currency` still settle in ILS, so summing `charged_amount` is correct, but FX-fee rows and rounding mean the bill rarely equals the sum to the agora. Attachment is therefore by *statement window + account linkage*, never by an exact-sum check. (2) Attachment is purely additive auditing: even if zero purchases are linked, flipping the single `bill_payment` leg to `kind='transfer'` already prevents double counting exactly as the existing migration `021_reclassify_credit_card_transfers.sql` does today. + +**Canonical selection (deterministic):** + +```typescript +function chooseCanonical(type: EventType, members: EventMembership[]): number | null { + switch (type) { + case "atm_withdrawal": + case "fee": + return members.find((m) => m.role === "canonical")?.transactionId + ?? members[0]?.transactionId ?? null; + case "internal_transfer": + case "loan_repayment": + case "investment_transfer": + return members.find((m) => m.role === "source_debit")?.transactionId ?? null; + case "credit_card_payment": + return members.find((m) => m.role === "bill_payment")?.transactionId ?? null; + case "refund_reversal": + return null; + } +} +``` + +Canonical is only ever used by *per-event* views, never by spend/income aggregation. Spend/income reads `kind` and nothing else. + +### 4. How this maps onto and supersedes `kind` + +`kind` stays exactly as it is and stays the fast path. The relationship is a strict, derivable invariant: + +``` +event type => kind set on member rows +-------------------- ------------------------------------------------- +internal_transfer => both legs kind='transfer' +credit_card_payment => bill_payment + dest_credit kind='transfer'; + underlying_purchase rows keep kind='expense' +atm_withdrawal => kind='transfer' (treatAtmAsTransfers) or 'expense' +loan_repayment => both legs kind='transfer' +investment_transfer => both legs kind='transfer' +refund_reversal => see "Refund handling" +fee => kind='expense' +(no event) => kind='expense' or 'income' (the common case) +``` + +`kind` is a **denormalized projection of grouping membership**: whenever a grouping event is created, confirmed, edited, split, or undone, each touched transaction's `kind` is re-synced through the *existing* `setTransactionKind` / `markTransfersByIds` in `queries/transactions.ts`. This is the backward-compatibility hinge: + +- Every reporting query (`getMonthlySummary`, `getCategoryBreakdown`, `getPeriodTotal`, `getTransactionsSummary`, `getTopMerchants`, `getCategorySpendInRange`, etc.) **keeps filtering on `kind` and works unchanged**. None of them JOIN `financial_events`. The event layer's job is to *set* `kind` correctly and explain *why*. +- `detectKind` remains the cheap insert-time first pass, but note it only classifies `hapoalim`/`leumi` today. The grouping pass is the provider-agnostic second pass that confirms, groups, scores, and corrects across all providers, and records the audit trail. + +**Refund handling (the one place that can silently hide or wrongly count money, so it is explicit):** a card refund arrives as a positive `charged_amount` on a card account. For card providers, `detectKind` does NOT mark it income (the income branch only fires for the two banks), so it defaults to `kind='expense'` with a positive amount, which reporting sums via `ABS(...)` as positive spend. That is a pre-existing latent bug the event layer must fix, not inherit. A `refund_reversal` event links the credit (`reversal_of`) to the original purchase. The chosen, auditable behavior: **leave both rows `kind='expense'`**, so the category's net spend in `getCategoryBreakdown`/`getCategorySpendInRange` is `SUM(ABS(charged_amount))` over a -100 and a +100 = 200, which is WRONG under the current `ABS` aggregation. Therefore the refund credit must be stored with `kind='income'` *or* the aggregation must net signed values. To avoid touching every aggregation query, the rule is: a matched refund credit is set `kind='income'` and linked; an *unmatched* positive card row stays `kind='expense'` only if it is genuinely income-like, otherwise it is flagged `needs_review`. This is called out because under the existing `ABS()`-based queries, any mishandled sign double-counts rather than hides, and double-counting a refund is the single most damaging silent error in this whole layer. + +### 5. MatchCandidate and MatchRule + +`MatchCandidate` is the staging area for *suggested* groupings (the Fellegi-Sunter clerical-review zone). High-confidence groupings skip straight to a `confirmed` `financial_events` row; medium-confidence ones land here for accept/reject. + +```sql +CREATE TABLE match_candidates ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + proposed_type TEXT NOT NULL CHECK(proposed_type IN ( + 'internal_transfer','credit_card_payment','atm_withdrawal', + 'loan_repayment','investment_transfer','refund_reversal')), + members TEXT NOT NULL, -- JSON: [{"transactionId":123,"role":"source_debit"},...] + confidence REAL NOT NULL CHECK(confidence >= 0 AND confidence <= 1), + score_detail TEXT, -- JSON: per-feature comparison vector + weights + reasons TEXT, -- JSON array, same shape as financial_events.reasons + -- guards stale candidates: a candidate referencing a since-deleted or + -- already-grouped transaction is dropped at apply time (membership unique + -- index enforces this), not surfaced. + state TEXT NOT NULL DEFAULT 'pending' + CHECK(state IN ('pending','accepted','rejected')), + created_at TEXT NOT NULL DEFAULT (datetime('now')), + resolved_at TEXT +); +CREATE INDEX idx_candidates_pending ON match_candidates(workspace_id, state); +``` + +`MatchRule` makes corrections sticky. The default text match is on the **raw `description`** (the bank original statement), which is stable, not a cleaned merchant name, matching Monarch's guidance and Spent's existing keyword regexes. + +```sql +CREATE TABLE match_rules ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + description_pattern TEXT, -- substring or /regex/ against raw description + provider TEXT, + account_id INTEGER REFERENCES accounts(id) ON DELETE CASCADE, + amount_min REAL, -- on ABS(charged_amount) + amount_max REAL, -- on ABS(charged_amount) + set_kind TEXT CHECK(set_kind IN ('expense','income','transfer')), + set_event_type TEXT CHECK(set_event_type IN ( + 'internal_transfer','credit_card_payment','atm_withdrawal', + 'loan_repayment','investment_transfer','fee')), + set_category_id INTEGER REFERENCES categories(id) ON DELETE SET NULL, + hide INTEGER NOT NULL DEFAULT 0, -- maps to transactions.is_excluded + priority INTEGER NOT NULL DEFAULT 100, -- lower runs first + enabled INTEGER NOT NULL DEFAULT 1, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')) +); +CREATE INDEX idx_rules_workspace ON match_rules(workspace_id, enabled, priority); +``` + +```typescript +export interface MatchCandidate { + id: number; + workspaceId: number; + proposedType: Exclude; + members: { transactionId: number; role: EventRole }[]; + confidence: number; + scoreDetail: Record | null; + reasons: string[] | null; + state: "pending" | "accepted" | "rejected"; + createdAt: string; + resolvedAt: string | null; +} + +export interface MatchRule { + id: number; + workspaceId: number; + descriptionPattern: string | null; + provider: string | null; + accountId: number | null; + amountMin: number | null; + amountMax: number | null; + setKind: "expense" | "income" | "transfer" | null; + setEventType: Exclude | null; + setCategoryId: number | null; + hide: boolean; + priority: number; + enabled: boolean; + createdAt: string; + updatedAt: string; +} +``` + +`hide` maps onto the **existing** `transactions.is_excluded` column and the `excluded_merchants` table (migration 020). "Is a transfer" (`kind` / grouping) is a *separate* control from "hide from reports" (`is_excluded`); balances/net worth (computed from raw `charged_amount`) ignore both, exactly as Monarch documents. Only spend/income analytics honor them. + +**Safety rule that prevents a rule from silently hiding a real expense:** a `set_kind='transfer'` rule with no `account_id` and no `amount_min` is rejected at save time. A blanket "everything matching this Hebrew substring is a transfer" rule is the single easiest way to vanish real spend. A transfer-setting rule must be scoped by at least one of `account_id` or an amount band, and the UI surfaces, before save, how many existing rows the rule would flip. + +### 6. Confidence and the three-zone auto/suggest/ignore gate + +Persisted `confidence` is the Fellegi-Sunter `P(match) = 2^M / (1 + 2^M)`, anchored to weight 4 ~ 0.95 and weight 7 ~ 0.99, and aligned to Plaid Enrich's `VERY_HIGH (>0.98) / HIGH (>0.90) / MEDIUM / LOW`. Thresholds are policy, stored in `workspace_settings` (the existing per-workspace key/value table from migration 013), not magic constants. + +```typescript +// read from workspace_settings; these are defaults +const AUTO_CONFIRM = 0.95; // write a confirmed financial_event (also syncs kind) +const SUGGEST_MIN = 0.70; // write a pending match_candidate for review +// below SUGGEST_MIN: discard + +// HARD GUARD: a two-leg grouping may auto-confirm ONLY if it has the exact +// structural anchor (opposite-sign, equal ABS(charged_amount) within epsilon, +// DIFFERENT account). Fuzzy description similarity can never, on its own, push a +// pair past AUTO_CONFIRM. This is the schema-level promise that a real expense +// is never silently flipped to transfer on weak signal alone. +function gate(c: Candidate): "confirm" | "suggest" | "drop" { + const p = c.confidence; + if (p >= AUTO_CONFIRM && c.hasExactAmountAnchor) return "confirm"; + if (p >= SUGGEST_MIN) return "suggest"; + return "drop"; +} +``` + +**The scorer (and the O(n^2) fix).** The candidate generator must not reproduce `findInternalTransferPairs`'s nested `debits x credits` loop over the whole window. First **bucket** the window's eligible rows by `(charged_currency, ROUND(ABS(charged_amount), 2))`. Opposite-sign exact-amount partners can only live in the same bucket, so comparison drops from O(n^2) to roughly O(n) plus small intra-bucket work. Only inside a bucket do we evaluate the date window and description signals. The same bucketing bounds the **backfill**: re-running pairing over tens of thousands of historical rows is quadratic if done naively, so the migration's app-side backfill iterates month-by-month windows, bucketed, and is idempotent (re-running it produces no new events because the membership unique index rejects duplicates). + +Features and `log2(m/u)` weights: + +- **opposite-sign, equal `ABS(charged_amount)` within `epsilon=0.01`, different `account_id`** (the existing `internal-transfers.ts` test, now using `account_id` with fallback to `credential_id`/`account_number`): the dominant positive signal, weight ~6. This is the `hasExactAmountAnchor` flag the gate requires. +- **date gap in days** (existing `dayWindow`, default 2; widened asymmetrically toward the future to absorb pending->posted settlement lag, ~1 week past / ~2 weeks future): full weight at gap 0, decaying to 0 at the window edge. +- **description matches `CREDIT_CARD_PAYMENT_PATTERNS` / `INTERNAL_TRANSFER_PATTERNS`** (the existing regex sets in `transfers.ts`): strong positive for the relevant type. +- **fuzzy description similarity** (new): normalize first for Hebrew/RTL: strip Unicode bidi control marks (U+200E/U+200F/U+202A-U+202E), strip niqqud (U+0591-U+05C7), normalize Hebrew final-letter forms to base forms, NFC-normalize, collapse whitespace; then Damerau-Levenshtein or token-set Jaccard. A **tiebreaker only**: it can lift a pair from `suggest` toward the top of the review queue, but per the hard guard it can never auto-confirm without the exact-amount anchor. + +Every confirmed event and candidate stores `reasons` (human-readable, e.g. `["opposite-sign equal amount 1,250.00","dates 1 day apart","desc matched CREDIT_CARD_PAYMENT_PATTERNS: ויזה"]`) and `match_candidates.score_detail` (per-feature vector + weights), so every auto-decision is explainable. + +**Reversal, split, and re-sync (idempotency).** No transaction row is ever deleted or mutated beyond `kind` / `category_id` / `is_excluded`; ledgers and balances stay intact. + +- **Undo** = set `financial_events.state='rejected'` (tombstone, suppresses re-suggestion), delete its `event_memberships`, and recompute each freed member's `kind` via `detectKind`. Because `detectKind` only classifies `hapoalim`/`leumi`, a freed bank-side CC-payment row on those two banks would be re-flipped to transfer on the next pass; to make undo *stick*, undo also writes a narrow `match_rule` (or a `rejected` tombstone keyed on the member set) so the grouping pass does not immediately re-propose the rejected event. Without this, undo is not idempotent across syncs. +- **Split** = delete one membership, recompute canonical, re-sync `kind` for affected legs. +- **Leg deletion** = `event_memberships` cascades on transaction delete; the orchestrator must then recompute the surviving event (re-choose canonical, and if a two-leg event loses a leg, dissolve it and restore the survivor's `kind`). The `canonical_transaction_id` `ON DELETE SET NULL` only nulls the pointer; it does NOT restore the survivor's `kind`, so the recompute step is mandatory. +- **Re-sync safety**: the grouping pass is rerun every sync over the window. Idempotency rests on the membership unique index (`idx_memberships_one_grouping_leg`): an already-grouped row cannot be re-added as a second grouping leg, so reruns are no-ops on stable data. Candidates referencing a now-grouped or deleted transaction are dropped at apply time rather than surfaced. + +--- + +Relevant grounding files for the implementing engineer: `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/lib/types.ts` (add the new interfaces/enums; `BANK_PROVIDERS` and `BankKind` drive `AccountType` seeding), `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/lib/transfers.ts` (note `isBankProvider` only covers `hapoalim`/`leumi`; `detectKind` is a two-bank first pass), `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/lib/internal-transfers.ts` (generalize into the bucketed candidate generator + scorer; today's loop is O(debits x credits)), `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/db/queries/transactions.ts` (`setTransactionKind`, `markTransfersByIds`, `getInternalTransferCandidates`, and the `kind`/`ABS(charged_amount)`-based reporting queries that stay unchanged; note the refund double-count risk under `ABS()` aggregation), and `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/db/migrations/` (migrations are applied in lexicographic filename order and tracked by filename in `_migrations`; duplicate `020_*`/`021_*` prefixes already exist; name the new file `022_financial_events.sql` so it sorts after `021_reclassify_credit_card_transfers.sql`). + +--- + +## Deduplication Architecture + +Spent already has a working intra-account exact deduper. This section keeps it untouched and wraps it in a second, additive layer that turns today's one-shot `kind`-flipping (`detectKind` at insert, `findInternalTransferPairs` after insert) into a persistent, auditable, reversible event-grouping pipeline. Both layers are deterministic, idempotent across re-syncs, and incremental over the sync window. + +Three facts from the current code shape everything below, so they are stated up front rather than glossed over: + +1. `isBankProvider` in `transfers.ts` currently recognizes **only** `hapoalim` and `leumi` as banks. `detectKind` therefore only ever flips bank rows from those two providers. Any design that says "for each bank debit" must NOT route through `isBankProvider`, or it silently ignores `mizrahi`, `discount`, `oneZero`, and every other bank. We instead drive bill detection off the credit-card keyword set plus "provider is not a card provider," and we expand `BANK_PROVIDERS_SET` as a prerequisite (tracked as a one-line change in `transfers.ts`). +2. The existing `getInternalTransferCandidates` pre-filters `kind != 'transfer'`. Once a row is flipped to transfer it becomes invisible to the next pairing pass. That is fine for the old fire-and-forget code but fatal for a reconcile-and-undo layer: a stale auto event could never be detected or rolled back. Layer 2 therefore reads candidates through a NEW query that does **not** pre-exclude transfers and instead excludes only rows already bound to a `confirmed`/`source='user'` event. +3. `is_excluded` (migration `020_excluded.sql`) exists on the row but is **not** referenced by any analytics `WHERE` clause in `transactions.ts`; every spend/income query filters on `kind` only. So the exclusion contract this layer must honor is `kind`, not `is_excluded`. We leave `is_excluded` alone and project through `kind`, exactly as `markTransfersByIds` does today. + +### Two layers, one pipeline + +``` + ONE SYNC RUN (per workspace, in syncWorkspace) + ┌──────────────────────────────────────────────────────────────────────────────┐ + │ │ + │ LAYER 1 INTRA-ACCOUNT EXACT DEDUP (UNCHANGED - src/server/lib/dedup.ts) │ + │ ┌─────────────┐ ┌─────────────────────────────────────────────────────┐ │ + │ │ scrapeBank │──▶│ insertTransactions(): computeDedupHash + │ │ + │ │ (per cred) │ │ dedup_sequence, ON CONFLICT pending->posted, idem. │ │ + │ └─────────────┘ └─────────────────────────────────────────────────────┘ │ + │ runs inside syncOneCredential(), once per credential │ + │ │ │ + │ ▼ (all credentials inserted) │ + │ LAYER 2 CROSS-ACCOUNT EVENT GROUPING (NEW - additive) │ + │ │ + │ (a) NORMALIZE backfill match-key columns once, then window-only │ + │ │ (norm_description, match_amount_cents) │ + │ ▼ │ + │ (b) BLOCK candidate generation: bucket by abs(amount_cents) │ + │ │ x date-window; cross-account only. Never O(n^2). │ + │ ▼ │ + │ (c) SCORE Fellegi-Sunter additive weights per candidate │ + │ │ (sign, account, date gap, amount drift, desc sim, │ + │ ▼ cc-payment / transfer keyword features) │ + │ (d) DECIDE 3 zones: auto-link | suggest (needs_review) | drop │ + │ │ + N:1 aggregate pass for CC bill <-> purchases │ + │ ▼ │ + │ (e) RECONCILE upsert financial_events + event_members by deterministic │ + │ │ event_key; retire stale auto memberships; pick canonical │ + │ ▼ │ + │ (f) PROJECT write kind='transfer' on excluded legs (back-compat); │ + │ purchase legs stay kind='expense' (no hidden spend) │ + └──────────────────────────────────────────────────────────────────────────────┘ + │ + ▼ + AI categorization (unchanged, skips kind='transfer') +``` + +Layer 1 answers "is this the same row I already stored?" (same account, same sign, exact hash). Layer 2 answers "do these N distinct rows describe one real-world money event?" (opposite sign, or one-to-many, across accounts). They are deliberately different problems with different candidate generation and different outputs (skip-insert vs link). 1:1 duplicate, opposite-sign transfer pair, and one-large-to-many-small aggregate are three distinct linkage tasks and are treated as such. + +### New schema (additive migration `022_financial_events.sql`) + +Migrations run inside `db.transaction()` with `foreign_keys` toggled off and re-verified by the runner (see `migrate.ts`), so this migration does not touch the pragma itself, and all of `CREATE TABLE` / `ALTER TABLE ADD COLUMN ... REFERENCES` / `CREATE INDEX` / `CREATE VIEW` are valid inside that transaction (verified against better-sqlite3). Nothing mutates or deletes a transaction row, so both account ledgers stay intact (the "link, do not collapse" rule). + +```sql +-- 022_financial_events.sql + +CREATE TABLE financial_events ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + -- Deterministic key so a re-sync re-derives the SAME event id instead of + -- spawning a duplicate. See computeEventKey() below. + event_key TEXT NOT NULL, + event_type TEXT NOT NULL CHECK(event_type IN ( + 'internal_transfer', -- account A debit <-> account B credit + 'credit_card_payment', -- bank debit <-> card credit (1:1 legs) + 'cc_statement', -- one card bill <-> N card purchases (N:1) + 'atm_withdrawal' -- cash leaving a tracked account (single leg) + )), + status TEXT NOT NULL DEFAULT 'suggested' + CHECK(status IN ('suggested','confirmed','rejected')), + source TEXT NOT NULL DEFAULT 'auto' + CHECK(source IN ('auto','user')), + confidence REAL, -- Fellegi-Sunter P(match), 0..1 + match_weight REAL, -- log2 total match weight (auditable) + -- Human-readable reasons + the raw comparison vector, so a merge is always + -- explainable and undoable. + reasons_json TEXT, + canonical_txn_id INTEGER REFERENCES transactions(id), + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + UNIQUE(workspace_id, event_key) +); + +CREATE TABLE event_members ( + event_id INTEGER NOT NULL REFERENCES financial_events(id) ON DELETE CASCADE, + transaction_id INTEGER NOT NULL REFERENCES transactions(id), + workspace_id INTEGER NOT NULL, + -- 'outflow' / 'inflow' for pairs; 'bill' / 'purchase' for cc_statement N:1. + role TEXT NOT NULL CHECK(role IN ('outflow','inflow','bill','purchase')), + -- The kind the leg had before THIS event excluded it. Restores precisely on + -- undo / stale-retire. NULL means "we did not change this leg" (e.g. purchase + -- legs in a cc_statement, which keep kind='expense'). + prior_kind TEXT, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + -- A transaction belongs to at most one event; re-running upserts in place. + PRIMARY KEY (workspace_id, transaction_id) +); + +CREATE INDEX idx_event_members_event ON event_members(event_id); + +-- Match-key columns precomputed once, reused by every blocking pass. +ALTER TABLE transactions ADD COLUMN match_amount_cents INTEGER; -- ROUND(charged_amount*100) +ALTER TABLE transactions ADD COLUMN norm_description TEXT; -- normalizeForMatch(description) +ALTER TABLE transactions ADD COLUMN event_id INTEGER REFERENCES financial_events(id); + +-- One-time backfill of the cent column for ALL existing rows, so the first +-- post-migration sync can match against history, not just the new window. +-- norm_description is backfilled lazily in stage (a) (it needs the JS +-- normalizer, which SQL cannot express) but match_amount_cents is pure SQL. +UPDATE transactions SET match_amount_cents = CAST(ROUND(charged_amount * 100) AS INTEGER); + +-- Blocking index: candidate generation joins on (workspace, abs amount). +CREATE INDEX idx_txn_block ON transactions(workspace_id, match_amount_cents); +CREATE INDEX idx_txn_event ON transactions(workspace_id, event_id); +``` + +`event_id` on `transactions` is a denormalized pointer kept in sync with `event_members`, so the candidate query can cheaply exclude rows already bound to a human-confirmed event without a join. It is the only place that pointer is authoritative; `event_members` remains the source of truth for membership. + +A note on the cent column: Layer 1's dedup hash is computed over `originalAmount` (the foreign-currency face value), but matching pairs across accounts must compare what actually moved in the shared currency, which is `charged_amount`. So `match_amount_cents` is deliberately derived from `charged_amount`, and every block additionally requires equal `charged_currency` so a 100 USD debit never pairs with a 100 ILS credit. + +### Stage (a) Normalize + +Runs after Layer 1 inserts. We need a Hebrew-aware normalizer because edit-distance and phonetic methods (Soundex, Metaphone) are unreliable on RTL and transliterated text, so we lean on amount and date as the high-signal fields and only normalize the string for a coarse comparator. + +```typescript +// src/server/lib/event-grouping/normalize.ts +import "server-only"; + +const FINAL_FORMS: Record = { + "\u05da": "\u05db", // final kaf -> kaf + "\u05dd": "\u05de", // final mem -> mem + "\u05df": "\u05e0", // final nun -> nun + "\u05e3": "\u05e4", // final pe -> pe + "\u05e5": "\u05e6", // final tsadi-> tsadi +}; + +export function normalizeForMatch(description: string): string { + return description + .replace(/[\u200e\u200f\u202a-\u202e]/g, "") // strip LRM/RLM/bidi controls + .replace(/[\u0591-\u05c7]/g, "") // strip niqqud / te'amim + .replace(/[\u05da\u05dd\u05df\u05e3\u05e5]/g, (c) => FINAL_FORMS[c] ?? c) + .replace(/\d{4,}/g, "#") // collapse long digit runs (ref/card #) + .replace(/\s+/g, " ") + .trim() + .toLowerCase(); +} + +export const toCents = (amt: number): number => Math.round(amt * 100); +``` + +Idempotency rule: `norm_description` is backfilled lazily. On every sync we run one bounded statement over the window plus any still-null row: + +```sql +-- written from JS row-by-row inside a transaction, since normalizeForMatch is JS. +-- selection: rows in the window OR with norm_description still NULL. +SELECT id, description FROM transactions +WHERE workspace_id = ? AND (date >= ? OR norm_description IS NULL); +``` + +This never re-touches an already-normalized historical row twice on a normal sync, and `match_amount_cents` is already populated for all rows by the migration backfill, so blocking cannot silently drop a candidate for a NULL cent value. (In SQLite `-NULL = NULL`, which would make a NULL-cent row match nothing, so populating it eagerly is load-bearing, not cosmetic.) + +### Stage (b) Block (candidate generation, never O(n^2)) + +The cheap, high-precision block for both transfers and CC-payment pairs is exact `abs(amount_cents)`, opposite sign, joined within a date window, scoped cross-account. Genuine pairs almost always agree on amount, so the block is exact-amount and the slack lives in the date. We use an asymmetric window biased to the future (about 1 week past, 2 weeks future) because the credit side of a transfer settles after the debit. + +The cross-account predicate is the single most error-prone part and must exactly reproduce the current `isDifferentAccount` semantics: two rows are different accounts when **either** their credential ids are both present and differ, **or** their account numbers differ. A naive `COALESCE(b.credential_id,-b.id) IS NOT COALESCE(a.credential_id,-a.id) AND b.account_number <> a.account_number` is wrong: it fails to pair a checking-to-savings transfer under a single bank login (same `credential_id`, different `account_number`), which is one of the most common real cases in Israel. Verified empirically. The correct predicate: + +```sql +-- src/server/db/queries/events.ts (getTransferCandidatePairs) +SELECT a.id AS outflow_id, b.id AS inflow_id, + a.norm_description AS a_desc, b.norm_description AS b_desc, + a.provider AS a_provider, b.provider AS b_provider, + a.date AS a_date, b.date AS b_date, + a.charged_amount AS a_amt, b.charged_amount AS b_amt, + a.dedup_hash AS a_hash, a.dedup_sequence AS a_seq, + b.dedup_hash AS b_hash, b.dedup_sequence AS b_seq +FROM transactions a +JOIN transactions b + ON b.workspace_id = a.workspace_id + AND b.match_amount_cents = -a.match_amount_cents -- equal magnitude, opposite sign + AND a.charged_amount < 0 AND b.charged_amount > 0 -- a = outflow, b = inflow + AND b.charged_currency IS a.charged_currency -- same currency (NULL-safe) + AND ( + (a.credential_id IS NOT NULL AND b.credential_id IS NOT NULL + AND a.credential_id <> b.credential_id) + OR a.account_number <> b.account_number + ) -- mirrors isDifferentAccount + AND julianday(substr(b.date,1,10)) - julianday(substr(a.date,1,10)) BETWEEN -7 AND 14 +WHERE a.workspace_id = ? + AND a.date >= ? -- sync window anchor on outflow + AND a.id NOT IN (SELECT transaction_id FROM event_members em + JOIN financial_events fe ON fe.id = em.event_id + WHERE em.workspace_id = a.workspace_id + AND (fe.status = 'confirmed' OR fe.source = 'user')) + AND b.id NOT IN (SELECT transaction_id FROM event_members em + JOIN financial_events fe ON fe.id = em.event_id + WHERE em.workspace_id = b.workspace_id + AND (fe.status = 'confirmed' OR fe.source = 'user')); +``` + +Note what changed versus today's `getInternalTransferCandidates`: this query does **not** filter `kind != 'transfer'`. It must see already-flipped rows so the reconcile pass can re-evaluate and, if a prior auto match no longer holds, restore them. The only rows it excludes are those already locked into a human-confirmed/user event. Returned candidate count is `O(n * avg_bucket_size)`; the bucket keys on the exact cent amount via `idx_txn_block`, so it stays tiny even on tens of thousands of rows. This replaces the in-memory double loop in `findInternalTransferPairs` with an indexed SQL block, which is what keeps it sub-O(n^2). + +A second, recall-recovery pass uses `match_amount_cents` bucketed to the nearest 10 agorot with a wider date window, to catch pending/posted drift where amounts shift slightly. Anything it surfaces can only ever become a `suggested` event, never an auto-link, because the amount disagreed. + +The CC-statement N:1 block is separate and explicitly NOT a subset-sum search (subset-sum is exponential and would be both slow and dangerous). For each candidate bill (a non-card-provider row whose `norm_description` matches `CREDIT_CARD_PAYMENT_PATTERNS`, reused verbatim from `transfers.ts`), we take the card credential it names and sum **all** of that card's purchase rows whose `processed_date` falls in the billing window, then compare that single aggregate to the bill amount: + +```sql +-- aggregate over the whole billing window, NOT a per-subset search +SELECT ABS(SUM(charged_amount)) AS purchases_total, COUNT(*) AS n +FROM transactions +WHERE workspace_id = ? AND credential_id = ? + AND kind = 'expense' + AND processed_date >= ? AND processed_date <= ?; +``` + +Because a missing or extra purchase would make the aggregate disagree, a `cc_statement` event is **only ever** created when the aggregate equals the bill within a tight tolerance (1 NIS), and even then it does not delete or net anything. It only excludes the bill leg (see stage (f)). If it does not balance, no `cc_statement` event is created and the bill is left to the ordinary `credit_card_payment` 1:1 path or to `detectKind`'s keyword flip. This guarantees the N:1 logic can never silently hide a real purchase. + +### Stage (c) Score (Fellegi-Sunter additive weights) + +Each pairwise candidate gets a transparent additive score. Per-feature `match weight = log2(m/u)`; total `M = sum of weights`; `P(match) = 2^M / (1 + 2^M)`, clamped to `[0,1]`. Anchors: `M=4 -> ~0.94`, `M=7 -> ~0.99`. We ship hand-set m/u priors (no labels needed to start; these can later be EM-tuned per workspace). + +```typescript +// src/server/lib/event-grouping/score.ts +import "server-only"; +import { matchesInternalTransfer, CREDIT_CARD_PAYMENT_PATTERNS } from "../transfers"; + +interface Feature { name: string; weight: number; } // log2(m/u) + +export function scoreTransferPair(c: TransferCandidatePair): { + weight: number; probability: number; reasons: Feature[]; +} { + const f: Feature[] = []; + + // Opposite-sign, cross-account is the defining transfer signal (strong). + f.push({ name: "opposite_sign_cross_account", weight: 3.0 }); + + // Exact amount agreement (m~0.99, u~0.02 at cent granularity in a workspace). + // Compare in integer cents, never floats, to avoid 0.1+0.2 drift. + if (c.aCents === -c.bCents) f.push({ name: "exact_amount", weight: 4.0 }); + else f.push({ name: "amount_drift", weight: -2.0 }); // rounded-bucket pass only + + // Date proximity, decaying with the gap. + const gap = Math.abs(dayGap(c.aDate, c.bDate)); + if (gap === 0) f.push({ name: "same_day", weight: 2.0 }); + else if (gap <= 2) f.push({ name: "date_within_2d", weight: 1.0 }); + else f.push({ name: "date_within_window", weight: 0.0 }); // already in-window by block + + // Keyword evidence (reuses existing regexes; no new keyword lists). Run on the + // RAW description, because CREDIT_CARD_PAYMENT_PATTERNS were authored against + // raw bank text; norm_description strips digit runs and could weaken a match. + const ccHit = CREDIT_CARD_PAYMENT_PATTERNS.some((re) => re.test(c.aRawDesc) || re.test(c.bRawDesc)); + const xferHit = matchesInternalTransfer(c.aRawDesc) || matchesInternalTransfer(c.bRawDesc); + if (ccHit) f.push({ name: "cc_payment_keyword", weight: 2.0 }); + else if (xferHit) f.push({ name: "internal_transfer_keyword", weight: 1.5 }); + + // Description similarity on the normalized strings (coarse, tie-breaker only). + const sim = jaroWinkler(c.aDesc, c.bDesc); + if (sim >= 0.9) f.push({ name: "desc_similar", weight: 1.0 }); + + const weight = f.reduce((s, x) => s + x.weight, 0); + const probability = Math.min(1, Math.max(0, 2 ** weight / (1 + 2 ** weight))); + return { weight, probability, reasons: f }; +} +``` + +Backward-compat: today's `findInternalTransferPairs` requires a keyword on at least one side AND exact amount AND a 2-day window as a hard gate. Those become weighted features. A keyword-less pair that is exact-amount, same-day, cross-account scores `3.0 + 4.0 + 2.0 = 9.0 -> P ~ 0.998`, so it now auto-links where the old code missed it for lack of a keyword. A keyword-present, exact-amount, within-2-day pair scores `>= 6.5 -> P ~ 0.99`, preserving prior behavior. Crucially, the recall expansion only ever *adds* exclusions for clean exact-amount cross-account pairs; it never expands by relaxing the amount, so it cannot start hiding ordinary spend that merely shares a keyword. + +### Stage (d) Decide (three zones, auto vs suggest) + +Two policy thresholds, not one cutoff (Fellegi-Sunter clerical-review model): + +```typescript +const AUTO_LINK = 0.97; // >= 0.97 -> event created, both legs flipped to transfer now +const SUGGEST = 0.80; // [0.80, 0.97) -> suggested event, needs_review on legs, kind UNCHANGED +// < 0.80 -> candidate dropped, no event row +``` + +- `P >= 0.97`: create/refresh a `financial_events` row and immediately flip the excluded legs to `kind='transfer'` (so analytics are correct now), recording each leg's `prior_kind`. `needs_review` stays false. Exact-amount same-day internal transfers and matched CC-payment pairs land here. This matches every aggregator's rule: classify both legs of a card-bill payment as transfer and exclude both, never net them against purchases. +- `0.80 <= P < 0.97`: create the event but leave both legs' `kind` unchanged and set `needs_review = 1` via the existing `batchSetNeedsReview`. Spend/income totals are unaffected until the user confirms in the review UI; confirming sets `status='confirmed'`, `source='user'` and applies the exclusion. Choosing to leave `kind` unchanged in this band is deliberate: a borderline auto-exclusion that hid a real expense would be a silent data error, so borderline cases must surface, not disappear. +- `P < 0.80`: nothing written. + +Greedy 1:1 assignment is preserved and made reproducible: candidates are processed in descending `P`, ties broken by `(min(a_hash,b_hash), max(a_hash,b_hash))` so the order is identical on every run regardless of row id churn. A transaction already claimed by a higher-scoring event is skipped (enforced by the `event_members` PK). This bounds the blast radius of a bad match and prevents transitive chaining across low-confidence edges. + +The `cc_statement` N:1 decision is separate and never goes through the pairwise scorer: it is created only on exact aggregate balance (stage (b)) and only ever excludes the bill leg. + +### Stage (e) Reconcile (idempotent grouping, deterministic event keys) + +Re-running a sync must not create duplicate events or thrash assignments. Solved with a deterministic `event_key` derived only from stable transaction facts, plus an upsert: + +```typescript +// src/server/lib/event-grouping/keys.ts +import "server-only"; +import crypto from "node:crypto"; + +// Order-independent over the participating (dedup_hash, dedup_sequence) pairs, +// so the same set of legs always yields the same key regardless of scan order. +export function computeEventKey( + eventType: string, + legs: { dedupHash: string; seq: number }[], +): string { + const ids = legs.map((l) => `${l.dedupHash}:${l.seq}`).sort().join("|"); + return crypto.createHash("sha256").update(`${eventType}\u0000${ids}`).digest("hex"); +} +``` + +Keying on `(dedup_hash, dedup_sequence)` rather than the autoincrement `id` means the key survives re-insertion and composes with Layer 1, which guarantees those identifiers are stable and idempotent. Reconcile, all inside one `db.transaction()`: + +```typescript +for (const decided of decisions) { // auto-link + suggest decisions + const key = computeEventKey(decided.type, decided.legs); + const existing = getEventByKey(workspaceId, key); // UNIQUE(workspace_id, event_key) + if (existing) { + // Never overwrite a human decision. + if (existing.status === "confirmed" || existing.source === "user") continue; + upsertEventMeta(existing.id, decided.confidence, decided.matchWeight, decided.reasons, decided.canonicalTxnId); + } else { + const eventId = insertEvent(workspaceId, key, decided); + for (const leg of decided.legs) { + const prior = getTransactionKind(workspaceId, leg.txnId); // capture before flip + upsertMember(eventId, leg.txnId, leg.role, leg.excluded ? prior : null); + setTransactionEventId(workspaceId, leg.txnId, eventId); + } + } +} + +// Retire stale AUTO memberships: any auto/suggested event in the window whose +// legs no longer clear SUGGEST is deleted; legs whose prior_kind was recorded +// are restored to prior_kind, and their event_id cleared. Confirmed/user events +// are untouched. +reconcileStaleAutoEvents(workspaceId, fromDate); +``` + +Because the key is content-addressed and confirmed/user events are skipped, re-running over the same window is a no-op, and a user's confirm/reject/split survives every subsequent sync. This is the generalization of today's `markTransfersByIds(findInternalTransferPairs(...))` + `batchSetNeedsReview`: "upsert events, then project `kind` from event membership." Critically, because the candidate query (stage (b)) no longer pre-filters `kind`, a row that was auto-flipped last sync but whose partner has since vanished (for example a pending row that never posted) will be re-evaluated, score below threshold, and have its `kind` restored from `prior_kind`. That restore path did not and could not exist in the old code. + +Incrementality: the pipeline evaluates only rows with `date >= fromDate` (anchored on the outflow side) plus any row already in a `suggested`/`auto` event touched by the window. It never does a full rebuild on a normal sync. A full rebuild is a separate explicit maintenance action (`rebuildEvents(workspaceId)`), useful after the user widens `monthsToSync`; it clears `event_id`/auto events and re-runs over all history while leaving confirmed/user events intact. + +### Stage (f) Project (analytics read path) + +Reporting keeps reading the same `kind='expense'` / `kind='income'` filters used in every query in `transactions.ts`, so the dashboard is untouched. Projection is: set `kind='transfer'` on auto-linked and confirmed *excluded* legs (internal-transfer pairs both legs; CC-payment pairs both legs; ATM legs), exactly as `markTransfersByIds` does today. The event layer is additive metadata on top of the unchanged `kind` contract. We do **not** rely on `is_excluded` for this, because no analytics query reads that column. + +For the N:1 `cc_statement` case the projection is asymmetric, matching Monarch/YNAB/Copilot: the **bill payment leg** is set `kind='transfer'` (excluded, `prior_kind` recorded), while the **purchase legs stay `kind='expense'`** and remain the single source of spend (their `event_members.prior_kind` is NULL, signalling "we did not change this leg"). The event row makes the previously-invisible relationship auditable ("this 4,200 NIS Isracard debit is composed of these 37 purchases") without changing a single analytics number. A thin view backs any future "show me the event behind this transfer" UI: + +```sql +CREATE VIEW v_transaction_events AS +SELECT t.id AS transaction_id, t.workspace_id, em.role, em.prior_kind, + fe.id AS event_id, fe.event_type, fe.status, fe.confidence, + fe.canonical_txn_id, fe.reasons_json +FROM transactions t +JOIN event_members em ON em.workspace_id = t.workspace_id AND em.transaction_id = t.id +JOIN financial_events fe ON fe.id = em.event_id; +``` + +### Canonical selection + +For a pair, the canonical (representative) transaction is the **outflow** leg (the debit), since that is what the user recognizes as "the payment." For a `cc_statement` event the canonical is the **bill** leg. Canonical is stored on `financial_events.canonical_txn_id` and is what a "transfers" or "events" report displays as the one row per event, avoiding the double-count that listing all legs would cause. + +### Sequence for one sync run (inside `syncWorkspace`) + +1. For each credential, `syncOneCredential` runs Layer 1 unchanged (`insertTransactions`, idempotent ON CONFLICT, which already preserves `kind` via `kind = transactions.kind`). No event logic here, so a single credential's insert is never blocked on cross-account data. +2. After all credentials are inserted, replace the current block at `orchestrator.ts` lines 409-418 with a single call to `groupFinancialEvents(workspaceId, fromDate)`, in the exact spot where `findInternalTransferPairs(...)` is invoked today. +3. `groupFinancialEvents` runs (a) normalize, (b) block (transfer pairs + CC-statement N:1), (c) score, (d) decide, (e) reconcile/upsert/retire, (f) project `kind`. It returns counts surfaced over SSE as a new `stage: "grouping"` event for the sync UI. +4. ATM handling (`getAtmExpenseCandidates` / `treatAtmAsTransfers`) folds into the same layer as `event_type='atm_withdrawal'` single-leg events, so its `kind`-flipping is now recorded with `prior_kind` and is undoable too. The non-transfer ATM path (filing under "Cash & ATM") stays as-is, since it sets a category, not a `kind`. +5. AI categorization runs last, unchanged, and continues to skip `kind='transfer'` (it filters `category_id IS NULL AND kind = ?` per kind), so excluded legs are never sent to the model. + +### Prerequisite and replaced / generalized code + +- **Prerequisite:** widen `BANK_PROVIDERS_SET` in `transfers.ts` beyond `["hapoalim","leumi"]` to the full bank list (`mizrahi`, `discount`, `oneZero`, etc.) before the CC-statement block can recognize their bills. Until then, `detectKind` and the bill detector only fire for the two providers, so the N:1 path is effectively dead for the rest. This is a one-set change but is load-bearing and must ship in the same PR. +- `findInternalTransferPairs` (in-memory greedy double loop, hard keyword gate) is superseded by the SQL block plus the Fellegi-Sunter scorer. It can remain as a pure, unit-testable helper, but the orchestrator calls `groupFinancialEvents` instead of `markTransfersByIds(findInternalTransferPairs(...))`. +- `detectKind` at insert time stays as the cheap first guess (it sets the initial `kind` so single-account data and the non-paired majority are correct immediately), but the event layer is authoritative and can override or restore it via `prior_kind`. +- `markTransfersByIds` and `batchSetNeedsReview` are still used, now driven by event decisions rather than ad-hoc pair lists. A new `restoreKindFromPriorKind` helper backs the stale-retire and user-reject paths. + +Relevant files for implementation: new `src/server/lib/event-grouping/{normalize,score,keys,group}.ts`, new `src/server/db/queries/events.ts`, new migration `src/server/db/migrations/022_financial_events.sql`, the `BANK_PROVIDERS_SET` widening in `src/server/lib/transfers.ts`, a new non-kind-filtering candidate query alongside `getInternalTransferCandidates` in `src/server/db/queries/transactions.ts`, and the swap in `src/server/sync/orchestrator.ts` at lines 409-418. + +--- + +## Matching Algorithms and Confidence Scoring + +This section specifies how Spent generates candidate links between transaction rows, scores them, and decides whether to auto-merge, suggest, or ignore. It generalizes the keyword-only `detectKind` / `findInternalTransferPairs` logic (`src/server/lib/transfers.ts`, `src/server/lib/internal-transfers.ts`) into a feature-scored matcher that emits a persisted confidence and human-readable reasons. The event-grouping schema (`financial_events`, `event_members`) and the review/undo workflow are specified in their own sections; here we produce the `(score, reasons, event_type, member_ids)` tuple that those layers consume. + +Three invariants are absolute and constrain everything below: + +1. The matcher never deletes rows, never mutates `charged_amount`, and never nets one row against another. It only proposes links. The underlying card purchases always remain the single source of truth for spend, exactly once. +2. The matcher only ever flips rows OUT of spend/income into `transfer`. It never flips a `transfer` back to `expense` (that is the undo path's job, driven by event deletion), and it never reclassifies a row a user has touched (`category_source = 'user'` or `kind` set by a prior accepted event). +3. Every auto-decision is explainable (persisted `reasons[]` and feature vector) and reversible (delete event, restore prior `kind`). + +Design stance, grounded in research: Monarch, Copilot, and Simplifi all avoid double-counting credit-card bills by typing both legs as transfers and excluding them by category, and none publicly document a real two-sided pair matcher. We go further (true pair linking is the differentiator) but adopt their safety stance: matching is reversible and the card purchases remain the spend. We borrow the three-zone auto/suggest/ignore decision model and additive log-weight scoring from the record-linkage literature (Fellegi-Sunter, Splink), and Plaid's confidence gate (VERY_HIGH > 98%, HIGH > 90%, MEDIUM, LOW) as the action threshold. + +### 1. Normalization + +All fuzzy comparison runs on a normalized form of `description` (and `memo`), never the raw string. Israeli scraper descriptions are mostly Hebrew RTL, occasionally English, and carry provider boilerplate plus masked card digits. Hebrew is hostile to phonetic and transliteration methods (Soundex/Metaphone are English-bound), so we normalize aggressively in the original script and lean on amount/date as the high-signal fields. + +```typescript +// src/server/lib/matching/normalize.ts +import "server-only"; + +const RTL_MARKS = /[\u200e\u200f\u202a-\u202e\u2066-\u2069]/g; // LRM/RLM/embeddings/isolates +const NIQQUD = /[\u0591-\u05c7]/g; // Hebrew cantillation + vowel points + +// Final-form Hebrew letters folded to their base form so a word at the end of a +// description collapses with the same word mid-string. Keyed by code point. +const HEBREW_FINALS: Record = { + "\u05da": "\u05db", // ך -> כ + "\u05dd": "\u05de", // ם -> מ + "\u05df": "\u05e0", // ן -> נ + "\u05e3": "\u05e4", // ף -> פ + "\u05e5": "\u05e6", // ץ -> צ +}; +const HEBREW_FINALS_RE = /[\u05da\u05dd\u05df\u05e3\u05e5]/g; + +// Provider/statement boilerplate that adds no merchant signal. Extend as banks +// reveal patterns. NOTE: these are stripped only for FUZZY comparison; the +// CC-payment and transfer keyword classifiers in transfers.ts still see the raw +// description, so removing "חיוב" here does not weaken classification there. +const BOILERPLATE: readonly RegExp[] = [ + /\u05d1\u05e2"?\u05de/g, // בע"מ (Ltd.) + /\bLTD\.?\b/gi, + /\bINC\.?\b/gi, + /\u05e4\u05e2\u05d5\u05dc\u05d4/g, // פעולה (transaction filler) + /\u05ea\u05e9\u05dc\u05d5\u05dd\s+\d+\s+\u05de\u05ea\u05d5\u05da\s+\d+/g, // "תשלום N מתוך M" +]; + +// Masked card / reference digits: "****1234", "xx-1234", standalone 4-6 digit runs. +const MASKED_DIGITS = /(?:[*xX#]{2,}[\s-]*)?\d{4,6}\b/g; + +export interface NormalizedDescription { + /** Lowercased, RTL-stripped, boilerplate-folded canonical string. */ + canonical: string; + /** Sorted unique token set for Jaccard / token-set similarity. */ + tokens: string[]; + /** Any masked card / reference digit groups, pulled out as their own signal. */ + digitGroups: string[]; +} + +export function normalizeDescription(raw: string): NormalizedDescription { + // Extract digit groups BEFORE stripping, normalize each to bare digits, then + // keep only the trailing 4 so "****1234" and "xx-1234" and "1234" all unify. + const digitGroups = Array.from( + new Set((raw.match(MASKED_DIGITS) ?? []).map((d) => d.replace(/\D/g, "").slice(-4))), + ).filter((d) => d.length === 4); + + let s = raw.normalize("NFKC").replace(RTL_MARKS, "").replace(NIQQUD, ""); + s = s.replace(HEBREW_FINALS_RE, (c) => HEBREW_FINALS[c] ?? c); + for (const re of BOILERPLATE) s = s.replace(re, " "); + s = s.replace(MASKED_DIGITS, " "); + s = s + .toLowerCase() + .replace(/[^\p{L}\p{N}\s]/gu, " ") // drop punctuation, keep Hebrew + Latin letters and numbers + .replace(/\s+/g, " ") + .trim(); + + const tokens = Array.from(new Set(s.split(" ").filter((t) => t.length > 1))).sort(); + return { canonical: s, tokens, digitGroups }; +} +``` + +Key choices and why: + +- Strip RTL control marks first. Israeli scrapers embed `\u200f`/`\u200e` inconsistently, so two visually identical descriptions can differ byte-for-byte. This is also why the existing `computeDedupHash` over the raw `description` can occasionally let a pending row and its posted twin escape exact dedup. Normalizing closes that gap for the fuzzy layer WITHOUT touching the exact-hash layer (the hash stays raw; see section 8). +- Unify final-form Hebrew letters (ם→מ etc.) so a word at the end of a description collapses with the same word mid-string. We keep niqqud-stripping and final-letter folding and deliberately do NOT transliterate, since transliteration explodes one Hebrew name into thousands of Latin variants. +- Pull masked card digits into `digitGroups` (normalized to the bare last-4) rather than leaving them inline. The last-4 is a strong link between a bank debit ("חיוב כרטיס ...1234") and a card account; inline it is just noise. Last-6 is reduced to last-4 because providers mask inconsistently; matching on last-4 is the reliable common denominator. + +### 2. Blocking (candidate generation, never O(n^2)) + +Full pairwise comparison over a sync window scales quadratically and is infeasible at tens of thousands of rows. We block first, comparing only rows that share a cheap key, then score within blocks. The cheapest high-precision block is exact rounded amount plus currency plus a date bucket, because genuine opposite legs almost always agree on magnitude (the slack is in the date). + +Candidate rows are always scoped to one workspace and bounded to the sync window, mirroring the existing `getInternalTransferCandidates(workspaceId, from)`. Two passes (a disjunction of keys) recover recall: an exact-amount pass (can reach auto) and a whole-shekel pass that absorbs FX and rounding drift (suggest-only). + +A correctness note on the join predicate. SQLite's `=` already treats two non-NULL currencies correctly, but NULL currencies must be matched explicitly, because `NULL = NULL` is NULL (false) in a WHERE/ON clause. We therefore compare a coalesced currency, not the bare column. We also gate both sides on `category_source IS NOT 'user'` and `kind <> 'transfer'` so the matcher never re-touches a row a user owns or a row a prior accepted event already classified (invariant 2). + +```sql +-- Pass 1: exact opposite-sign pairs within a date window, same workspace. +-- Bounded by the amount+currency block (each bucket holds a handful of rows), +-- so it is near-linear in practice. Debit (negative) joined to credit (positive). +SELECT d.id AS debit_id, c.id AS credit_id, + d.charged_amount AS debit_amount, c.charged_amount AS credit_amount, + d.date AS debit_date, c.date AS credit_date, + d.account_number AS debit_acct, c.account_number AS credit_acct, + d.credential_id AS debit_cred, c.credential_id AS credit_cred, + d.provider AS debit_provider, c.provider AS credit_provider, + d.description AS debit_desc, c.description AS credit_desc, + d.type AS debit_type, c.type AS credit_type +FROM transactions d +JOIN transactions c + ON c.workspace_id = d.workspace_id + AND COALESCE(c.charged_currency, '') = COALESCE(d.charged_currency, '') + AND ROUND(ABS(c.charged_amount), 2) = ROUND(ABS(d.charged_amount), 2) -- exact magnitude + AND c.charged_amount > 0 AND d.charged_amount < 0 -- opposite sign + AND c.id <> d.id + AND ( -- different account + c.credential_id IS NULL OR d.credential_id IS NULL + OR c.credential_id <> d.credential_id + OR c.account_number <> d.account_number + ) + AND ABS(julianday(substr(c.date, 1, 10)) - julianday(substr(d.date, 1, 10))) <= :dayWindow +WHERE d.workspace_id = :workspaceId + AND d.date >= :from + AND d.kind <> 'transfer' AND c.kind <> 'transfer' + AND d.category_source IS NOT 'user' AND c.category_source IS NOT 'user' + AND d.charged_amount <> 0; +``` + +To make this index-driven instead of a full scan, add a covering expression index on the blocking columns. The leading columns must match the equality predicates in the ON clause; `date` trails so the planner can range-probe the date window: + +```sql +CREATE INDEX IF NOT EXISTS idx_txn_block_amount + ON transactions ( + workspace_id, + charged_currency, + ROUND(ABS(charged_amount), 2), + date + ); +``` + +SQLite supports indexes on the deterministic expression `ROUND(ABS(charged_amount), 2)`, so the join probes the index per debit row. Caveat that must be respected: the index uses bare `charged_currency`, while the join uses `COALESCE(charged_currency, '')`. The planner will only use the index for the currency equality when the predicate is on the bare column. We therefore split the candidate fetch by currency-presence in the calling code (one parameterized query for non-NULL currency that uses the index, a rare fallback for NULL currency that does not), rather than wrapping the column in COALESCE inside the indexed predicate. Israeli scrapers virtually always populate `charged_currency` (default ILS), so the NULL branch is a cold path. + +Pass 2 is identical but blocks on `ROUND(ABS(...), 0)` (whole-shekel) with a wider `:dayWindow`. It backs a second index `idx_txn_block_amount_shekel` on `(workspace_id, charged_currency, ROUND(ABS(charged_amount), 0), date)`. Pass 2 only ever produces suggest-tier candidates, never auto-merges, because a rounded-amount match is inherently lower precision. For credit-card-bill reconciliation, blocking is different (one-to-many, sum-based) and is specified in section 5.2. + +Idempotency of candidate generation: rows already in an accepted event are skipped by the `kind <> 'transfer'` filter (accepted events flip members to `transfer`). Rows in a still-pending suggested event are skipped by joining against `event_members` for events in `state = 'suggested'` (defined in the schema section) so a re-sync does not generate duplicate suggestions for the same pair. The pipeline (section 7) enforces this with a `claimed` set seeded from existing event members. + +### 3. Feature set + +For each candidate pair the matcher computes a feature vector. Exact/numeric comparators run on the high-signal fields (amount, date, sign, account/provider kind); fuzzy string comparators run only on the normalized description. This is the core guidance: do not waste fuzzy logic where exact equality is the right test. + +| Feature | Type | Definition | +|---|---|---| +| `amountExact` | bool | `ROUND(abs(a),2) === ROUND(abs(b),2)` | +| `amountTolerance` | float 0..1 | `1 - min(1, abs(abs(a)-abs(b)) / (abs(a) * FX_TOL))`; `FX_TOL = 0.02` to absorb FX + rounding when currencies differ | +| `oppositeSign` | bool | `sign(a) !== sign(b)` and neither is 0 | +| `dateGapDays` | int | signed `creditDay - debitDay`; settlement runs forward, so the window is asymmetric (favor credit on or after the debit) | +| `dateProximity` | float 0..1 | `max(0, 1 - abs(dateGapDays) / DATE_HORIZON)`, `DATE_HORIZON = 5` | +| `crossAccount` | bool | different `credential_id`, or different `account_number` when credential is shared/null (reuses `isDifferentAccount`) | +| `directionMatch` | bool | source and dest account `kind` are consistent with the event type (e.g. CC payment = `bank` debit + `card` credit, via `BankProviderInfo.kind`) | +| `descJaroWinkler` | float 0..1 | Jaro-Winkler over the two `canonical` strings | +| `descTokenSet` | float 0..1 | token-set ratio (Jaccard over `tokens`) | +| `digitOverlap` | bool | any shared `digitGroups` entry (shared masked card last-4) | +| `keywordTransfer` | bool | either side matches the internal-transfer keyword set (via `matchesInternalTransfer`) | +| `keywordCardPayment` | bool | either side matches the CC-payment keyword set (via a new `matchesCreditCardPayment` exported from `transfers.ts`) | +| `installmentConsistent` | bool | `type`/`installmentTotal` agree (an installments row should not pair with a one-shot transfer) | + +The provider `kind` (`bank` vs `card`) comes from `BANK_PROVIDERS` in `src/lib/types.ts` (`BankProviderInfo.kind: BankKind = "bank" | "card"`), looked up by the row's `provider` column. This replaces the brittle hardcoded `BANK_PROVIDERS_SET = new Set(["hapoalim", "leumi"])` currently in `transfers.ts`, which is wrong: it omits mizrahi, discount, oneZero, and every other bank, so `detectKind` today silently misclassifies their credits as `expense`. Section 8 specifies the required refactor of `isBankProvider` to read `BANK_PROVIDERS` instead. + +Two functions the draft assumed are reusable but are not yet exported must be added in `transfers.ts`: the internal-transfer regex set is private (only `matchesInternalTransfer` is exported, which is sufficient for `keywordTransfer`), and there is no `matchesCreditCardPayment` predicate (only the private `matchesTransferPattern` over `CREDIT_CARD_PAYMENT_PATTERNS`). We export `matchesCreditCardPayment` as a thin wrapper so the feature builder reuses the single regex source. + +### 4. Fuzzy string similarity: which metric and why + +We use two complementary string comparators and take their max as the description signal, because Israeli merchant/transfer descriptions vary in both spelling and token order: + +- Jaro-Winkler (`descJaroWinkler`) for spelling drift with a reliable prefix. Israeli statements tend to keep a stable leading token (the bank/branch or "העברה" prefix) and vary the tail, which the Jaro-Winkler prefix bonus rewards. Caveat: Jaro-Winkler is unreliable on very short strings, so we gate it (`canonical.length >= 4` on both sides) and fall back to token-set otherwise. +- Token-set ratio / Jaccard (`descTokenSet`) for reordered or extra tokens. Bank descriptions frequently reorder ("העברה לחשבון" vs "לחשבון העברה") or add a branch token; set overlap ignores order. + +Damerau-Levenshtein (handles adjacent transpositions, common in Hebrew typos) is a reasonable alternative to Jaro-Winkler behind the same interface; we default to Jaro-Winkler for the prefix behavior. Embeddings (Sentence-BERT for "AMZN" ~ "Amazon") are explicitly out of scope for v1: they add a model dependency to a local-only app and the research positions them as an optional extra comparator, not a replacement for the transparent scorer. We leave a seam (`descSemantic?: number`) in the feature vector so a future optional embedding signal slots in additively. + +Thresholds for the description signal: `descSim = max(descJaroWinkler, descTokenSet)`. We treat `>= 0.90` as strong agreement, `0.75 .. 0.90` as weak agreement, `< 0.75` as disagreement. These feed the weighted score below rather than gating on their own, since amount + date + opposite-sign carry most of the discriminative power for transfers (Hebrew description quality is too noisy to gate on). + +### 5. Per-event-type matching rules + +Each event type has its own candidate generator and scoring profile. The matcher runs them in priority order (CC payment, then internal transfer, then ATM, then loan/investment) and a row belongs to at most one accepted event. + +#### 5.1 internal_transfer (1:1 opposite legs) + +This generalizes `findInternalTransferPairs`. Candidate = opposite-sign, cross-account, same-currency (or FX-tolerant), amounts equal within tolerance, dates within window. The current code requires a transfer keyword on at least one side as a HARD gate (lines 99-103 of `internal-transfers.ts`); we soften that into a scored feature so a keyword-less but otherwise perfect pair (exact amount, opposite sign, cross-account, 0-day gap) can still reach the suggest tier instead of being silently dropped. Person-to-person payments (the main false positive per Monarch) stay in suggest, never auto-merge, because they look identical to internal transfers but are real outflows. + +We preserve the existing greedy, deterministic 1:1 selection (`sortKey` order, closest-date wins, no credit reused) so behavior on currently-matched pairs is unchanged; scoring is layered on top, not substituted. + +#### 5.2 credit_card_payment (1:1 leg AND the N:1 bill link) + +This is the hard, high-value case, with two distinct linkages that must NOT be conflated: + +1. The transfer-pair leg: the bank debit ("חיוב כרטיסי אשראי ...1234", `provider.kind === "bank"`, negative) pairs 1:1 with the credit posted on the card account ("תשלום/זיכוי", `provider.kind === "card"`, positive). This is an ordinary opposite-sign pair, scored like an internal transfer but with `keywordCardPayment` and `directionMatch (bank->card)` adding weight, and `digitOverlap` (shared last-4) a strong bonus. Both legs become `kind = 'transfer'`. Critically, MANY card scrapers do NOT emit a payment-credit row on the card account, so this leg is often absent. The design must not require it: in its absence we fall straight to linkage 2 and classify the bank debit alone as `transfer` on keyword + provider, exactly as `detectKind` does today. + +2. The bill-to-purchases aggregate link: the single bank debit equals the SUM of the card account's purchases within a billing cycle. This is one-to-many and must NEVER be matched by per-transaction amount similarity. We link the bill to the set of purchases for audit/visibility, but we do NOT net them: the purchases remain the spend, exactly once. This is precisely what today's design lacks: the bill is flagged `transfer` so it is not double-counted, but the relationship is invisible and unauditable. + +```typescript +// src/server/lib/matching/cc-bill.ts (pseudocode) +// For each bank debit that looks like a CC bill payment, find the card-account +// purchase set whose ABS-sum matches the debit within a cycle window. + +const CYCLE_WINDOW_DAYS = 38; // Israeli cards bill monthly; ~35 days + slack. + +interface CcBillMatch { + billTxnId: number; + cardCredentialId: number; + memberPurchaseIds: number[]; + sumAbs: number; + residual: number; // abs(billAbs - sumAbs) + reasons: string[]; + memberConfidence: "auto" | "suggest"; +} + +function reconcileCcBill(bill: BankDebit, cardPurchases: CardPurchase[]): CcBillMatch | null { + const billAbs = Math.abs(bill.chargedAmount); + + // Card purchases this bill plausibly settles: same currency, in the cycle + // window ENDING at the bill date (purchases precede settlement). Exclude + // installments-future rows and any row a user owns. + const windowed = cardPurchases.filter( + (p) => + p.kind === "expense" && + p.categorySource !== "user" && + (p.chargedCurrency ?? "") === (bill.chargedCurrency ?? "") && + daysBetween(p.date, bill.date) >= 0 && + daysBetween(p.date, bill.date) <= CYCLE_WINDOW_DAYS, + ); + if (windowed.length === 0) return null; + + const fullSum = sumAbs(windowed); + + // Exact cycle total: cheap, exact, the common happy path. Auto-confirm the link. + if (approxEqual(fullSum, billAbs, 0.01)) { + return mk(bill, windowed, fullSum, billAbs, ["Bill equals full cycle total of N purchases"], "auto"); + } + + // Otherwise: partial/minimum payment, or installments split the cycle. Do NOT + // attempt subset-sum (NP-hard; a WRONG subset is worse than none, and could + // hide a real expense if it ever drove netting). Link to the whole cycle with + // a residual note and leave the member set in SUGGEST. The classification of + // the bill as `transfer` is independent and still auto-applies (linkage 1). + return mk( + bill, + windowed, + fullSum, + billAbs, + [`Partial payment: bill ${billAbs} vs cycle total ${fullSum}, residual ${Math.abs(billAbs - fullSum)}`], + "suggest", + ); +} +``` + +The critical correctness rule: reconciliation only ever (a) types the bill (and its mirror credit, if present) as `transfer` so it leaves spend, and (b) records an audit link to the cycle's purchases. It never subtracts the bill from the purchases, never deletes a purchase, and never converts a purchase to `transfer`. Partial payments, minimum payments, and installment cards mean the bill rarely equals a clean subset, so we deliberately avoid subset-sum and link to the whole cycle with a residual annotation, leaving the member set in suggest. + +#### 5.3 atm_withdrawal + +Reuses `isAtmWithdrawal`. An ATM row is a single transaction, not a pair, so it has no candidate-generation step. It is classified by keyword: filed under "Cash & ATM" by default, or flipped to `transfer` when the `treatAtmAsTransfers` setting is on (the existing `orchestrator.ts` behavior, lines 422-431). It enters the scored pipeline only to attach a reason string and a confidence (keyword match = HIGH), so the new event layer can show and undo it like any other classification. No change to the existing setting semantics. + +#### 5.4 loan_repayment and investment_transfer + +Both are bank-debit to an off-platform or off-budget destination. When the counterpart account is not tracked in Spent (common: a mortgage at another institution, a brokerage), there is no opposite leg to pair, so these are single-row classifications gated on keyword + provider, scored like ATM. When the destination IS tracked (its own credential synced into Spent), they reduce to the 1:1 internal-transfer rule with a `directionMatch` expecting `bank -> investment/loan` account kinds. Following YNAB's budget-boundary rule, a transfer crossing into an UNTRACKED account is the legitimate outflow and may stay categorized (not excluded), whereas a transfer between two tracked owned accounts is excluded; this is a per-event-type policy flag, not new matching logic. + +### 6. Confidence scoring + +We use a transparent additive log-weight score (Fellegi-Sunter style: per-feature weight `= log2(m/u)`, total weight additive, `P = 2^M / (1 + 2^M)`). Additive weights keep the score explainable, which is mandatory because every auto-decision must be shown to the user and undone. Weights are seeded from the research reference points (weight 4 ~ p 0.94, weight 7 ~ p 0.99), are tunable per event type, and should later be validated against a clerically-labeled sample, never frozen as magic numbers. + +```typescript +// src/server/lib/matching/score.ts +import "server-only"; + +export interface FeatureVector { + amountExact: boolean; + amountTolerance: number; // 0..1 + oppositeSign: boolean; + dateProximity: number; // 0..1 + dateGapDays: number; + crossAccount: boolean; + directionMatch: boolean; + descSim: number; // max(jaroWinkler, tokenSet) + digitOverlap: boolean; + keywordTransfer: boolean; + keywordCardPayment: boolean; + installmentConsistent: boolean; + descSemantic?: number; // reserved seam for a future optional embedding signal +} + +// Per-event-type weight table (log2 match weights). Positive => evidence FOR a +// link; negative => evidence AGAINST. Seeded, not learned (yet). +const WEIGHTS_INTERNAL_TRANSFER = { + oppositeSign: 3.0, // structural prerequisite; absence is disqualifying + crossAccount: 2.5, + amountExact: 3.0, // exact magnitude is the strongest single signal + amountToleranceScale: 2.0, // multiplied by amountTolerance when not exact + dateProximityScale: 2.0, // multiplied by dateProximity + descStrong: 1.5, // descSim >= 0.90 + descWeak: 0.5, // 0.75 <= descSim < 0.90 + keywordTransfer: 1.5, + installmentInconsistent: -2.5, +} as const; + +export interface ScoreResult { + weight: number; + probability: number; + reasons: string[]; +} + +export function scoreInternalTransfer(f: FeatureVector): ScoreResult { + const w = WEIGHTS_INTERNAL_TRANSFER; + let M = 0; + const reasons: string[] = []; + + // Structural prerequisites: their absence is disqualifying, not just penalized. + // Without these the row pair cannot be a transfer regardless of other signals. + if (!f.oppositeSign) return { weight: -Infinity, probability: 0, reasons: ["Not opposite sign"] }; + if (!f.crossAccount) return { weight: -Infinity, probability: 0, reasons: ["Same account"] }; + M += w.oppositeSign + w.crossAccount; + reasons.push("Opposite sign", "Different accounts"); + + if (f.amountExact) { + M += w.amountExact; + reasons.push("Exact amount match"); + } else { + M += w.amountToleranceScale * f.amountTolerance; + reasons.push(`Amounts within ${(100 * (1 - f.amountTolerance)).toFixed(1)}%`); + } + + M += w.dateProximityScale * f.dateProximity; + reasons.push(f.dateGapDays === 0 ? "Same day" : `${Math.abs(f.dateGapDays)} day gap`); + + if (f.descSim >= 0.9) { + M += w.descStrong; + reasons.push("Descriptions strongly match"); + } else if (f.descSim >= 0.75) { + M += w.descWeak; + reasons.push("Descriptions partially match"); + } + + if (f.keywordTransfer) { + M += w.keywordTransfer; + reasons.push("Transfer keyword present"); + } + if (!f.installmentConsistent) { + M += w.installmentInconsistent; + reasons.push("Installment mismatch"); + } + + const probability = 2 ** M / (1 + 2 ** M); + return { weight: M, probability, reasons }; +} +``` + +For `credit_card_payment` the same machinery runs with a profile that adds `keywordCardPayment` (+2.0), `directionMatch` bank->card (+2.0), and `digitOverlap` shared last-4 (+2.5), and drops the `keywordTransfer` term. `crossAccount` is still a hard prerequisite. The bill-to-purchases aggregate link (5.2) uses a separate, simpler score: exact cycle-sum match dominates (+4.0), `directionMatch` (+1.5), within-cycle-window (+1.0); a nonzero residual caps the member-set decision at suggest regardless of score. + +`oppositeSign` and `crossAccount` are treated as hard prerequisites (return `-Infinity`) rather than large negative penalties, because no accumulation of weak description/date signal should ever drag a same-account or same-sign pair across the auto threshold. This is the safety property that prevents a real expense from being silently merged away. + +### 7. Decision: auto-merge / suggest / ignore + +Three zones, not one cutoff (Fellegi-Sunter clerical-review band; Plaid's auto-vs-suggest gate). Thresholds are policy choices trading review volume against false merges, and are stored in `settings` so an install can tune them. + +| Probability | Weight (approx) | Action | +|---|---|---| +| `>= 0.97` | `>= ~5` | `auto`: create the event, flip member rows to `transfer`, set `needs_review = 0` | +| `0.80 .. 0.97` | `~2 .. 5` | `suggest`: create the event in a proposed state, set `needs_review = 1`, surface in the review queue | +| `< 0.80` | `< ~2` | `ignore`: no event created (still queryable as a near-miss for the manual-match UI) | + +Auto-merge is reserved for the high-precision tier (exact amount + tight date + cross-account + opposite sign, optionally a keyword or shared last-4). The middle band (rounded amount, several days apart, partial description match, P2P-looking) always goes to suggest, because amounts and names legitimately drift between pending and posted, and P2P false positives are common. The whole-shekel blocking pass (section 2) is clamped so it can only ever produce suggest or ignore, never auto, regardless of computed probability. + +```typescript +// src/server/lib/matching/pipeline.ts (candidate -> score -> decision loop) +export type MatchAction = "auto" | "suggest" | "ignore"; + +const T_AUTO = 0.97; +const T_SUGGEST = 0.8; + +function decide(p: number, blockingPass: "exact" | "shekel"): MatchAction { + if (blockingPass === "shekel") { + return p >= T_SUGGEST ? "suggest" : "ignore"; // rounded-amount can never auto + } + if (p >= T_AUTO) return "auto"; + if (p >= T_SUGGEST) return "suggest"; + return "ignore"; +} + +export interface ProposedEvent { + eventType: EventType; + memberIds: number[]; + confidence: number; + reasons: string[]; + action: MatchAction; +} + +export function runMatcher( + workspaceId: number, + from: string, + existingMemberIds: ReadonlySet, // members of already-accepted or still-open suggested events +): ProposedEvent[] { + const proposed: ProposedEvent[] = []; + // A row joins at most one accepted event. Seed with rows already claimed by a + // prior sync so re-runs are idempotent and never double-propose the same pair. + const claimed = new Set(existingMemberIds); + + for (const eventType of EVENT_TYPE_PRIORITY) { + // cc_payment, internal_transfer, atm, loan/investment + for (const cand of generateCandidates(workspaceId, from, eventType)) { + if (cand.memberIds.some((id) => claimed.has(id))) continue; + const f = buildFeatures(cand, eventType); + const { probability, reasons } = scoreFor(eventType, f); + const action = decide(probability, cand.blockingPass); + if (action === "ignore") continue; + + proposed.push({ eventType, memberIds: cand.memberIds, confidence: probability, reasons, action }); + // Claim only on auto. Suggested pairs do NOT claim, so a stronger CC-payment + // suggestion later in the same run can still consider the same row; the + // review layer resolves competing suggestions when the user accepts one. + if (action === "auto") for (const id of cand.memberIds) claimed.add(id); + } + } + return proposed; +} +``` + +This greedy, priority-ordered loop preserves the deterministic 1:1 behavior of `findInternalTransferPairs` (closest-date wins, no row reused for an accepted event) while adding scoring and the suggest tier. The `reasons[]` array is persisted on the event so every merge is explainable and the review UI shows "why" without recomputing. Seeding `claimed` with existing event members is what makes a second sync idempotent: a pair already auto-merged or already sitting in the review queue is not re-proposed. + +### 8. Outputs and backward compatibility + +- The matcher produces proposed events, not row mutations. The event layer, on accept, flips member rows to `kind = 'transfer'` via the existing `markTransfersByIds` / `setTransactionKind` paths, so all current spend/income queries (`getMonthlySummary`, `getCategoryBreakdown`, `getTransactionsSummary`, every `kind = 'expense'` filter) keep working unchanged. The legacy `kind` column stays the source of truth for analytics; events are an additive grouping/audit layer on top. +- Required upstream fix, not optional: `isBankProvider` in `transfers.ts` currently tests a hardcoded two-entry set `new Set(["hapoalim", "leumi"])`, so `detectKind` misclassifies credits and CC payments for every other bank (mizrahi, discount, oneZero, etc.). The matcher's `directionMatch` and CC-payment leg depend on correct `bank` vs `card` typing, so we replace `BANK_PROVIDERS_SET` with a lookup over `BANK_PROVIDERS` (filter `kind === "bank"`) built once at module load. This is a prerequisite migration of logic, not new behavior, and it also makes the existing `detectKind` correct for all banks. +- New exports in `transfers.ts`: `matchesCreditCardPayment(description)` (thin wrapper over the existing private `matchesTransferPattern` / `CREDIT_CARD_PAYMENT_PATTERNS`) so the feature builder reuses the single regex source instead of duplicating it. `matchesInternalTransfer` is already exported and is reused for `keywordTransfer`. +- Reasons and confidence are emitted as data, persisted on the event, never logged (credentials and descriptions must never hit logs per project security rules). +- Reversibility: an event is a link plus a `kind` flip, so undo is "delete event, restore each member's prior `kind`." The pre-merge `kind` of each member is captured on the `event_members` row (schema section) so undo restores the exact prior state, including the case where a member was already `income`, not `expense`. The persisted feature vector explains the original decision. +- The exact-hash dedup (`computeDedupHash`, `dedup_sequence`, the `ON CONFLICT` pending->posted update) is untouched and runs first; the matcher operates strictly on already-deduped, persisted rows. The hash stays over the RAW `description` (normalization is fuzzy-layer only) so exact-dedup behavior is byte-stable across releases. A separate settlement-reconciliation pass (pending->posted, where name/amount may drift) is the natural home for the whole-shekel blocking pass and is cross-referenced from the dedup section rather than duplicated here. + +Migration note: all of the above is additive. The expression indexes (`idx_txn_block_amount`, `idx_txn_block_amount_shekel`) and any new columns (persisted `confidence`, `reasons`, per-member pre-merge `kind`) ship in a new migration. The repository's numbering is NOT a clean monotonic sequence (it already contains parallel pairs `020_excluded.sql` + `020_multiple_bank_credentials.sql` and `021_chat_sessions.sql` + `021_reclassify_credit_card_transfers.sql` from merged branches), so the new migration must take the next free index above the highest existing prefix (i.e. `022_event_matching.sql`) and the migration runner must apply files in a total order that tolerates the existing duplicate prefixes. No existing column changes type and no data is rewritten; the expression indexes are `CREATE INDEX IF NOT EXISTS` so re-running the migration is safe. + +--- + +## Database Schema Suggestions (SQLite) + +This section gives runnable DDL for the event-grouping layer described elsewhere in this document. It is one additive migration, `022_financial_events.sql`, plus a small TypeScript backfill. The existing runner (`src/server/db/migrate.ts`) picks the file up automatically: it sorts `*.sql` lexically, runs each unapplied file inside one `db.transaction()` with `PRAGMA foreign_keys = OFF`, then runs `PRAGMA foreign_key_check` and aborts the whole migration if any dangling reference remains. Two consequences shape everything below: + +- `022` sorts after the two existing `021_*` files (`021_chat_sessions.sql`, `021_reclassify_credit_card_transfers.sql`), so the number is free and ordering is correct. +- Because `foreign_key_check` runs at the end, the migration must not leave a single dangling FK. The DDL below only adds columns and tables and seeds rows that reference already-existing `workspaces`, so it passes. The pair-reconstruction backfill, which is the part most likely to create a bad reference, is deliberately moved out of the SQL and into idempotent TypeScript (see "Backfill"). + +No table is dropped, no column is removed. The `kind` column and the count-based dedup in `src/server/lib/dedup.ts` stay exactly as they are. The new layer sits beside `kind`, not on top of it. + +### Design decision: membership column on `transactions`, not a junction table + +The brief asks whether a transaction joins an event through an `event_transactions(event_id, transaction_id, role)` table (many events per txn) or through columns on `transactions` (`event_id`, `event_role`, `match_confidence`, at most one event per txn). Recommendation: columns on `transactions`, one event per transaction. + +Rationale grounded in the three linkage problems this design serves (the 1:1 internal-transfer pair, the credit-card bill-payment to N-purchases aggregate link, and the existing `kind`-flip): + +- Every real grouping in Spent is a partition, not an overlap. A bank debit is either the funding leg of one transfer or the bill payment for one card statement, never both at once. A card purchase belongs to exactly one billing batch. There is no case where one row legitimately participates in two distinct financial events, so the many-to-many flexibility of a junction table buys nothing and costs an extra join on the hottest analytics path. +- The reporting filter (below) must ask, on every spend query, "is this row the canonical row of its event, or a suppressed member?" With a column that is a single predicate on `t.event_id` plus one correlated lookup; with a junction table it is a join plus a `GROUP BY` just to prove a row has no membership. +- It mirrors what the market leaders ship. Monarch, Copilot, and Simplifi keep the classification on the leg itself (a category or a type), not in a separate pair object. YNAB stores cross-pointers (`transfer_account_id`, `transfer_transaction_id`) but only for the strict 1:1 transfer case, which `financial_events` plus `canonical_transaction_id` already expresses more generally. + +The cost is that re-membership (moving a row from event A to event B) is an `UPDATE` of the row rather than a delete-and-insert in a junction. That is fine and is actually easier to make atomic. If a future feature genuinely needs one row in many events (none is on the roadmap), the junction can be added later as another additive migration without disturbing this one. + +### Migration `022_financial_events.sql` + +```sql +-- 022_financial_events.sql +-- First-class grouping layer over transactions. Generalizes the kind='transfer' +-- flip (007, 008, 021_reclassify_credit_card_transfers) into auditable, +-- reviewable, reversible events. Backward compatible: the kind column is left +-- intact and kept in sync; the existing dedup (src/server/lib/dedup.ts) is +-- untouched. This file is DDL + seed only. No row-pairing logic runs here +-- (it would risk a dangling FK that foreign_key_check would reject); that lives +-- in src/server/db/backfill/022_events.ts. + +-- 1. The event: one logical real-world money movement made of N transaction rows. +CREATE TABLE financial_events ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL + REFERENCES workspaces(id) ON DELETE CASCADE, + + -- internal_transfer: A->B move between two owned accounts (1:1, two legs). + -- cc_bill_payment: one bank debit settling a card statement (1:N to the + -- underlying purchases). Today only the bank-side debit + -- exists in our data; the mirrored card credit is usually + -- NOT scraped, so this event commonly has a single leg. + -- atm_withdrawal: cash leaving an account (1:1, or 1:0 when cash untracked). + -- duplicate: pending<->posted or re-pull artifact kept as a link, not + -- collapsed (see "link, do not collapse" in research). + event_type TEXT NOT NULL + CHECK (event_type IN + ('internal_transfer','cc_bill_payment','atm_withdrawal','duplicate')), + + -- The one row that REPRESENTS this event in reporting, or NULL when the event + -- contributes nothing to spend. For an internal_transfer that is fully between + -- owned accounts it is NULL (neither leg is spend). For a cc_bill_payment it + -- is NULL: the bill itself is not spend, the underlying purchases are. + -- NOTE: ON DELETE SET NULL means deleting the canonical row makes the event + -- contribute zero to spend, which is the safe direction (never silently + -- re-counts), but the review UI must surface "canonical deleted" events. + canonical_transaction_id INTEGER + REFERENCES transactions(id) ON DELETE SET NULL, + + status TEXT NOT NULL DEFAULT 'open' + CHECK (status IN ('open','confirmed','dismissed')), + + -- Match probability in [0,1]. REAL is correct here: this is a score, not money. + confidence REAL NOT NULL DEFAULT 0 + CHECK (confidence >= 0 AND confidence <= 1), + + -- Human-readable, machine-parseable reasons, e.g. + -- {"amount_delta":0.0,"day_gap":1,"keyword":"\u05d4\u05e2\u05d1\u05e8\u05d4", + -- "desc_sim":0.94,"comparator":"jaro_winkler"} + -- JSON1 is compiled into better-sqlite3's bundled SQLite, so json_* work. + match_reasons TEXT + CHECK (match_reasons IS NULL OR json_valid(match_reasons)), + + -- Idempotency key: deterministic over the event's defining fields so a re-run + -- of detection over the same (overlapping) sync window does not create a + -- second event. Same spirit as dedup_hash. For a transfer, hash the two leg + -- dedup_hashes in sorted order so leg order does not matter: + -- sha256("internal_transfer|"+min(hashA,hashB)+"|"+max(hashA,hashB)). + -- For a cc_bill_payment, hash the bank debit's dedup_hash: + -- sha256("cc_bill_payment|"+debitDedupHash). + event_key TEXT NOT NULL, + + source TEXT NOT NULL DEFAULT 'auto' + CHECK (source IN ('auto','user')), + + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + + UNIQUE (workspace_id, event_key) +); + +CREATE INDEX idx_events_workspace_status + ON financial_events(workspace_id, status); +CREATE INDEX idx_events_workspace_type + ON financial_events(workspace_id, event_type); +-- Review queue: open, auto-detected, low confidence first. +CREATE INDEX idx_events_review + ON financial_events(workspace_id, status, confidence) + WHERE status = 'open' AND source = 'auto'; + +-- 2. Membership lives on transactions (one event per row, see rationale above). +-- SQLite forbids ADD COLUMN with a REFERENCES clause, so these added columns +-- are plain INTEGER/TEXT. The FK relationship is enforced in the app layer; +-- ON DELETE behavior for event deletion is handled explicitly in the +-- delete/unmerge code path (set event_id = NULL on members first). +ALTER TABLE transactions ADD COLUMN event_id INTEGER; + +ALTER TABLE transactions ADD COLUMN event_role TEXT + CHECK (event_role IS NULL OR event_role IN + ('transfer_out','transfer_in','bill_payment','bill_item','atm','duplicate_of')); + +-- Per-leg score, distinct from the event-level confidence: lets one weak leg be +-- flagged while the event as a whole is confirmed. +ALTER TABLE transactions ADD COLUMN match_confidence REAL + CHECK (match_confidence IS NULL OR (match_confidence >= 0 AND match_confidence <= 1)); + +-- The single most important index for analytics: every spend query now also +-- considers event membership. Partial index keeps it tiny (on a fresh install +-- almost no rows have an event). +CREATE INDEX idx_transactions_event + ON transactions(event_id) + WHERE event_id IS NOT NULL; + +-- 3. The BLOCKING index. This is what keeps detection out of O(n^2). Candidate +-- generation buckets rows by currency + a rounded absolute amount + a day +-- key, so the matcher only compares rows inside one small bucket. abs/round/ +-- cast/substr are deterministic, so the expression index is valid. +-- +-- CAUTION (read before relying on a single-bucket scan): rounding/truncation +-- is a HARD bucket boundary. Two amounts 0.01 apart can straddle it +-- (199.50 -> 200, 199.49 -> 199 under round; or different days under the +-- date key). The matcher MUST therefore probe the row's own bucket AND the +-- two neighbor amount buckets (b-1, b, b+1) AND every day key inside the +-- rule's day_window, not just the exact bucket. The index makes each of +-- those point lookups O(log n); the exact epsilon test runs afterward on the +-- handful of returned rows. Bucketing on truncated-shekel (CAST AS INTEGER of +-- the absolute amount) gives stable, currency-independent buckets. +CREATE INDEX idx_transactions_block + ON transactions( + workspace_id, + charged_currency, + CAST(ABS(charged_amount) AS INTEGER), + substr(date, 1, 10) + ); + +-- 4. User-tunable rule params (open-source priority: retune without a code edit). +-- Detection reads these with a hard-coded fallback (see note after the seed) +-- so a workspace created AFTER this migration still has correct defaults. +CREATE TABLE match_rules ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL + REFERENCES workspaces(id) ON DELETE CASCADE, + event_type TEXT NOT NULL + CHECK (event_type IN + ('internal_transfer','cc_bill_payment','atm_withdrawal','duplicate')), + -- Max allowed |abs(amtA) - abs(amtB)|. Matches DEFAULT_EPSILON = 0.01 in + -- internal-transfers.ts; surfaced here so it is editable. + epsilon REAL NOT NULL DEFAULT 0.01, + -- Max gap in days. Matches DEFAULT_DAY_WINDOW = 2. Tunable per event type; + -- cc_bill_payment legitimately needs a wider window than a same-bank transfer. + day_window INTEGER NOT NULL DEFAULT 2, + -- Auto-confirm at or above auto_score; suggest in [min_score, auto_score); + -- ignore below min_score. + min_score REAL NOT NULL DEFAULT 0.5, + auto_score REAL NOT NULL DEFAULT 0.95, + -- Require a description keyword on at least one side (today's behavior for + -- internal transfers). Turn off to rely on amount + date + opposite sign. + require_keyword INTEGER NOT NULL DEFAULT 1, + enabled INTEGER NOT NULL DEFAULT 1, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')), + UNIQUE (workspace_id, event_type) +); + +-- Seed defaults per EXISTING workspace so behavior is unchanged on upgrade. +INSERT INTO match_rules (workspace_id, event_type, epsilon, day_window, require_keyword) +SELECT w.id, 'internal_transfer', 0.01, 2, 1 FROM workspaces w; +INSERT INTO match_rules (workspace_id, event_type, epsilon, day_window, require_keyword) +SELECT w.id, 'cc_bill_payment', 0.01, 5, 0 FROM workspaces w; +INSERT INTO match_rules (workspace_id, event_type, epsilon, day_window, require_keyword) +SELECT w.id, 'atm_withdrawal', 0.01, 2, 1 FROM workspaces w; +INSERT INTO match_rules (workspace_id, event_type, epsilon, day_window, require_keyword) +SELECT w.id, 'duplicate', 0.00, 10, 0 FROM workspaces w; + +-- 5. Audit table: candidate pairs the scorer considered, kept so a match (and a +-- near-miss) can be explained and reversed. Prune on a schedule. +CREATE TABLE match_candidates ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL + REFERENCES workspaces(id) ON DELETE CASCADE, + event_type TEXT NOT NULL + CHECK (event_type IN + ('internal_transfer','cc_bill_payment','atm_withdrawal','duplicate')), + txn_a_id INTEGER NOT NULL REFERENCES transactions(id) ON DELETE CASCADE, + txn_b_id INTEGER REFERENCES transactions(id) ON DELETE CASCADE, + score REAL NOT NULL CHECK (score >= 0 AND score <= 1), + -- Same shape as financial_events.match_reasons. + reasons TEXT CHECK (reasons IS NULL OR json_valid(reasons)), + -- Outcome the scorer assigned: above auto_score, in the review band, or below. + decision TEXT NOT NULL + CHECK (decision IN ('auto','suggest','reject')), + -- Set once promoted to a real event; lets us not re-suggest the same pair. + event_id INTEGER REFERENCES financial_events(id) ON DELETE SET NULL, + created_at TEXT NOT NULL DEFAULT (datetime('now')), + -- Idempotency: re-running detection over the same window must not pile up + -- duplicate candidate rows. Order the pair deterministically (min id, max id) + -- before insert so the unique key is stable. + UNIQUE (workspace_id, event_type, txn_a_id, txn_b_id) +); +CREATE INDEX idx_candidates_review + ON match_candidates(workspace_id, decision, score) + WHERE event_id IS NULL; +``` + +Two notes on the schema: + +- **`event_id` is not a declared FK.** SQLite cannot `ALTER TABLE ... ADD COLUMN ... REFERENCES`. That is why migration 013 recreates whole tables to add FK columns. Recreating `transactions` again (tens of thousands of rows, plus every index) just to make `event_id` a declared FK is not worth it for a self-hosted single file; the relationship is enforced in the app layer, and the unmerge/delete path nulls `event_id` on members before deleting an event so no orphan points at a missing event. If a declared FK is wanted later, fold it into a future recreate. +- **`match_rules` defaults must also exist for workspaces created after this migration.** The seed only covers workspaces that exist at upgrade time. New workspaces will have no `match_rules` rows, so detection must read a rule with a fallback (`SELECT ... ` then `?? HARD_CODED_DEFAULT`), or workspace creation must insert the four default rows. Pick the read-with-fallback path; it is the one that cannot silently disable detection. + +A note on the `event_key` UNIQUE constraint: it is the events-layer analogue of `UNIQUE(workspace_id, dedup_hash, dedup_sequence)` on `transactions`. Detection re-runs over an overlapping window (very common: Israeli scrapers re-pull recent history) must be idempotent. Compute `event_key` from the sorted leg `dedup_hash`es (or from the bank-debit `dedup_hash` for a `cc_bill_payment`) and let `INSERT ... ON CONFLICT(workspace_id, event_key) DO UPDATE SET confidence = excluded.confidence, match_reasons = excluded.match_reasons, updated_at = datetime('now')` refresh in place instead of spawning a duplicate, exactly as `insertTransactions` already does for rows. Do NOT key on `transactions.id`: ids are not stable across a re-pull that re-inserts a row, but `dedup_hash` is. + +### Backfill from today's `kind`-flipping + +Migration `021_reclassify_credit_card_transfers.sql` flipped bank-side credit-card payments to `kind='transfer'` (only for `provider IN ('hapoalim','leumi')`, matching the keyword set in `transfers.ts`), and `findInternalTransferPairs` flips matched 1:1 pairs at sync time. None of that grouped anything. The backfill turns those flags into events without changing any `kind` value, so reporting output is byte-for-byte identical the instant it finishes; the events just make the existing exclusions auditable. + +Do the backfill in TypeScript, not SQL, for two concrete reasons: + +1. **Correctness of the matcher.** `findInternalTransferPairs` (`src/server/lib/internal-transfers.ts`) filters out `r.kind === "transfer"` on its very first line (`const eligible = rows.filter((r) => r.kind !== "transfer" ...)`). Feeding it the already-flipped `kind='transfer'` rows yields zero pairs. The backfill must pass each leg with a non-transfer `kind` (its pre-flip sign-based kind) so the matcher runs, then write the events for the pairs it returns. A pure-SQL correlated-subquery reconstruction is both fragile (a `MIN(c2.id)` tiebreak can attach the wrong credit when several credits share an amount) and would have to re-implement the keyword and account-difference logic that already lives, tested, in `internal-transfers.ts` and `transfers.ts`. One source of truth wins. +2. **The `foreign_key_check` gate.** A SQL backfill that sets `event_id` while the migration's FK check runs at the end is one typo away from a rejected migration. Keeping pairing out of the SQL keeps `022` trivially safe. + +```typescript +// src/server/db/backfill/022_events.ts (sketch) +import "server-only"; +import crypto from "node:crypto"; +import { findInternalTransferPairs } from "@/server/lib/internal-transfers"; + +// Load the already-flipped transfer legs, but present each with its SIGN-BASED +// kind so the matcher (which discards kind='transfer') will actually pair them. +const legs = db + .prepare( + `SELECT id, credential_id AS credentialId, account_number AS accountNumber, + date, charged_amount AS chargedAmount, + charged_currency AS chargedCurrency, description, dedup_hash AS dedupHash + FROM transactions + WHERE workspace_id = ? AND kind = 'transfer' AND event_id IS NULL`, + ) + .all(workspaceId) as Array<{ + id: number; + credentialId: number | null; + accountNumber: string; + date: string; + chargedAmount: number; + chargedCurrency: string | null; + description: string; + dedupHash: string; +}>; + +const byId = new Map(legs.map((l) => [l.id, l])); +const candidates = legs.map((l) => ({ + ...l, + kind: l.chargedAmount < 0 ? ("expense" as const) : ("income" as const), +})); + +const eventKey = (a: string, b: string) => + crypto + .createHash("sha256") + .update(`internal_transfer|${[a, b].sort().join("|")}`) + .digest("hex"); + +const insertEvent = db.prepare( + `INSERT INTO financial_events + (workspace_id, event_type, status, confidence, source, event_key, match_reasons) + VALUES (?, 'internal_transfer', 'confirmed', 1.0, 'auto', ?, json_object('backfill', 1)) + ON CONFLICT(workspace_id, event_key) DO UPDATE SET updated_at = datetime('now') + RETURNING id`, +); +const attachLeg = db.prepare( + `UPDATE transactions SET event_id = ?, event_role = ?, match_confidence = 1.0 + WHERE id = ? AND event_id IS NULL`, +); + +const run = db.transaction(() => { + for (const { debitId, creditId } of findInternalTransferPairs(candidates)) { + const debit = byId.get(debitId); + const credit = byId.get(creditId); + if (!debit || !credit) continue; + const { id } = insertEvent.get(workspaceId, eventKey(debit.dedupHash, credit.dedupHash)) as { + id: number; + }; + attachLeg.run(id, "transfer_out", debitId); + attachLeg.run(id, "transfer_in", creditId); + } +}); +run(); +``` + +The backfill is idempotent: `event_id IS NULL` in the source query and on both `attachLeg` writes means a second run does no extra work, and the `ON CONFLICT(event_key)` makes re-creating an event a no-op refresh. + +The bank-side `cc_bill_payment` rows from `021_reclassify_credit_card_transfers.sql` are different: they were flipped to `transfer` but never linked to the card purchases they pay, and the mirrored card credit is usually not in our data at all (we scrape the card's per-purchase rows, not a statement-level credit). Back-fill each such bank debit as a single-leg `cc_bill_payment` event: + +```sql +-- Wrap each existing bank-side CC-payment transfer in a cc_bill_payment event. +-- canonical_transaction_id stays NULL: the bill is not spend, the purchases are. +INSERT INTO financial_events + (workspace_id, event_type, status, confidence, source, event_key, match_reasons) +SELECT t.workspace_id, 'cc_bill_payment', 'confirmed', 1.0, 'auto', + 'cc_bill_payment|' || t.dedup_hash, + json_object('backfill', 1) +FROM transactions t +WHERE t.kind = 'transfer' + AND t.provider IN ('hapoalim','leumi') + AND t.event_id IS NULL +ON CONFLICT (workspace_id, event_key) DO NOTHING; + +UPDATE transactions +SET event_id = (SELECT fe.id FROM financial_events fe + WHERE fe.workspace_id = transactions.workspace_id + AND fe.event_key = 'cc_bill_payment|' || transactions.dedup_hash), + event_role = 'bill_payment' +WHERE kind = 'transfer' + AND provider IN ('hapoalim','leumi') + AND event_id IS NULL + AND EXISTS (SELECT 1 FROM financial_events fe + WHERE fe.workspace_id = transactions.workspace_id + AND fe.event_key = 'cc_bill_payment|' || transactions.dedup_hash); +``` + +This SQL is safe to keep inline only if run AFTER the migration commits (so it is not under the `foreign_key_check` umbrella), or run it from the same `022_events.ts` script. Linking the N underlying purchases to the bill payment (the 1:N aggregate match) is described elsewhere and can be filled in later by a detection pass without changing any total, because the bill payment is already excluded and the purchases are already counted. + +One caveat the totals depend on: a bank debit and the bank-side payment row can disagree because Israeli statements sometimes split a single card bill into more than one debit, and `021` flips each matching row independently. The backfill wraps each flipped row in its own `cc_bill_payment` event, which preserves today's exclusion exactly. Merging several debits into one bill event is a later refinement, not a day-one requirement. + +### How analytics queries change + +Every spend query in `src/server/db/queries/transactions.ts` currently repeats `status = 'completed' AND kind = 'expense'` (in `getMonthlySummary`, `getTopMerchants`, `getCategoryBreakdown`, `getPeriodTotal`, `getTransactionsSummary`, and roughly ten others). That `kind='expense'` predicate already excludes transfers, so those queries keep working unchanged on day one. The events layer adds exactly one more exclusion: a transaction that is a non-canonical member of an event must not be counted, even if its `kind` is still `expense`. To express "count this row in spend exactly once," introduce a view. + +```sql +-- Part of migration 022. The canonical "what counts as spend" definition. +-- A row is spendable when it is a completed expense that is EITHER not part of +-- any event, OR is the canonical row of its event. If canonical_transaction_id +-- is NULL the subquery is NULL and t.id = NULL is never true, so a member of a +-- canonical-less event (every internal_transfer and cc_bill_payment today) is +-- correctly excluded. This sums each real expense once; transfers, bill +-- payments, and suppressed duplicates fall out. +CREATE VIEW spendable_transactions AS +SELECT t.* +FROM transactions t +WHERE t.status = 'completed' + AND t.kind = 'expense' + AND ( + t.event_id IS NULL + OR t.id = (SELECT fe.canonical_transaction_id + FROM financial_events fe WHERE fe.id = t.event_id) + ); +``` + +The query-layer changes are then mechanical: swap `FROM transactions` for `FROM spendable_transactions` and delete the now-redundant `status = 'completed' AND kind = 'expense'` clauses (keep `workspace_id`, `date`, and any `credential_id` filters, since the view is workspace-agnostic and date-agnostic). For example `getPeriodTotal` becomes: + +```sql +SELECT COALESCE(SUM(ABS(charged_amount)), 0) AS total +FROM spendable_transactions +WHERE workspace_id = ? AND date >= ? AND date <= ?; +``` + +Three details that the migration is responsible for getting right: + +- **Do not let the view change any total on day one.** Because the backfill leaves every event with `canonical_transaction_id = NULL`, every currently-excluded row stays excluded and every currently-counted expense (which has no `event_id`) stays counted. The view's output equals the old predicate's output exactly until a future canonical-picking pass runs. That is the property that makes this migration shippable: zero reporting drift on upgrade. +- **Income and the largest-income picker keep the raw table.** `getTransactionsSummary` sums `kind='income'` directly; income's only event interaction is transfers, which `kind` already excludes. If income should later be deduped against events too, add a parallel `countable_income` view with the same canonical rule rather than parameterizing one view, so the planner can use the `kind` index and `idx_transactions_event` cleanly for each. +- **`is_excluded` and excluded merchants are untouched.** The current analytics SUMs do not filter on `is_excluded` (exclusion is applied upstream at categorization time, not netted out of these SUMs), so the view must not add an `is_excluded` predicate either; adding one would silently change historical totals. The view's job is only event-membership, nothing else. + +A view in SQLite is a stored SELECT, so the planner still uses `idx_transactions_date`, `idx_transactions_category`, and the new partial `idx_transactions_event`. The correlated `canonical_transaction_id` subquery runs only for the small set of rows that have an `event_id` (the partial index keeps that set tiny), so the common all-rows-have-no-event path that dominates a fresh install does not regress. + +`queryTransactions` (the user-facing list, not analytics) should NOT switch to the view: users want to SEE transfer legs and bill payments in the list with an event badge, the way Copilot shows `[T]` and Simplifi shows "Go to other side." Keep that on the raw table and `LEFT JOIN` the event for the badge: + +```sql +LEFT JOIN financial_events fe ON t.event_id = fe.id +-- expose fe.event_type, fe.status, t.event_role to the row mapper +``` + +### SQLite specifics + +- **JSON1 for `match_reasons` / `reasons`.** better-sqlite3 bundles SQLite with JSON1 compiled in, so `json_valid()`, `json_object()`, and `json_extract()` work with no extension load. Store reasons as TEXT with a `CHECK(json_valid(...))` guard so a malformed write fails loudly, and read them in the review UI with `json_extract(match_reasons, '$.day_gap')`. Do not normalize reasons into columns; they are explanatory payload, read only on the review screen, and their shape evolves per event type. +- **Partial indexes.** Used deliberately for `idx_transactions_event` (`WHERE event_id IS NOT NULL`), `idx_events_review`, and `idx_candidates_review`. On a typical install almost no rows belong to an event, so each partial index is a fraction of a full one and the hot all-expense scan is untouched. SQLite has supported partial indexes since 3.8.0, far below the bundled version. +- **Expression / blocking index, and the bucket-boundary trap.** `idx_transactions_block` indexes `(workspace_id, charged_currency, CAST(ABS(charged_amount) AS INTEGER), substr(date,1,10))`. The expressions are deterministic (`abs`, `cast`, `substr`), so the index is valid. This index is the single thing that keeps detection off the O(n^2) path: each row is matched only against rows in its bucket. The non-obvious correctness requirement is that bucketing is a hard boundary, so the matcher must NOT scan only the exact bucket. For each row it must probe amount buckets `b-1, b, b+1` and every `substr(date,1,10)` key within the rule's `day_window`, then apply the precise `ABS(ABS(a) - ABS(b)) <= epsilon` and exact day-gap tests on the small returned set. Probing three amount buckets times a few day keys is still a handful of O(log n) point lookups per row, not a scan. For the pending-vs-posted case where amounts can drift more than a shekel, widen the amount-bucket probe in application code rather than coarsening the index. +- **Never compare two amounts with `=`.** `charged_amount` and `original_amount` are REAL in this codebase, inherited as-is; the events layer does not make that worse. Always use the `epsilon` tolerance from `match_rules` (`ABS(ABS(a) - ABS(b)) <= epsilon`), exactly as `findInternalTransferPairs` already does with `DEFAULT_EPSILON = 0.01`. The blocking index's integer-shekel bucket is only for candidate generation; the precise epsilon test runs afterward, so bucketing never causes a false match, only (with the neighbor-bucket probe) avoids a missed one. `confidence`, `match_confidence`, `score`, `min_score`, and `auto_score` are scores, not money, so REAL is correct and intended for them; their `CHECK (… BETWEEN 0 AND 1)` guards catch a bad write. +- **Sign convention is load-bearing in detection.** `charged_amount` is NEGATIVE for debits/expenses and POSITIVE for credits/income. A transfer pair is one negative leg and one positive leg with equal absolute value; the matcher must compare `ABS()` and require opposite raw signs, which `findInternalTransferPairs` already does (`debits = ... < 0`, `credits = ... > 0`). When choosing a `canonical_transaction_id` for an event that does cross the budget boundary, pick the leg whose sign matches the direction being counted (the expense leg for spend), never the absolute-larger leg. +- **Hebrew / RTL in keys and matching.** `event_key` and `dedup_hash` are SHA-256 hex, so RTL text in descriptions never reaches a key; do not put raw descriptions in `event_key`. Description-based matching stays keyword-driven via the tested regexes in `transfers.ts` (which already handle the Hebrew CC and `\u05d4\u05e2\u05d1\u05e8\u05d4`/transfer forms and the final-letter variants). Note the current keyword set fires only for `provider IN ('hapoalim','leumi')` (`BANK_PROVIDERS_SET`), so cc_bill_payment detection is presently scoped to those two banks; widening it to the other banks in `BANK_PROVIDERS` is a `transfers.ts` change, not a schema change, and `require_keyword` plus `match_rules` let a self-hoster opt other providers in. +- **FTS5 for fuzzy description matching (optional, not in `022`).** Hebrew RTL descriptions defeat Soundex/Metaphone (English-pronunciation bound) and are awkward for edit distance. If fuzzy scoring becomes necessary, the lightweight option is an FTS5 contentless/external-content table over a normalized description (strip RTL marks and niqqud, unify final-letter forms) used purely as a candidate-generation blocker, then score the short list with Damerau-Levenshtein or token-set Jaccard in TypeScript. better-sqlite3 ships FTS5, but keep it out of the core migration: amount-plus-date blocking does the heavy lifting and description similarity is only one feature in the score, per record-linkage guidance to use exact comparators on high-signal fields and reserve fuzzy logic for free text. + +### Why this generalizes the current code rather than replacing it + +`kind` stays authoritative for the coarse "exclude from spend/income" decision, so nothing in the existing query layer breaks on day one, and the new view reproduces today's totals exactly until a canonical-picking pass runs. `financial_events` adds the three things the current flags cannot express: the grouping (which rows form one event, via `event_id`/`event_role`), the provenance (`source`, `confidence`, `match_reasons`, `match_candidates`), and the workflow (`status` open/confirmed/dismissed plus a reversible `event_id` that can be nulled to unmerge). `detectKind` and `findInternalTransferPairs` keep running at sync time; their output becomes the seed for events (high confidence, auto-confirmed) instead of a terminal flag, and `match_rules` lets a self-hosting user retune `epsilon`, `day_window`, `require_keyword`, and the auto-versus-suggest thresholds without editing the constants in `internal-transfers.ts`. + +--- + +## Edge Cases and Failure Scenarios + +This section enumerates the concrete ways the event-grouping layer can go wrong, the failure mode of each, and an implementation-ready mitigation. The cardinal rule, applied everywhere, is the **no-auto-hide bias**: when the engine is not confident, it must NOT flip a row to `kind='transfer'` (the ONLY flag that today removes a row from both spend and income analytics). It must instead leave the row's `kind` untouched and write a *suggestion* into the new `financial_events` / `event_links` tables with `status='suggested'` and `needs_review=1`, so a real expense can never disappear without a human confirming it. + +### Ground-truth correction: what actually excludes a row from analytics + +Every spend query in `transactions.ts` (`getMonthlySummary`, `getTopMerchants`, `getCategoryBreakdown`, `getCategorySpendInRange`, `getTopMerchantPerCategory`, `getCategorySpendByDay`, `getTopMerchantsForCategory`, `getPeriodTotal`, `getPeriodCount`, the expense/income aggregates in `getTransactionsSummary`) filters on `kind` plus `status='completed'`. **None of them filter on `is_excluded`.** The `is_excluded` column (migration `020_excluded.sql`) is driven only by the `excluded_merchants` "hide this merchant" feature and the per-row "Hide / Show" toggle; it is surfaced in `mapTransactionRow` for the UI but it does NOT subtract a row from any aggregate today. Therefore: + +- The single load-bearing exclusion mechanism for this design is `kind='transfer'`. The earlier draft's repeated phrasing "excluded by `kind`/`is_excluded`" is wrong and is corrected throughout to "excluded by `kind='transfer'`." +- This design does NOT repurpose `is_excluded`. If a future change wants `is_excluded` to also drop rows from spend, that is a separate, explicit migration that must add `AND is_excluded = 0` to all ten queries above; until then, setting `is_excluded` alone hides nothing from totals and is unsafe to rely on for double-count prevention. + +The new `financial_events.status` and `event_links` are audit/UX state; the actual analytics effect of any decision is realized solely by writing `kind` (and, for AUTO, clearing `needs_review`). This keeps the design backward compatible with the existing `detectKind` + `markTransfersByIds` orchestrator passes, which are the AUTO tier of the new engine. + +### Provider-set reality (constrains every cc-payment claim below) + +`isBankProvider` in `transfers.ts` returns true for EXACTLY `hapoalim` and `leumi` (`BANK_PROVIDERS_SET`). `detectKind`'s cc-payment flip and its `chargedAmount > 0 -> income` flip therefore only fire for those two providers; the reclassify migration `021` likewise hardcodes `provider IN ('hapoalim','leumi')`. Consequences the design must own: + +- A credit-card bill debited from Mizrahi, Discount, or One Zero checking is currently classified `expense` and is NOT auto-flipped, so it can double-count against synced card purchases. Adding banks to `BANK_PROVIDERS_SET` (and to `021`'s provider list, or better, replacing `021`'s inline LIKE list with a re-run of `detectKind`) is a prerequisite for the cc-payment AUTO path to be correct for those banks. Until a provider is in the bank set, its cc-payment rows fall to the SUGGEST path, never AUTO. +- All cc-payment claims below are scoped to "provider is in the recognized bank set." For unrecognized-bank providers, the engine emits a SUGGEST with `needs_review=1` rather than silently doing nothing or silently hiding. + +### Confidence tiers (referenced throughout) + +Every candidate group carries a `confidence` in `[0,1]` (Fellegi-Sunter three-zone policy) and resolves to one action: + +| Tier | Score | Action | Schema effect | +| --- | --- | --- | --- | +| AUTO | `>= 0.97` | Group + flip all legs to transfer | `financial_events.status='confirmed'`, legs `kind='transfer'`, `needs_review=0` | +| SUGGEST | `0.80 - 0.969` | Group as proposal, legs keep `kind` | `status='suggested'`, legs unchanged `kind`, `needs_review=1` | +| IGNORE | `< 0.80` | Do nothing | no event row | + +AUTO is reachable only by (a) the cc-payment keyword path for a recognized bank provider, gated on a tracked card (case 5), and (b) exact same-currency, same-amount, opposite-sign internal-transfer keyword pairs (case 8). Pure fuzzy or amount-only matches cap in SUGGEST. This preserves the current `detectKind` flip as AUTO and makes everything new suggest-first. + +### 1. Partial, over, and minimum credit-card payments + +**Failure mode.** The bank-side debit (one row, `-1,200`) does not equal the sum of the card-account purchases (`-3,800` of line items). A 1:1 amount matcher never pairs them; a matcher that nets payment against purchases would erase `2,600` of real spend on an under-payment, or invent negative spend on an over-payment. + +**Mitigation.** Never net the payment against the purchases. Spend is counted once at purchase time; the bill payment is pure money movement. The `type='cc_payment'` event links only money-movement legs and does NOT require amounts to reconcile: + +- `role='cc_payment_bank_leg'`: the checking debit (recognized bank provider). Flipped to `kind='transfer'`. +- `role='cc_payment_card_leg'`: the credit posted on the card account, if that account is synced. Flipped to `kind='transfer'`. +- The underlying card purchases are NOT linked; they stay `kind='expense'` and remain the single source of spend. + +Minimum, full, and over-payment are identical to the engine: it transfers the money-movement legs and leaves purchases untouched. The revolving balance is a liability concern, out of scope for spend analytics. No amount tolerance exists on the bill relationship because we deliberately do not assert `payment == sum(purchases)`. + +### 2. One bank payment covering multiple cards; one card paid from multiple banks + +**Failure mode.** A `-5,000` bank debit settles three cards. The keyword path flips the one bank row fine, but a 1:1 card-side pairing would pick one card and leave two card credits as un-flipped `income` (double-counted inflow). Mirror case: one card paid from two banks yields two bank debits each looking like a partial payment. + +**Mitigation.** The model is N:M by construction (a join table, not a paired FK). The bank leg joins one `cc_payment` event; each card-side credit is a separate `cc_payment_card_leg` in the *same* event. Detection is per-leg, not per-pair: each card-side "payment received" credit is flipped to transfer independently by a per-provider rule set, so an unmatched card credit never counts as income. + +Note the current code cannot do this yet: `detectKind`'s income flip only runs for `hapoalim`/`leumi`, and card providers (isracard, cal, max, amex) get `kind='expense'` regardless of sign, so a `+5,000` "payment received" credit on a card account is currently mislabeled `expense` with a positive amount. Because spend queries `SUM(ABS(charged_amount))` over `kind='expense'`, that positive credit is wrongly summed as `5,000` of spend. The fix is a card-side "payment received" rule that flips these specific credits to `kind='transfer'`: + +```typescript +// Per-provider card-side "payment received" patterns (Hebrew + English). +// Examples seen on card statements: "תשלום שהתקבל", "זיכוי", "PAYMENT RECEIVED", "PAYMENT - THANK YOU" +const CARD_PAYMENT_RECEIVED_PATTERNS: readonly RegExp[] = [ + /תשלום\s*שהתקבל/i, + /פירעון/i, + /\bPAYMENT\s+RECEIVED\b/i, + /\bPAYMENT\b.*\bTHANK\s*YOU\b/i, +]; +``` + +Only credits (`charged_amount > 0`) on a card credential whose description matches are flipped; a positive card row that does NOT match is a refund (case 10), not a payment. The grouping into one event is audit/UX only; an incomplete grouping degrades to "each leg still correctly flipped," not "money reappears." + +### 3. Cross-currency (FX) and rounding/agorot + +**Failure mode.** A `-100` USD transfer lands as `+372.40` ILS (rate plus fee). `findInternalTransferPairs` requires `sameCurrency` AND `|amtA-amtB| <= 0.01`, so it correctly refuses to pair these, but a genuine FX self-transfer then stays counted on both sides. Separately, agorot rounding (`-99.99` vs `+100.00`) exceeds `epsilon=0.01`. + +**Mitigation.** + +- **Same-currency rounding:** keep `DEFAULT_EPSILON = 0.01`. This absorbs agorot. Do not widen it for same-currency: `0.01` ILS is the smallest real unit and widening invites case-8 false positives. +- **Cross-currency:** a separate, SUGGEST-only FX matcher that is self-disabling when no rate source is configured (this is a local-only app; we do not require a network call). It compares `original_amount`/`original_currency` of one leg against `charged_amount`/`charged_currency` of the other: + +```typescript +const FX_FEE_TOLERANCE = 0.03; // 3% absorbs spread + conversion fee + +// rate: mid-market units of b.currency per unit of a.originalCurrency, or null +// if no rate source is configured. Null -> matcher does not run (no-auto-hide). +function isFxTransferCandidate(a: TransferCandidate, b: TransferCandidate, rate: number | null): boolean { + if (rate == null || !Number.isFinite(rate) || rate <= 0) return false; + const expected = Math.abs(a.originalAmount) * rate; + if (expected <= 0) return false; + const actual = Math.abs(b.chargedAmount); + return Math.abs(actual - expected) / expected <= FX_FEE_TOLERANCE; +} +``` + +An FX match can never reach AUTO; it caps at `0.90` (SUGGEST). The user confirms. Note `TransferCandidate` today only carries `chargedAmount`/`chargedCurrency`; the candidate query (`getInternalTransferCandidates`) must be widened to also select `original_amount`/`original_currency` for this path. Store the implied rate and tolerance in `match_reasons` so the suggestion is explainable. + +### 4. Statement-cycle timing and history-window cutoffs + +**Failure mode.** Purchases happen in cycle N; the bill debits in month N+1, 30 to 45 days after the earliest purchase, far beyond the 2-day `DEFAULT_DAY_WINDOW`. With Yahav's 6-month limit (and configurable `months_to_sync`, default 3), the bank debit can be inside the sync window while the matching card credit or purchases fall outside it, or vice versa. `getInternalTransferCandidates(workspaceId, from)` filters `date >= from`, so a window-bounded query misses the relationship. + +**Mitigation.** + +- The cc-payment bank leg is flipped by the **keyword path**, which is window-independent: it runs at insert in `detectKind` and in migration `021`. Cycle lag never reintroduces double counting for the bank leg; it is flipped the moment it is seen, whether or not the card side has synced. +- For non-CC internal transfers, widen the candidate window asymmetrically (transfers usually settle a few days later, not earlier). Make it tunable in `workspace_settings` (default conservative): + +```typescript +const TRANSFER_LOOKBACK_DAYS = 3; +const TRANSFER_LOOKAHEAD_DAYS = 5; +// candidate query lower bound becomes date(from, '-3 days'); upper bound is open. +``` + +- For the history-window edge (one leg outside the synced range), the engine treats a single observed leg as case 5: it does not auto-hide. A bank debit whose card account is unsynced stays flipped via keyword only if a card is tracked (case 5); a transfer leg whose counterpart is outside the window stays counted, un-grouped. The user sees one real movement, never a silently-vanished half-transfer. + +### 5. Only one leg synced (the most dangerous case: hiding real spend) + +**Failure mode.** The user connects their bank but not their credit card. The bank shows a `-1,200` "ויזה" debit. Today `detectKind` flips it to `transfer` unconditionally for `hapoalim`/`leumi`, so a user who does NOT track that card sees `1,200` of real spending vanish, with no card-side purchases to replace it. + +**Mitigation.** This is where Spent must diverge from the current blind keyword flip. The cc-payment flip is only *safe* when the card's purchases exist somewhere in the workspace. Gate the AUTO flip on "is any credit-card credential connected in this workspace?": + +```typescript +function shouldAutoFlipCcPayment(workspaceId: number): boolean { + // AUTO only when the workspace tracks at least one card provider, so the + // purchases that replace this payment as spend actually exist. + return countCardCredentials(workspaceId) > 0; +} +``` + +- Card IS tracked: keyword cc-payment legs go AUTO (`kind='transfer'`, `needs_review=0`). +- NO card tracked: the bank "ויזה" debit stays `kind='expense'`, tagged `needs_review=1`, with a suggestion "Looks like a credit-card bill. Connect that card to avoid double counting, or mark this as a transfer." The money stays visible. + +This is a behavioral change to today's `detectKind`, which currently flips unconditionally. Because `detectKind` runs at insert time and has no DB access, the gate cannot live inside `detectKind`; it must run as a post-insert pass in the orchestrator (alongside `findInternalTransferPairs`) that flips cc-payment expense rows to transfer only when `shouldAutoFlipCcPayment` is true, and otherwise sets `needs_review=1`. The migration `021` reclassify must be similarly gated, or it will retroactively hide bills for card-less workspaces. + +For ordinary internal transfers, a lone leg never pairs (the matcher needs two opposite-sign rows in different accounts), so it simply stays counted, which is the safe default. + +### 6. Re-sync re-matching, and pending->posted with changed amounts + +**Failure mode.** A pending row and its posted version may differ in name and amount (tip added, auth hold released). `dedup_hash` includes `description` and `originalAmount`, so a posted row with a changed amount gets a *different* hash and inserts as a NEW row; the `ON CONFLICT(... ) DO UPDATE ... WHEN status='pending'` upgrade never fires. That double-counts the charge and orphans any event link pointing at the now-stale pending row. + +**Mitigation.** + +- A **settlement-reconciliation pass** runs after insert and before transfer detection. It retires a pending row when a posted row arrives for the same logical charge within a small window, tolerating drift on noisy fields and blocking on high-signal ones: + +```sql +SELECT p.id AS pending_id, q.id AS posted_id +FROM transactions p +JOIN transactions q + ON q.workspace_id = p.workspace_id + AND q.account_number = p.account_number + AND q.status = 'completed' AND p.status = 'pending' + AND (q.charged_amount < 0) = (p.charged_amount < 0) -- same sign + AND ABS(julianday(substr(q.date,1,10)) - julianday(substr(p.date,1,10))) <= 5 + AND ABS(ABS(q.charged_amount) - ABS(p.charged_amount)) <= 0.20 * ABS(p.charged_amount) +WHERE p.workspace_id = ? +``` + +`julianday` and `substr` are core SQLite/better-sqlite3 functions, safe here. Resolve each pending to at most one posted (closest amount, then closest date). Tombstone the pending (do not hard-delete): add a `superseded_by INTEGER REFERENCES transactions(id)` column so undo can restore it and event links can re-home. Analytics must additionally filter `superseded_by IS NULL` (a new `AND` on the same ten queries) so a tombstoned pending is not counted; this is an explicit, additive part of this migration, not a free side effect. Drift beyond 20% does not auto-merge; it surfaces as a manual-dedup suggestion. + +- **Idempotent re-matching.** The detector is safe to run every sync: it only reconsiders `status='suggested'` events and un-grouped, non-locked rows. Before creating any suggestion it checks the override log (case 7); confirmed events are never recomputed. + +### 7. User manual override conflicts (engine must respect user decisions permanently) + +**Failure mode.** The user un-merges an engine event, or sets a `transfer` back to `expense` via `setTransactionKind`. Next sync, the deterministic detector re-runs and re-hides the row. `batchUpdateCategories` already guards `category_source IS NOT 'user'`, but there is NO equivalent guard on `kind`: the `ON CONFLICT` clause hardcodes `kind = transactions.kind` (preserving kind on re-insert, good), yet the post-sync `markTransfersByIds` pass will happily overwrite a user's manual un-transfer, and the gated cc-payment pass would re-flip it. + +**Mitigation.** A durable, append-only `user_overrides` table keyed by a **stable fingerprint**, not the autoincrement `id` (which changes when a pending row is tombstoned in case 6). Use `(workspace_id, dedup_hash, dedup_sequence)`, the natural key that survives re-sync: + +```sql +CREATE TABLE user_overrides ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + workspace_id INTEGER NOT NULL REFERENCES workspaces(id) ON DELETE CASCADE, + dedup_hash TEXT NOT NULL, + dedup_sequence INTEGER NOT NULL, + field TEXT NOT NULL CHECK(field IN ('kind','event_membership')), + decided_value TEXT NOT NULL, -- e.g. 'expense', or 'unlinked' + created_at TEXT NOT NULL DEFAULT (datetime('now')), + UNIQUE(workspace_id, dedup_hash, dedup_sequence, field) +); +CREATE INDEX idx_user_overrides_lookup ON user_overrides(workspace_id, dedup_hash, dedup_sequence); +``` + +The detector loads the override set once per workspace per sync (one query, O(overrides), not per-row), and treats it as a hard skip-list: + +```typescript +const overrideSet = loadKindOverrides(workspaceId); // Set<`${dedupHash}:${dedupSequence}`> +function isLocked(c: { dedupHash: string; dedupSequence: number }): boolean { + return overrideSet.has(`${c.dedupHash}:${c.dedupSequence}`); +} +const eligible = candidates.filter((c) => !isLocked(c)); +``` + +This requires extending `TransferCandidate` and `getInternalTransferCandidates` to also select `dedup_hash`/`dedup_sequence` (today they select only `id`). Every code path that writes `kind` from the engine (`markTransfersByIds`, the cc-payment pass, ATM-transfer pass) must filter out locked rows first. The override is permanent and survives DB compaction because it is data, not state. Optionally offer to promote an override into a reusable normalized-description rule (handled in the rules section). + +### 8. False positives: two unrelated equal-amount, same-day transactions + +**Failure mode.** A `-50.00` coffee on a card and a `-50.00` standing order on the bank, same day. A pure amount+date matcher pairs and hides both, erasing `100` of real spend. `findInternalTransferPairs` is exactly this matcher, saved today only by its `matchesInternalTransfer` keyword guard (lines 99-104). + +**Mitigation.** Keep the **keyword-on-at-least-one-side requirement** as a hard precondition for any internal-transfer auto-grouping. Amount+date+sign alone never reaches even SUGGEST: + +```typescript +let score = 0; +if (exactAmountSameCurrency) score += 0.45; +if (oppositeSignDifferentAccount) score += 0.20; +if (dayGap <= TRANSFER_LOOKAHEAD_DAYS) score += 0.10; +if (matchesInternalTransfer(a.description) || matchesInternalTransfer(b.description)) score += 0.30; +const confidence = Math.min(score, 1); +// no keyword: 0.75 -> below 0.80 floor -> IGNORED +// with keyword + exact same-currency amount + sign + window: clamps to 1.0 -> AUTO eligible +``` + +Without a transfer keyword, two equal same-day rows score `0.75` and are ignored, so unrelated charges stay counted. Same-account same-sign equal amounts are dedup candidates (handled by `dedup_hash`/`dedup_sequence`), never routed to the transfer matcher; the matcher already filters to opposite signs and different accounts. + +Hebrew/RTL note: `matchesInternalTransfer` normalizes whitespace (`replace(/\s+/g, " ").trim()`) but does NOT strip RTL/LTR control marks (`\u200e`, `\u200f`) or normalize the maqaf (`־`, `\u05be`) versus ASCII hyphen that Israeli statements mix freely. The internal-transfer set (`העברה`, `העברת`) should add the geresh/quote variants and tolerate an embedded bidi mark, mirroring how `CREDIT_CARD_PAYMENT_PATTERNS` already handles `ישרא[\s־-]?כארד`. Without this, a transfer labeled `"העברה\u200f"` fails to match and stays double-counted (a no-auto-hide-safe failure, but a missed legitimate exclusion). + +### 9. Installments versus lump sum + +**Failure mode.** An Isracard installment purchase (`type='installments'`) posts as N rows over N months while the bank may show the full lump-sum charge or N monthly debits. A matcher could pair one installment leg against the lump-sum debit and hide a real installment, or try to re-sum installments against the bill. + +**Mitigation.** Installment rows are spend and stay spend; they are purchases, never money movements, so they are never transfer-grouped. `dedup_hash` already includes `installmentNumber` + `installmentTotal`, so the N rows are distinct and individually counted (correct: each month's installment is that month's spend). The bank-side bill is flipped by the cc-payment keyword path (case 1), with no attempt to reconcile installment counts against the bill. Add an explicit guard so installment rows are never offered as transfer legs (note `type` is not currently on `TransferCandidate` and must be added to the candidate query): + +```typescript +const eligible = rows.filter( + (r) => r.kind !== "transfer" && r.chargedAmount !== 0 && r.type !== "installments", +); +``` + +### 10. Refunds, chargebacks, reversals + +**Failure mode.** A refund is a positive `charged_amount` on a card account; a reversal is a positive row on a bank account. The internal-transfer matcher could pair a refund with an unrelated same-amount debit and hide a real expense. `detectKind`'s `chargedAmount > 0 -> income` flip applies only to `hapoalim`/`leumi`, so a bank reversal can be mislabeled `income`. + +**Mitigation.** + +- A card-account positive row that does NOT match `CARD_PAYMENT_RECEIVED_PATTERNS` (case 2) is a refund: keep it `kind='expense'` with positive `charged_amount`. The card-provider branch of `detectKind` already leaves card rows as `expense` (the income flip is bank-only), so a refund on Isracard stays `expense`. Caveat: the category-level analytics use `SUM(ABS(charged_amount))`, which counts a `+200` refund as `+200` of spend, NOT as a `-200` offset. Truly netting a refund against its original charge requires either signed sums in the per-category queries or an explicit `reversal` event that flips both legs out of spend. This design chooses the explicit `reversal` event (below) so the existing `ABS` queries stay untouched and net-zero is achieved by excluding both legs, not by mixing signs. +- Refunds are excluded from transfer-candidate generation unless they carry a transfer keyword. Link a refund only to its original charge as a same-account, same-`identifier`, opposite-sign reversal (`type='reversal'`), never as a cross-account transfer: + +```sql +-- reversal candidates: same workspace + account, opposite sign, shared identifier +WHERE a.workspace_id = b.workspace_id + AND a.account_number = b.account_number + AND a.identifier IS NOT NULL AND a.identifier = b.identifier + AND ABS(ABS(a.charged_amount) - ABS(b.charged_amount)) <= 0.01 + AND (a.charged_amount < 0) <> (b.charged_amount < 0) +``` + +Use `ABS` difference within epsilon rather than `a.charged_amount = -b.charged_amount` (exact float equality on `REAL` columns is unreliable). `identifier` is "not reliably unique across banks" per the known quirks, but same-account same-identifier opposite-sign within the window is a strong reversal signal; matched pairs flip both to `kind='transfer'` with `match_reasons` recording the shared identifier. When `identifier` is null, do not auto-link; SUGGEST instead. + +### 11. Deleting a bank credential an event depends on + +**Failure mode.** The user removes a `bank_credentials` row. If transactions cascade-delete with the credential, all that account's history vanishes and orphans `event_links`. A cc-payment event could lose its bank leg, leaving a flipped card-side credit with no explanation. + +**Mitigation, with a caveat the design must resolve first.** The grounding header lists `credential_id` on `transactions`, and `mapTransactionRow` reads `r.credential_id` and `queryTransactions` LEFT JOINs `bank_credentials` on `t.credential_id`. However, the `013_workspaces.sql` transactions recreate does NOT include a `credential_id` column, so this column is added by a later migration in the chain (the multiple-credentials work, `020_multiple_bank_credentials.sql`) and its FK on-delete behavior must be confirmed before relying on it. The required behavior: + +- `transactions.credential_id` must be `ON DELETE SET NULL` (NOT cascade), so deleting a credential preserves history. `mapTransactionRow` already tolerates null (`r.credential_id ?? null`) and `queryTransactions` LEFT JOINs, so a null link renders fine. +- `event_links.transaction_id` and `financial_events` are CASCADE toward `transactions` (so deleting a transaction cleans its links), but since transactions are NOT deleted on credential removal, links survive too. +- On credential delete, run an event-integrity pass: any `financial_event` that loses a required leg is downgraded `confirmed -> needs_review` (not deleted). A cc-payment card-leg whose bank leg is gone but whose card purchases remain stays flipped (purchases are still the spend source). Surface "an account this group referenced was removed" in review. + +If the actual FK turns out to be cascade, this design is blocked until it is changed to SET NULL via a table recreate (SQLite cannot ALTER an FK in place; follow the `013` recreate pattern). + +### 12. Many identical-amount transfers same day (greedy mispairing + O(n^2)) + +**Failure mode.** Three `-1,000` transfers between the same two accounts on the same day. The greedy `for (debit) { pick closest credit }` in `findInternalTransferPairs` can pair across the wrong triple, and crucially can pull in an *unrelated* same-amount row in a *third* account and hide a real expense. Separately, the current matcher is O(debits x credits) per candidate set; on a wide window with many same-amount rows this is a real quadratic trap. + +**Mitigation.** + +- Replace the nested scan with **hash-bucket blocking**: group candidates by unordered account-pair + rounded amount + day, then pair only within a block. This removes the cross-account false pairing AND collapses the quadratic to near-linear (each row hashes into one bucket; buckets are small): + +```typescript +const blockKey = (a: TransferCandidate, b: TransferCandidate) => + [a.accountNumber, b.accountNumber].sort().join("|") + + `:${Math.abs(a.chargedAmount).toFixed(2)}:${a.date.slice(0, 10)}`; +``` + +In practice you bucket each row once by `(amount.toFixed(2), date)` and only compare debits to credits sharing that bucket and a different account, never scanning all credits for every debit. + +- Within a block, require **balanced cardinality** before AUTO: only AUTO-flip when the block has equal numbers of debits and credits between exactly two accounts (a clean 3-out/3-in, all interchangeable, so mispairing among them is harmless because every leg is flipped regardless). An unbalanced block (3 debits, 1 credit, suggesting a real third-account charge leaked in) drops to SUGGEST and asks the user. Optimal min-cost (Hungarian) matching is unnecessary for interchangeable equal amounts and is not worth the complexity here. + +### 13. Self-transfers within one bank across sub-accounts + +**Failure mode.** A move from checking to savings under the same `credential_id` (one Hapoalim login, two account numbers). `isDifferentAccount` returns true on differing `accountNumber`, which is correct. The inverse risk: some providers reuse or omit `account_number`, so two genuinely different sub-accounts can share an `account_number` (then `isDifferentAccount` returns false and the self-transfer stays double-counted), or a provider returns one `account_number` for the whole login (intra-login transfers invisible). + +**Mitigation.** + +- `isDifferentAccount` is correct as a *necessary* condition. Harden it: two rows are the same account only when BOTH `credential_id` and `account_number` match. When `account_number` collides but the rows are clearly two sub-accounts and the scraper exposes a sub-account list, use it to disambiguate. Where the scraper cannot disambiguate, the safe behavior is to NOT auto-pair (leave both counted) and SUGGEST, never to risk hiding a real expense on a shared account number. +- Same-`credential_id`, cross-`account_number` transfers are the cleanest case (same owner) and may reach AUTO when they also carry an internal-transfer keyword and an exact same-currency, same-amount, opposite-sign match within the window. Self-transfers never carry a credit-card-payment keyword, so they go through the internal-transfer path, not the cc-payment path, and are correctly excluded from both spend and income. + +### Cross-cutting invariants + +1. **Excluding is per-leg via `kind='transfer'`, never per-pair, and never via `is_excluded`** (which today subtracts nothing from totals). An incomplete or wrong grouping degrades to "each leg still correctly flipped or still correctly counted," never "money reappears or vanishes." +2. **No-auto-hide.** Only (a) the cc-payment keyword path for a recognized bank provider AND gated on a tracked card, and (b) exact same-currency internal-transfer keyword pairs, reach AUTO. Everything fuzzy is SUGGEST with `needs_review=1`; the row stays in analytics until a human confirms. This is a deliberate, behavior-changing tightening of today's unconditional `detectKind` cc-payment flip. +3. **Every auto decision is explainable and reversible.** Persist the comparison vector and feature weights in `financial_events.match_reasons` (JSON text column; better-sqlite3 has no native JSON type, store stringified and parse in app code, or use SQLite's `json_*` functions for queries). Unlink is a delete of `event_links` rows plus a `user_overrides` entry, plus reverting `kind` on the affected legs; it never mutates amounts. +4. **User decisions are permanent and survive re-sync** via `user_overrides` keyed on `(workspace_id, dedup_hash, dedup_sequence)`, loaded once per sync as a hard skip-list and consulted before every engine write to `kind`. +5. **Backward compatible and additive.** New tables (`financial_events`, `event_links`, `user_overrides`), a new `superseded_by` column with the matching `AND superseded_by IS NULL` added to the ten analytics queries, plus the existing `kind` column. No destructive change. The current `detectKind` + `findInternalTransferPairs` + `markTransfersByIds` orchestrator passes become the AUTO tier, now gated (case 5) and override-aware (case 7). Where a behavior depends on a fact not yet true in the codebase (banks beyond `hapoalim`/`leumi` in `BANK_PROVIDERS_SET`; `credential_id` FK being `ON DELETE SET NULL`), that prerequisite is called out at the point of use and must be landed before the dependent AUTO behavior is enabled. + +--- + +## UX Recommendations for Merged Transactions + +This section specifies the user-facing surface for the FinancialEvent grouping layer defined elsewhere in this document. Everything here is additive over the current `transactions-table.tsx`. It reuses the existing `Badge`, `DropdownMenu`, `Card`, `Switch`, `Select`, and `SettingCard` primitives, and it never deletes or hard-mutates a scraped row: an event is a separate link layer that annotates rows, so the existing exact dedup (`UNIQUE(workspace_id, dedup_hash, dedup_sequence)`) and the `kind` column stay authoritative and re-sync stays idempotent. + +The single non-negotiable principle (Monarch, Copilot, YNAB, Simplifi alike): **never silently hide money, always show the trail, always make it reversible.** Spending math already drops `kind='transfer'`; the moment we group rows, the user must see exactly which rows were grouped, why, and be able to undo it in one click. + +One grounding correction that shapes the whole UX: today only `hapoalim` and `leumi` are in `BANK_PROVIDERS_SET` (verified in `src/server/lib/transfers.ts` line 7), so `detectKind`'s card-payment and bank-income branches only fire for those two providers. Card payments from other banks (mizrahi, discount, oneZero, etc.) are currently mis-filed as `expense` and double-counted. The event layer must therefore be the *general* fix and the UI must not assume a row already carries `kind='transfer'` just because it is the bank side of a card bill. The migration that widens the provider set and the matcher that scores these events are specified in the data-model section; this section consumes their outputs. + +### Shared assumptions about the data shape this UX consumes + +The list endpoint (`GET /api/transactions`) returns, per row, an optional event envelope. The UX only needs these fields (names match the migration columns proposed in the data-model section): + +```ts +// extends TransactionWithCategory in src/lib/types.ts +interface FinancialEventRef { + eventId: number; + eventType: "internal_transfer" | "card_payment" | "atm" | "manual"; + // role of THIS row within the event: + // representative = the row shown in the ledger + // leg = a participating side (transfer/atm legs) + // underlying = a card purchase sitting behind a bill + role: "representative" | "leg" | "underlying"; + status: "auto_confirmed" | "suggested" | "user_confirmed" | "user_rejected"; + // Fellegi-Sunter total match weight; null for user-created merges (no score). + matchWeight: number | null; + // i18n keys + params, localized client-side (see Tier 1 below). Never pre-rendered + // server-side, because the UI locale is per-request, not per-row. + matchReasons: Array<{ key: string; params?: Record }>; + legCount: number; // total participating rows (legs + underlying), excluding representative + // pre-summed on the server so the client never re-sums (avoids double-count and + // avoids shipping every underlying row just to compute a header total). + underlyingTotalAbs: number | null; // ABS sum of charged_amount of underlying purchases + underlyingCurrency: string | null; // currency of that sum; null if mixed (see RTL/currency note) +} + +interface TransactionWithEvent extends TransactionWithCategory { + event: FinancialEventRef | null; +} +``` + +The list query collapses an event to its representative row **server-side**, so the existing 50-row pagination (`PAGE_SIZE = 50`, line 87) still counts ledger rows, not raw rows. Choosing the representative deterministically: + +- **internal_transfer:** the debit leg (the outflow, `charged_amount < 0`). Per the sign convention, debit is negative. +- **card_payment:** the bank debit (the one negative bank row that settles the bill). +- **atm:** the bank debit. + +The non-representative legs and any `underlying` purchases are **not** sent with the page; they are fetched lazily on expand via `GET /api/events/:id` so the list payload and the SSE-driven home refresh stay small. This matters: a card bill can have 30 to 100+ underlying purchases, and shipping all of them inline would blow up the 50-row page. + +### 1. How a merged event appears in the transaction list + +Render the **representative row only**, in place, with a small inline event badge after the description. Do not introduce a second list or move events out of the ledger; this preserves the "ledger is complete" mental model and matches Copilot's badge-in-list approach. + +Badge styling reuses the exact pattern used for the `needsReview` pill (`transactions-table.tsx` lines 454-474) so it is visually native: + +```tsx +// new component: src/components/transactions/event-badge.tsx +import { ArrowLeftRight, Banknote, CreditCard, Link2 } from "lucide-react"; +import { useTranslations } from "next-intl"; +import type { FinancialEventRef } from "@/lib/types"; + +const EVENT_META = { + internal_transfer: { icon: ArrowLeftRight, key: "eventTransfer" }, + card_payment: { icon: CreditCard, key: "eventCardPayment" }, + atm: { icon: Banknote, key: "eventAtm" }, + manual: { icon: Link2, key: "eventMerged" }, +} as const; + +export function EventBadge({ event }: { event: FinancialEventRef }) { + const t = useTranslations("transactions"); + const meta = EVENT_META[event.eventType]; + const Icon = meta.icon; + const suggested = event.status === "suggested"; + // All four event types are "money routed, not spent", so they share the neutral + // token. We differentiate by ICON, not color, to stay legible for color-blind users. + const color = "var(--status-neutral)"; + return ( + + + {t(meta.key)} + {event.legCount > 1 && {`\u00D7${event.legCount + 1}`}} + + ); +} +``` + +Note the count is `legCount + 1` (legs plus the representative) so "×2" reads honestly for a simple two-sided transfer. The `\u00D7` escape avoids a literal `×` that some editors mangle in RTL files. + +Microcopy (en.json keys under `transactions`): +- `eventTransfer`: "Transfer" +- `eventCardPayment`: "Card payment" +- `eventAtm`: "ATM" +- `eventMerged`: "Merged" + +The representative row also gets a **chevron disclosure**. Do **not** reuse the leading 32px column (line 372): that column already holds the direction arrow (`ArrowUpRight`/`ArrowDownRight`, lines 440-446) keyed off `chargedAmount > 0`, and overloading it would conflict on every row. Instead, render a `ChevronRight`/`ChevronDown` toggle *inside* the description cell, before the description text, only when `event != null`. The direction arrow stays where it is. The toggle uses `ms-0`/`me-1` logical spacing so it sits on the leading edge in both LTR and RTL. + +**Expanded state** renders the participating legs as indented sub-rows: a single `` whose one `` (the table has 7 columns: spacer, date, description, category, account, amount, actions) hosts a nested compact list. This preserves outer column alignment without a second layout. Legs are fetched lazily on first expand (`GET /api/events/:id`) and cached in React Query under `["event", eventId]`. Each leg shows: its account (reuse `TransactionSourceCell` with `provider` + `accountLabel`), its own signed amount via `formatCurrency(amount, currency ?? "ILS", locale)` so debits render negative and credits positive exactly as the ledger does, and a **role chip**: + +- **Internal transfer:** "Out of {account}" on the debit leg, "Into {account}" on the credit leg. Clicking a leg deep-links to that row in the ledger (Simplifi's "go to other side"). +- **Card payment:** the bank debit is labeled "Bill payment". Underlying card purchases, if linked, are summarized under a subheading "Counted as spending here" showing `legCount` purchases and `underlyingTotalAbs` (the server-computed ABS sum). This is where Spent exceeds Monarch/Copilot: the one-to-many bill↔purchases relationship is made visible and auditable, which is the exact gap in the brief. The purchases are **not** re-summed client-side and the header explicitly states they, not the bill, are the source of spend. + +Worked microcopy for the expanded card-payment header, defusing the double-count confusion in plain language (note the amount is the bill's ABS value, since `charged_amount` is negative for the debit): +> "This ₪4,210 bill payment settles your Isracard card. It is not counted as spending. The 38 purchases it covers are already counted on the card." + +If `underlyingCurrency` is null (the bank bill is in ILS but underlying purchases were in mixed currencies, common for foreign card spend), drop the summed total and show only the count plus a per-purchase list, because summing across currencies would be a money bug. + +### 2. The "why" explanation (match reasons + confidence) + +Surface this in two tiers so casual users are not overwhelmed but power users can audit. + +**Tier 1, inline on expand:** a one-line plain-English summary built client-side from `matchReasons` (keys + params, localized at render so Hebrew/English follow the active locale): + +```ts +// reason keys map to phrases in en.json/he.json under transactions.reasons +function reasonLine( + reasons: FinancialEventRef["matchReasons"], + t: ReturnType, +): string { + const phrases = reasons.map((r) => t(`reasons.${r.key}`, r.params)); + return t("matchedBecause", { reasons: phrases.join(t("reasonJoin")) }); +} +// reasons.amountExact -> "same amount" +// reasons.amountClose -> "amounts within {eps}" (params: { eps: "₪0.01" }) +// reasons.dateWindow -> "{d} days apart" (params: { d: 2 }) +// reasons.oppositeSignCrossAccount -> "opposite directions in two of your accounts" +// reasons.keywordTransfer -> "description says “{kw}”" (params: { kw: "העברה" }) +// reasons.keywordCardPayment -> "looks like a credit card bill" +// matchedBecause -> "Matched because: {reasons}." +// reasonJoin -> ", " +``` + +Reason params carry the matched data (the actual keyword, the epsilon, the day gap) so a Hebrew keyword renders correctly inside an otherwise-English or RTL sentence; we wrap it in directional quotes and let the surrounding `dir="auto"` container resolve bidi, rather than concatenating raw Hebrew into a fixed string. + +**Tier 2, confidence:** reuse the `/7` display the AI confidence already uses (line 471: `{txn.aiConfidence}/7`) so the two confidence systems read consistently. The Fellegi-Sunter total match weight is on a 0..7 scale; we map it to a label, not a raw probability, to avoid implying false precision. Show it only in the expanded detail and the review queue, never on the collapsed auto-confirmed ledger row (it would be noise). + +```tsx +function confidenceLabel( + weight: number | null, + t: ReturnType, +): { label: string; tone: string; score: string | null } { + if (weight == null) return { label: t("confManual"), tone: "var(--status-neutral)", score: null }; + const score = `${Math.round(weight)}/7`; + if (weight >= 6) return { label: t("confHigh"), tone: "var(--status-on-track)", score }; + if (weight >= 4) return { label: t("confMedium"), tone: "var(--status-heads-up)", score }; + return { label: t("confLow"), tone: "var(--status-over)", score }; +} +``` + +Thresholds wired to behavior (the default policy; the user can shift them in §6): +- **weight ≥ 6:** auto-confirm. Event is grouped and collapsed; no review. Both transfer legs are set to `kind='transfer'` at sync time, exactly as `findInternalTransferPairs` already does, so spend/income math is correct immediately. +- **4 ≤ weight < 6:** suggested. Goes to the review queue with the dashed badge. **Legs keep their detected `kind` and stay in spend/income totals** until the user confirms, so nothing vanishes silently. +- **weight < 4:** not grouped, not suggested; rows stay independent. This keeps false positives near zero. + +### 3. Review queue for suggested merges + +Person-to-person payments and look-alike same-amount coincidences are the dominant false positives (the brief's P2P warning). So suggested merges (the 4..6 band) **must never alter spending math until confirmed.** The existing greedy pairing already excludes `kind='transfer'` rows from re-pairing (line 79), so confirming/rejecting is stable across re-syncs as long as the matcher only proposes from un-confirmed, un-rejected rows. + +Surface the queue in two places: + +1. A count chip on the existing `needs-attention-card.tsx` home widget: "3 possible transfers to review", reusing that widget's affordance rather than a new nav item. The count is computed server-side as `COUNT(*)` of events with `status='suggested'` for the workspace, returned alongside the existing `needsReviewCount` field (verified present in `src/lib/api.ts` line 328), so the home payload gains one integer, not a row dump. + +2. A dedicated review view: a stack of `Card`s, one per suggested event. Each card shows the N legs side by side, the Tier-1 reason line, the confidence label, and three actions: + +``` +┌─────────────────────────────────────────────────────────┐ +│ ⇄ Possible transfer Medium confidence 4/7 │ +│ Matched because: same amount, 2 days apart, opposite │ +│ directions in two of your accounts. │ +│ │ +│ -₪3,000 Hapoalim checking "העברה לחשבון" May 3 │ +│ +₪3,000 Leumi savings "העברה מחשבון" May 4 │ +│ │ +│ [ Confirm transfer ] [ Not a transfer ] [ Split ] │ +└─────────────────────────────────────────────────────────┘ +``` + +Action semantics (all mutations go through new event routes, then invalidate the same query keys the existing `handleKindChange` uses, lines 150-153): +- **Confirm transfer** → `POST /api/events/:id/confirm`. Sets event `status='user_confirmed'`, sets every leg's `kind='transfer'`, invalidates `["transactions"]`, `["summary"]`, `["transactions-summary"]`, `["categories"]`, and `["home"]`. Offers "apply to similar" as a follow-up toast action (§6). +- **Not a transfer** → `POST /api/events/:id/reject`. Sets `status='user_rejected'`, dissolves the link, reverts each leg to its detected `kind` (the pre-merge kind is stored on the event-link row so revert is exact), and writes a `rejected_event_signatures` row keyed on a *normalized, order-independent* signature of the legs (sorted leg dedup_hashes + rounded amount + event_type) so the same pair is not re-suggested next sync. Stickiness is the Copilot/Monarch lesson. +- **Split** → opens the split flow (§4) pre-scoped to this event. + +Bulk affordance: "Confirm all" at the top, enabled only for events with `weight ≥ 6` (the auto band is already confirmed, so in practice this button confirms any 6+ events the user manually downgraded; we never bulk-confirm the ambiguous 4..6 middle). This mirrors YNAB "approve matched" / Quicken "Accept All" without the false-positive risk. + +### 4. Undo, manual merge, and split controls + +All three live in the row's existing trailing `MoreHorizontal` dropdown (lines 560-579), extending today's kind-flip menu (which today only calls `setTransactionKind`, lines 569-577) rather than adding a control surface. + +**Undo / unlink.** For any grouped row, add a `DropdownMenuItem` "Ungroup" → `POST /api/events/:id/unlink`. This dissolves the event and restores each leg's pre-merge `kind` (stored on the link row, so undo is lossless), then shows a toast with an inline **Undo** action that re-creates the event from the same stored legs. Because we link rather than collapse, ungroup can never lose a scraped row. Unlinking an auto-confirmed event does **not** write a `rejected_event_signatures` row by default (the user may just want to inspect), but the toast offers "Don't suggest this again" to opt into stickiness. + +**Manual merge.** Two entry points: +- **Multi-select mode:** a checkbox column toggled by a "Select" button in the `CardHeader`. Selecting exactly two opposite-sign rows in **different accounts** enables "Mark as transfer"; selecting one bank debit plus N card-purchase rows enables "Link as card payment". This creates a `manual` event with `matchWeight = null`. Guard the two-row transfer case against same-account selection (a transfer within one account is meaningless) and against same-sign selection. +- **From a single row:** "Mark as transfer to…" opens a small `Dialog` to pick the counterpart row (searchable by amount/date, scoped to the opposite sign and a different account). If the user picks an account but no matching row exists, do **not** fabricate a phantom leg (unlike Wave). Instead set just this row's `kind='transfer'` via the existing `setTransactionKind` path and record a one-sided `manual` event with a single leg. This keeps balances honest when the counterpart account is not tracked in Spent, and reuses machinery that already exists. + +**Split.** "Split this group" detaches a chosen leg back to its own detected `kind`. For the card-payment case, "This bill is partly an old balance" lets the bank debit become an `expense` (for the carried-over portion) while the card-side legs stay transfers (the Copilot historic-balance escape hatch). This is a per-leg `kind` override plus an event annotation marking the event as partially split, not a whole-event toggle, so the rest of the grouping survives. + +### 5. Spending vs net-worth: how the views treat events differently + +Make the orthogonality visible: the point of grouping is that a transfer **moves** money without **spending** it (YNAB/Simplifi: counted once, at purchase, never at payment). + +- **Spending and category views** (`category-grid`, `hero-card`, budgets): behavior unchanged; they already filter `kind='expense'` and use `ABS(charged_amount)`. Grouped transfer/card-payment legs are excluded by construction. Add one trust affordance under the period total: a muted "Excluded: 4 transfers, 1 card payment", each segment linking to a ledger pre-filtered to those event types. This is Simplifi's "Excluded this month" and reassures the user money was routed, not lost. The counts come from one server-side aggregate over the same period filter the total already uses, so it is one extra grouped query, not an N-row scan. +- **Per-event "exclude from spending" indicator:** on the expanded event and in the representative row's tooltip, a muted pill "Not counted as spending" with the reason ("Transfer between your accounts" / "Credit card bill; the purchases are counted instead"). +- **Keep "transfer" (an event property) separate from "hide from reports" (the per-merchant `excluded_merchants` table).** Do not overload `excluded_merchants` with transfer logic: hiding a merchant is decluttering; marking a transfer is routing money correctly. Net worth must be computed from account balances, never from the spend ledger, so excluding a transfer from spend has zero net-worth effect (the bill payment settles a liability and nets to zero across the two accounts). State this explicitly in the data-model section so the balance computation is never derived from `SUM(charged_amount WHERE kind='expense')`. + +### 6. Settings: matching aggressiveness + the existing ATM toggle + +Add a new `MatchingCard` to `src/app/[locale]/settings/general/page.tsx`, immediately above `AtmCard`, following `AtmCard`'s exact structure (verified lines 293-333): controlled local state, `useMutation(updateSettings)`, dirty check, save button gated on `dirty`, `["settings"]` invalidation, success/error toast. Expose a three-way `Select` (remember: base-ui `Select`, `onValueChange` returns `string | null`, so guard the setter): + +```tsx +function MatchingCard({ initial }: { initial: "conservative" | "balanced" | "aggressive" }) { + const t = useTranslations("settings.general"); + const tCommon = useTranslations("common"); + const queryClient = useQueryClient(); + const [mode, setMode] = useState(initial); + // conservative: auto >= 6.5, suggest >= 5.5 (almost only exact matches auto-apply) + // balanced: auto >= 6.0, suggest >= 4.0 (default; the §2 thresholds) + // aggressive: auto >= 5.0, suggest >= 3.0 (more auto-grouping, more review noise) + const mutation = useMutation({ + mutationFn: updateSettings, + onSuccess: () => { + queryClient.invalidateQueries({ queryKey: ["settings"] }); + toast.success(tCommon("saved")); + }, + onError: (err) => { + toast.error(err instanceof Error ? err.message : t("failedToSave")); + }, + }); + const dirty = mode !== initial; + return ( + + +

{t(`matchHint.${mode}`)}

+
+ +
+
+ ); +} +``` + +Microcopy: +- `matchConservative`: "Conservative - only group obvious matches automatically" +- `matchBalanced`: "Balanced - group clear matches, ask about the rest" (default) +- `matchAggressive`: "Aggressive - group more, expect more review" +- `matchHint.balanced`: "We auto-group transfers we are sure about and send borderline ones to your review queue." + +The mode persists as one new `settings` key, `transferMatching` (string, defaulting to `"balanced"`), read by the matcher in the data-model section. Changing it never retroactively dissolves `user_confirmed` or `user_rejected` events (those are sticky user decisions); it only changes thresholds for future syncs. Offer an optional "Re-scan existing transactions" button that triggers a one-off matcher pass over the existing ledger. That pass must be **bounded**: the current `findInternalTransferPairs` is O(debits × credits) within its input set, which is safe for a single sync window but would be quadratic over a full multi-year ledger. The re-scan must therefore window candidates (e.g. group by currency, then slide a `dayWindow + epsilon` bucket over date-sorted rows) rather than feeding the entire `transactions` table into one pairing call. Specify this bound in the matching section. + +Keep `AtmCard` exactly as-is (`treatAtmAsTransfers`); it now reads as the ATM-specific complement to the general control, and ATM events flow through the same `EventBadge` (`eventType: "atm"`). + +### 7. Empty / first-run guidance + +- **Before any sync, or with a single connected account:** cross-account transfer detection is impossible with one account, so suppress the review-queue affordance and show an inline note: "Transfers are detected once you connect more than one account. Connect a bank and a card to see them here." Link to the bank-connection settings. (Card-payment grouping can occur with a single bank + single card, so do not gate that case on account count.) +- **First detected event:** the first time an event is auto-confirmed, show a one-time dismissible info banner above the ledger, reusing the `ai-not-connected-banner.tsx` pattern: "We grouped a transfer between two of your accounts so it is not double-counted as spending. You can review or undo any grouping from the row menu." Persist the dismissal in `settings` (one boolean key) so it does not reappear each session. +- **Empty review queue (healthy state):** "Nothing to review. Transfers and card payments we are confident about are grouped automatically." with a link to "See grouped transactions" (the ledger filtered to `event != null`). + +### 8. Trust checklist (acceptance criteria for the UI) + +Review the implementation against these; trust is the feature. +1. Every grouped row expands to show every participating leg and underlying purchase, each with its account and signed amount. No scraped row is ever invisible. +2. Every auto-confirmed grouping is reversible in ≤ 2 clicks (row menu → Ungroup) with a toast-level Undo, and unlink restores each leg's exact pre-merge `kind`. +3. No money leaves spend math without a visible counterweight: the "Excluded: N transfers" line under every period total links to the excluded rows. +4. Suggested (4..6 band) groupings never alter spend/income totals until the user confirms; legs keep their detected `kind` while suggested. +5. Confidence and reasons are always retrievable (expanded detail + review queue), using the same `/7` scale and the same status-color tokens as AI confidence. +6. Rejecting a suggestion is sticky via a normalized, order-independent leg signature (not re-suggested next sync); confirming offers "apply to similar". +7. Re-sync is idempotent: re-running a sync does not duplicate, re-suggest, or silently re-flip `user_confirmed`/`user_rejected` events. +8. Currency-safe: no header or total ever sums across currencies; mixed-currency underlying purchases show a count and per-row list, not a fabricated total. +9. RTL-safe: all new components use logical properties (`ms-*`, `me-*`, `ps-*`, `paddingInlineStart`) as the current table does, wrap mixed Hebrew/English reason strings in `dir="auto"`, and inject matched Hebrew keywords as i18n params rather than concatenating them into fixed strings. + +Cross-references: the FinancialEvent schema, the widened provider set, the bounded match-weight scorer, and the per-leg `kind` reconciliation are specified in the data-model and matching sections; this section consumes their outputs and assumes the additive SQLite migrations they define. + +--- + +## Scalable Implementation Strategy + +This section specifies how to build the financial-event grouping layer so it is correct on a single local SQLite file at the realistic scale (tens of thousands of rows per workspace), stays sub-quadratic so it would survive much larger histories, and remains backward compatible with the existing `kind` column and the exact dedup in `src/server/lib/dedup.ts`. The guiding principle, borrowed from the record-linkage literature (Splink, Christen): never do full pairwise comparison. Block first, score the small candidate set, then route by confidence. The existing `findInternalTransferPairs` in `src/server/lib/internal-transfers.ts` is already a pure, DB-free, clock-free function; we generalize that shape into the whole matcher so it stays unit-testable. + +Before any of this, three facts about the current code constrain the design and correct several tempting-but-wrong assumptions: + +- **There is no credit-card "bill credit" row to pair against.** Card providers (`isracard`, `cal`, `max`, `amex`, ...) return per-purchase rows only; they do not return a single "bill posted" credit on the card account. The bank side returns one debit (the monthly bill) with `charged_amount < 0`. So a CC bill payment is **not** a symmetric two-leg amount match and cannot be discovered by amount blocking against a phantom credit. It is a single bank debit whose amount approximately equals the **sum** of the underlying purchases over a billing cycle. Today `detectKind` already flips that single bank debit to `kind='transfer'` so it does not double-count; the event layer must make the one-to-many relationship explicit without inventing a second leg. +- **`detectKind`, the keyword sets, and `021_reclassify_credit_card_transfers.sql` only fire for `hapoalim` and `leumi`** (`BANK_PROVIDERS_SET` in `src/server/lib/transfers.ts` contains exactly those two). Every place the design says "bank provider" means "a provider in that set." New banks do nothing until added there. The event layer must inherit, not silently widen, this gate. +- **The dedup hash keys on `originalAmount`/`originalCurrency`, not `chargedAmount`.** The `ON CONFLICT` upsert in `insertTransactions` updates `status`, `charged_amount`, and `processed_date` only when the prior row was `pending`, and it **explicitly preserves `kind`** (`kind = transactions.kind`). So a pending->posted transition keeps the same hash/sequence and updates the row in place; it never creates a duplicate, but it also never re-runs `detectKind`. Re-classification therefore has to treat "row whose `status` flipped to `completed` this sync" as dirty even though no new row was added. + +### 1. Keep candidate generation near-linear (blocking) + +`findInternalTransferPairs` is already better than naive O(n^2): it splits into debits and credits, sorts by `abs(chargedAmount)`, and for each debit scans credits picking the closest-date match (it is closest-date-wins, not first-fit, and ties are broken by the deterministic `sortKey`). On a 3-month window the inner credit list is small, so this is fine today, but it degrades quadratically as the window or account count grows. Replace the nested scan with explicit amount blocking so work is proportional to rows-per-amount-bucket, not the product of debit and credit counts. + +The block key for transfer pairing is the integer charged amount in agorot plus the charged currency: `Math.round(Math.abs(chargedAmount) * 100)` combined with `chargedCurrency`. Two halves of an internal transfer agree on charged amount and currency; only the date carries slack. Bucketing on the integer amount turns pair generation into: build a `Map`, then within each bucket do the small cross-product only for opposite-sign rows in different accounts whose dates fall inside the window. This is O(n) to build the map plus O(sum of per-bucket cross-products), which is effectively linear because legitimate same-amount collisions are rare. Use `chargedAmount`/`chargedCurrency` (not `originalAmount`) for blocking because that is the field both transfer legs share after FX, and it is the field analytics use. + +```ts +// src/server/lib/matching/blocking.ts (pure, no DB, no clock) +export interface Leg { + id: number; + credentialId: number | null; + accountNumber: string; + provider: string; + dayNumber: number; // Math.floor(Date.parse(date.slice(0, 10)) / 86_400_000) + amountAgora: number; // Math.round(chargedAmount * 100), signed; debit < 0 + currency: string | null; // charged currency + description: string; + kind: "expense" | "income" | "transfer"; +} + +export function blockByAmount(legs: readonly Leg[]): Map { + const buckets = new Map(); + for (const leg of legs) { + if (leg.kind === "transfer" || leg.amountAgora === 0) continue; + const key = `${Math.abs(leg.amountAgora)}|${leg.currency ?? ""}`; + const arr = buckets.get(key); + if (arr) arr.push(leg); + else buckets.set(key, [leg]); + } + return buckets; +} +``` + +Blocking on integer agorot also removes the floating-point `epsilon` comparison the current code does (`Math.abs(...) > 0.01`): equal agora counts are exactly equal integers, so there is no FP slack and no money rounding bug. Mixed-currency rows never collide because currency is part of the key; a genuine cross-currency transfer (charged ILS on one side, USD on the other) will **not** auto-pair, which is correct, because we have no FX rate to assert equality and must not silently net out two different-currency amounts. + +For fuzzy description matching in Phase 2 (when amount alone is ambiguous, e.g. several identical-amount transfers in one bucket), add a second blocking pass on `amountAgora + dayBucket` and only then run the string comparator on the few residual collisions. Do not run string similarity across the whole window; reserve it for tie-breaking inside an already-blocked bucket, which keeps the expensive comparator off the hot path. This matches Baxter/Christen's finding that exact-key blocking plus a narrow string pass beats global similarity. + +### 2. Bound work to the sync window, and treat the CC bill as a one-to-many sum, not a pair + +Ordinary internal transfers are already bounded by `getInternalTransferCandidates(workspaceId, fromDate)` where `fromDate = toLocalISODate(startDate)`. Both legs land within `dayWindow` (default 2) of each other, well inside the window. Keep that bound for transfers. + +The credit-card bill payment is structurally different and must not be modelled as a 1:1 amount match. One bank debit (the bill) corresponds to the **sum** of N card purchases that posted over the prior billing cycle (Israeli cards: purchases through cycle close, billed mid-to-late the following month, so purchases can be up to ~45 days before the bill date). There is no single card-account row equal to the bill amount. Two consequences: + +- The bank debit keeps the existing behavior: `detectKind` already flips it to `kind='transfer'` (for `hapoalim`/`leumi` only) when it matches `CREDIT_CARD_PAYMENT_PATTERNS`, so it is excluded from spend and does not double-count against the per-purchase rows. The event layer wraps that single row in a `cc_payment` event with one `bill_payment` member, preserving today's reporting exactly. +- Linking the bill to its underlying purchases is a **separate, looser, audit-only linkage** keyed on the card credential and the billing period, validated by `abs(sum(purchases) - abs(bill)) <= tolerance` (a few agorot for rounding plus optional fees). It must never change `kind` on the purchases (they are real expenses and must stay counted) and never change `kind` on the bill beyond the transfer flag it already has. It is purely explanatory: "this bill = these 37 purchases." Because it is a sum check, not an amount-equal check, it does not go through `blockByAmount`; it is computed per card credential over the lookback window. + +Bound the purchase lookback explicitly so a bill near the start of the sync window can still reach its purchases without rescanning all history, and scope that fetch to card-provider credentials only so it does not enlarge the transfer-pair candidate set: + +```ts +const CC_LOOKBACK_DAYS = 45; +const ccFrom = toLocalISODate(addDays(startDate, -CC_LOOKBACK_DAYS)); +``` + +If a card credential is not synced (the user only connected the bank, not the card), the sum will not reconcile; in that case create no purchase links and leave the bill as a plain `cc_payment` event with `confidence` reflecting the missing evidence. Never fabricate purchase members. + +### 3. Incremental re-matching (touch only new and affected events) + +Re-running a sync must not re-evaluate every historical row. The matcher operates on a working set of dirty legs. `insertTransactions` returns `{ added, updated }` and stamps `sync_run_id` on inserted rows, but note that the `ON CONFLICT` upsert preserves the **original** `sync_run_id` (it is not in the `DO UPDATE SET` list), so `updated` rows do not advertise themselves via `sync_run_id`. The dirty set is therefore the union of: + +- rows with `sync_run_id = currentSyncRunId` (genuinely new this sync), and +- rows whose `status` changed to `completed` during this sync (pending->posted reconciliations, which `insertTransactions` counts in `updated` but does not re-`kind`). + +To surface the second set without scanning everything, have `insertTransactions` collect the ids it upserted-as-update and return them (a small additive change: add `updatedIds: number[]` to `InsertResult`), or, more conservatively, select rows in the window with `updated_at >= syncStartedAt AND status = 'completed' AND kind = 'expense'`. Expand the dirty set by one blocking hop (other rows sharing the same amount bucket within the date window, plus same-card-credential rows within the CC lookback). Anything not in a touched bucket is left alone. + +Re-matching is idempotent: if a dirty leg is already a member of an event whose membership and scores are unchanged, skip it. Recompute an event only when a member leg is new, or when a pending->posted flip changed its `charged_amount` (which can move it into a different amount bucket, since the upsert does update `charged_amount`). A no-op sync touches zero events; a sync adding 200 rows touches only the events those 200 rows participate in. + +### 4. Data model: events as additive migrations + +Two new tables plus the existing `kind` stay the source of truth for reporting. The event is an explicit, auditable, reversible entity with member legs and a representative leg, generalizing the boolean-ish `kind='transfer'` flag without breaking it: `kind` remains the fast reporting filter, the event tables explain why. + +Migration files are applied in lexicographic name order by `src/server/db/migrate.ts`, and the repo already contains duplicate numeric prefixes (`020_excluded.sql` and `020_multiple_bank_credentials.sql`; `021_chat_sessions.sql` and `021_reclassify_credit_card_transfers.sql`). Numeric uniqueness is not enforced, only filename uniqueness, and sort order between same-number files is alphabetical, which has already changed apply order once. Use a fresh, unused prefix and a descriptive name to avoid an ambiguous ordering: `022_financial_events.sql`. Each migration runs inside a transaction with `foreign_keys = OFF` for its duration and a `foreign_key_check` afterward, so any FK declared here is verified at apply time. + +```sql +-- src/server/db/migrations/022_financial_events.sql +CREATE TABLE financial_events ( + id INTEGER PRIMARY KEY, + workspace_id INTEGER NOT NULL, + event_type TEXT NOT NULL, -- 'internal_transfer' | 'cc_payment' | 'atm' | 'loan' | 'investment' + representative_txn_id INTEGER, -- leg used for reporting/display (nullable until chosen) + confidence REAL NOT NULL DEFAULT 0, -- 0..1, P(match) from the additive score + match_reasons TEXT, -- JSON: {"reasons":["amount_exact","opposite_sign","date_gap=1","kw:העברה"],"priorKind":{"123":"expense","456":"income"}} + source TEXT NOT NULL DEFAULT 'auto', -- 'auto' | 'user' | 'rule' | 'backfill' + status TEXT NOT NULL DEFAULT 'suggested', -- 'suggested' | 'confirmed' | 'rejected' + created_at TEXT NOT NULL DEFAULT (datetime('now')), + updated_at TEXT NOT NULL DEFAULT (datetime('now')) +); + +CREATE TABLE financial_event_members ( + event_id INTEGER NOT NULL REFERENCES financial_events(id) ON DELETE CASCADE, + txn_id INTEGER NOT NULL, + role TEXT NOT NULL, -- 'debit' | 'credit' | 'bill_payment' | 'purchase' + prior_kind TEXT NOT NULL, -- the leg's kind BEFORE this event flipped it; lossless undo + PRIMARY KEY (event_id, txn_id) +); + +-- One txn can belong to at most one *active* (non-rejected) event of the merge kind. +-- Enforce single membership for the flip-bearing roles so a row is never double-grouped. +CREATE UNIQUE INDEX idx_fem_single_active_member + ON financial_event_members(txn_id) + WHERE role IN ('debit', 'credit', 'bill_payment'); + +CREATE INDEX idx_fe_workspace_type ON financial_events(workspace_id, event_type); +CREATE INDEX idx_fem_txn ON financial_event_members(txn_id); +``` + +`prior_kind` lives on the member, not in a JSON blob, so undo is a trivial deterministic write and does not depend on parsing `match_reasons`. (The JSON copy in `match_reasons.priorKind` is for display only.) The partial unique index makes "a transaction is in at most one active transfer/bill event" a database invariant rather than a convention, which prevents the matcher from ever flipping a row into two events on a re-sync. `purchase` members are deliberately excluded from that index because one purchase belongs to exactly one bill, but a bill links many purchases, and purchases must remain individually counted as expenses regardless. + +Backward compatibility: every spend/income query keeps filtering on `kind`; no analytics query changes. When an event auto-confirms, the same `markTransfersByIds` write flips `kind` to `transfer` for the `debit`/`credit`/`bill_payment` members exactly as today, so reporting is byte-for-byte unchanged for the keyword-confident cases. `representative_txn_id` is what dashboards display for a grouped event, so N legs collapse to one line without deleting any row (a pointer-join, not a destructive merge). Note the FK story: there is no FK from event tables to `transactions(id)` here because that would require `transactions` to be the parent and complicate the existing recreate-style migrations; integrity is instead maintained by the matcher always writing members for live rows and by `ON DELETE` being a non-issue (the app never hard-deletes transactions). If a referential guarantee is wanted later, add `txn_id INTEGER NOT NULL REFERENCES transactions(id) ON DELETE CASCADE` in a follow-up migration once the recreate pattern for `transactions` is retired. + +### 5. Store-agnostic, pure matching core + +All scoring and pairing live in pure functions under `src/server/lib/matching/`, mirroring `findInternalTransferPairs`: plain rows in, plain proposals out, no DB and no `Date.now()`. The DB layer only loads the bounded candidate `Leg[]` and persists the resulting events. This is what lets the project move from SQLite to Postgres later without rewriting matching; only the thin query layer (`src/server/db/queries/transactions.ts` plus a new `events.ts`) is store-specific. + +```ts +// src/server/lib/matching/index.ts (pure) +export interface MatchInput { + legs: readonly Leg[]; + thresholds: { autoMin: number; reviewMin: number }; // policy, injected + weights: FeatureWeights; // m/u-derived, injected + ccToleranceAgora: number; // bill-vs-sum slack, injected +} +export interface ProposedEvent { + eventType: "internal_transfer" | "cc_payment" | "atm" | "loan" | "investment"; + members: { txnId: number; role: string; priorKind: Leg["kind"] }[]; + representativeTxnId: number; + confidence: number; // 0..1 + matchReasons: string[]; +} +export function proposeEvents(input: MatchInput): ProposedEvent[] { /* block + score + sum-check */ } +``` + +Thresholds, weights, and the CC sum tolerance are injected, never hardcoded inside the matcher, because the Fellegi-Sunter framing makes thresholds policy choices, not statistical facts. This keeps the core deterministic for tests: feed a fixed `Leg[]` and fixed policy, assert the exact `ProposedEvent[]`. The `dayNumber` precompute keeps all date parsing out of the scorer, so there is no clock and no flakiness. + +### 6. Scoring: Fellegi-Sunter with three zones + +Replace the boolean "keyword AND amount-within-epsilon AND date-within-window" gate with an additive weighted score, yielding a confidence to persist and to route on. Per feature, weight = log2(m/u); total weight M is additive; P(match) = 2^M / (1 + 2^M). Seed plausible weights, tune later (m/u can be learned via EM once labelled data exists): + +| Feature | Agree weight | Notes | +|---|---|---| +| amount exact (agora-equal, same currency) | +6 | high-signal, near-required for transfer pairs | +| opposite sign, different owned account | +4 | distinguishes transfer from a same-account duplicate | +| date gap 0-1 day | +2 | from `dayNumber` diff | +| date gap 2-3 days | +1 | settlement lag | +| internal-transfer keyword (either side) | +2 | `matchesInternalTransfer` | +| CC-payment keyword on bank side (hapoalim/leumi) | +3 | `CREDIT_CARD_PAYMENT_PATTERNS` | +| CC bill sum reconciles (abs(bill) == sum(purchases) +/- tolerance) | +5 | one-to-many evidence, replaces amount-exact for `cc_payment` | +| description similarity (Phase 2, normalized Jaro-Winkler >= 0.9) | +1 | tie-break within bucket only | + +Currency disagreement is a hard veto for transfer pairs, not a negative weight: if currencies differ, the pair is never proposed (Section 1), because we cannot assert amount equality across FX. + +Three zones (policy, tunable via settings): `confidence >= 0.97` auto-confirm (write event with `status='confirmed'`, flip `kind` on flip-bearing members); `0.80 <= confidence < 0.97` suggest (write event with `status='suggested'`, set `needs_review = 1` on those members, surface in the review queue) and **do not flip `kind`** so a merely-suggested merge can never silently hide a real expense; `< 0.80` propose nothing. This maps Plaid's VERY_HIGH/HIGH/MEDIUM gate and Splink's upper/lower clerical-review band onto the existing `needs_review` column that already feeds `getNeedsReviewCountByCategory` and `pendingReviewCount`. The crucial safety property: a row only leaves spend/income analytics when an event is `confirmed`; suggestions are visible-and-counted until a human (or a high-confidence auto rule) accepts them. + +### 7. Performance budget and inline vs background + +Today reclassification runs inline in `syncWorkspace` (the `findInternalTransferPairs` block around lines 409-418 of `orchestrator.ts` and the ATM block at 420-438), inside the SSE request, after all inserts and before AI categorization. Budget and placement: + +- Per-sync matching budget: < 250 ms for a typical 3-month window (a few thousand candidate legs). Blocking makes this trivially achievable; the old nested loop is the only thing that could blow it. +- Keep matching inline but move it behind one new orchestrator step `runEventMatching(workspaceId, syncRunId, startDate, syncStartedAt)` that wraps the dirty-set load, the pure `proposeEvents` call, and the persist. Emit `send("stage", { stage: "matching" })` so the UI shows progress, consistent with the existing `"categorizing"` and ollama stages. +- Hard escape hatch by candidate count, not by row count: if the bounded `Leg[]` exceeds a ceiling (e.g. 50k legs, only on a first full historical sync), still run the bill-payment single-row flips inline (cheap, per-credential) but defer the cross-account transfer pairing to a second `runEventMatching` call issued after the SSE `complete` event with the full window. Because matching is idempotent and incremental, running it twice is safe and the second pass only does the deferred work. There is no job queue in the stack; do not invent one. Reuse the existing inline call site and gate on `legs.length`. + +Memory bounds: the matcher holds only the bounded `Leg[]` (a handful of scalars per row) plus the per-bucket `Map`. For 50k legs that is single-digit MB. Never load full `TransactionRow` objects into the matcher; the dedicated candidate queries (`getInternalTransferCandidates` already selects only its needed columns) are the pattern to follow. Add a parallel `getCcPurchaseCandidates(workspaceId, ccFrom, cardCredentialIds)` that selects only `id, credential_id, account_number, date, charged_amount, charged_currency, description, kind`. + +### 8. Reuse the existing batch insert pattern + +Persisting events reuses `db.transaction(() => { ... })` exactly as `insertTransactions`, `batchUpdateCategories`, and `batchSetNeedsReview` do, so event writes inherit the WAL and `busy_timeout` settings from `src/server/db/index.ts` and are atomic. One transaction per sync wraps: insert into `financial_events`, insert members (with `prior_kind` captured from the live row before any flip), set `representative_txn_id`, and the existing `markTransfersByIds` / `batchSetNeedsReview` calls for confirmed events only. A partially-matched state is never visible. The partial unique index from Section 4 means a concurrent or repeated insert of the same membership fails loudly inside the transaction rather than producing a double-grouped row. + +### 9. Deterministic unit testing + +Because the core is pure, tests use `bun test` with fixed inputs and no DB: + +- `blockByAmount`: assert bucket membership for crafted amounts/currencies, including the 0-amount and already-`transfer` exclusions, and that two currencies with equal agora counts land in different buckets (no cross-currency netting). +- `proposeEvents`: feed fixed `Leg[]` with precomputed `dayNumber`, fixed weights/thresholds/tolerance; assert exact `ProposedEvent[]`, confidence, and `matchReasons`. Cover: a clean transfer pair (closest-date-wins when two credits tie on amount, matching today's `sortKey` behavior); a CC bill payment where `abs(bill)` equals the sum of N purchases within tolerance (one `bill_payment` + N `purchase` members, none of which flip `kind`); a CC bill whose purchases are absent because the card was not synced (single-member `cc_payment`, no fabricated purchases, lower confidence); two identical-amount transfers in one bucket (greedy assignment must not cross-link, and the partial-unique invariant must hold); an opposite-sign pair in the **same** account (must not pair); a cross-currency opposite-sign pair (must not pair); and a pending->posted `charged_amount` drift that moves a leg into a new bucket (must re-bucket and reconcile, not duplicate). +- Keep the existing `findInternalTransferPairs` tests as a regression backstop during Phase 1: Phase 1 must reproduce its exact output for the keyword-confident cases. + +Keep the `--conditions react-server` runner so `server-only` resolves to a no-op, as the CI gate already does. + +### 10. Concrete phased rollout + +**Phase 0: Ship the event model and backfill (no behavior change).** +- Migration `022_financial_events.sql` (tables and indexes above). +- Migration `023_backfill_events.sql`: derive `financial_events` from current state so history is not lost, idempotently (guard with `WHERE NOT EXISTS` against `financial_event_members` so a re-apply, or a second backfill of the same row, is a no-op). For every existing `kind='transfer'` row on `hapoalim`/`leumi` that matches `CREDIT_CARD_PAYMENT_PATTERNS` (reuse the exact `LIKE` clauses from `021_reclassify_credit_card_transfers.sql` so the set matches), create a `cc_payment` event with that row as the `bill_payment` member, `prior_kind='expense'`, `source='backfill'`, `confidence=1`, `status='confirmed'`. For the remaining `kind='transfer'` rows, a one-time Node backfill script (not pure SQL, since pairing needs `findInternalTransferPairs`) re-runs the pairing over historical windows and writes `internal_transfer` events for recovered pairs, capturing each member's current `kind` as `prior_kind`. `kind` is untouched throughout, so every dashboard keeps working identically. +- New query module `src/server/db/queries/events.ts` (create/read/confirm/reject/unlink), and pure `src/server/lib/matching/blocking.ts`. + +**Phase 1: Generalize internal-transfer and CC-payment into events with confidence and a review queue.** +- Refactor the inline blocks in `orchestrator.ts` into one `runEventMatching` call that uses `proposeEvents`. Keyword-confident transfer and CC-payment cases must produce events whose auto-confirm flips `kind` exactly as `markTransfersByIds` does today, so reporting is byte-for-byte unchanged for those cases. +- Generalize `detectKind` in `src/server/lib/transfers.ts`: the bank-side CC-payment detection (still gated on `hapoalim`/`leumi`) becomes a single-leg `cc_payment` event proposer so the per-row insert-time flip and the cross-account pass share one code path. Crucially, keep the insert-time flip itself (it is the cheap, correct default that prevents double-counting on the very first render); the event proposer only adds the auditable wrapper. Do not move CC detection out of the insert path, or a freshly synced bill would count as spend until the matching step ran. +- Build the review queue UI on the existing `needs_review` plumbing plus `financial_events.status`. Add confirm/reject/unlink endpoints that update `financial_events` and re-derive `kind`. Unlink restores each member's `kind` from `prior_kind` (lossless undo), deletes the event (cascading members), and clears `needs_review`. Confirm flips `kind` for the flip-bearing members; reject leaves `kind` untouched (it was never flipped for a suggestion) and marks `status='rejected'`. +- Persist `confidence` and `match_reasons` for every event; show them in detail: "matched because: exact amount, opposite sign, dates 1 day apart, keyword העברה." + +**Phase 2: Fuzzy matching, ATM/loan/investment, rule tuning.** +- Add the description-similarity feature only as an intra-bucket tie-breaker (Section 1). Normalize Hebrew first: strip RTL/LTR control marks (U+200E, U+200F, U+202A-U+202E), strip niqqud (U+0591-U+05C7), collapse the geresh/gershayim and whitespace, and apply NFC, before comparison. Lean on amount+date over the Hebrew string, since phonetic/Latin-oriented similarity is unreliable on Hebrew script; Jaro-Winkler here operates on already-normalized code points, not transliteration. +- Promote ATM (`isAtmWithdrawal`, currently the ad hoc block at lines 420-438) into an `atm` proposer, and add `loan` and `investment` proposers behind the same `proposeEvents` interface. Preserve the existing `treatAtmAsTransfers` setting semantics exactly (flip to transfer vs. file under "Cash & ATM"). +- Add a per-event-type rule layer (Monarch/Copilot pattern): user rules keyed on the raw `description` (not a cleaned name) that set event type and/or auto-confirm, applied retroactively. Reuse the existing `merchant-memory` and `category-corrections` infrastructure rather than building a parallel store. +- Add a settings-driven threshold tuner so `autoMin`/`reviewMin`/`ccToleranceAgora` are adjustable, with a validation harness that replays a labelled sample to estimate false-merge rate at each cutoff before applying. False-merge is the dangerous direction here (it hides real expenses), so the harness must report it separately from missed-merge. + +**Phase 3: Embeddings and scale-out.** +- Use Sentence-BERT-style embeddings of normalized descriptions purely as a fast candidate blocker and one more scorer feature, never as a replacement for the transparent additive scorer. Keep it optional and local, consistent with Spent's local-first, AI-optional posture; it must degrade to keyword+amount blocking when no model is present. +- Abstract `events.ts` and `transactions.ts` behind a store interface so Postgres becomes a drop-in for multi-install hosting. The matching core needs zero changes because it is already pure and store-agnostic. At Postgres scale, the same amount-bucket blocking becomes an indexed join, and the per-bucket cross-product stays the bounded unit of work. + +--- + +Files referenced (all absolute): +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/lib/internal-transfers.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/lib/transfers.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/lib/dedup.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/sync/orchestrator.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/db/queries/transactions.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/db/migrate.ts` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/server/db/migrations/007_transfers.sql` and `021_reclassify_credit_card_transfers.sql` +- `/Users/alonb/conductor/workspaces/spent-v1/pattaya-v3/src/app/api/sync/route.ts` + +--- + +## Appendix: Industry Practices + +Condensed from the research phase. Used to ground the recommendations above. + +### Record linkage / entity resolution literature and engineering docs. Primary authoritative sources: Splink (UK Ministry of Justice) Fellegi-Sunter theory + string comparators docs (moj-analytical-services.github.io/splink), Python Record Linkage Toolkit indexing docs (recordlinkage.readthedocs.io), Baxter/Christen "A Comparison of Fast Blocking Methods for Record Linkage" (ANU, users.cecs.anu.edu.au/~Peter.Christen), Christen "A Comparison of Blocking Methods" (arxiv.org/pdf/1407.3191), Ditto "Deep Entity Matching with Pre-Trained Language Models" (arxiv.org/pdf/2004.00584), "Pre-trained Embeddings for Entity Resolution" (vldb.org/pvldb/vol16/p2225-skoutas.pdf), Babel Street fuzzy name matching, and reconciliation engineering writeups (zerabooks.com, techinterview.org). NOTE: this research targets generic record-linkage technique; finance-aggregator-specific thresholds are mostly NOT publicly documented and are flagged as such. + +- **Transfer detection:** Not the focus of this research strand (transfer detection between owned accounts is a domain-specific pairing problem, not generic entity resolution). The transferable mechanism: treat candidate transfer pairs the same way you treat duplicate pairs - generate candidates via blocking, then score. Concrete blocking key for transfers = bucket on abs(amount) + a date window (the reconciliation literature uses amount-exact + date +/-1 to +/-3 days; settlement-delay-aware matching allows up to N settlement days). The distinguishing signal vs a true duplicate is OPPOSITE SIGN amounts across DIFFERENT owned accounts (one debit, one credit of equal magnitude), whereas a duplicate is SAME account or same import source with same sign. So the same Fellegi-Sunter additive-weight machinery applies; you just add a feature comparator for "sign-opposite & cross-account" that pushes toward transfer rather than duplicate. Not publicly documented for any specific consumer aggregator. +- **Credit-card payment matching:** Not directly covered by the entity-resolution sources. The transferable principle from the bank-reconciliation literature: holds/provisional rows and settlement rows should be LINKED rather than COLLAPSED ("deduplication logic linking rather than collapsing holds into settlements using merchant and amount tolerance"). Applied to CC-bill-payment vs underlying purchases: a CC bill payment is one large debit on the checking account that equals the SUM of many small purchases on the card. This is NOT a 1:1 duplicate and must not be matched by per-transaction amount/date similarity - the amounts and counts differ. The correct model is a one-to-many aggregate linkage (one payment <-> a statement batch of purchases), distinct from the pairwise dedup problem. The same payment also forms a transfer pair with the corresponding credit on the card account. Concrete consumer-aggregator thresholds for this are not publicly documented. +- **Internal linking / data model:** Data-model patterns from the literature: (1) Generate candidate pairs via an INDEX/blocking step, score each pair, then assign a shared cluster/linked identifier to pairs above a merge threshold - "record pairs above a specified threshold are considered the same person and assigned a new linked identifier" (Splink/MoJ). (2) Three-bucket classification per Fellegi-Sunter: MATCH (auto-link), NON-MATCH (ignore), INDETERMINATE (route to clerical/user review). (3) Persist the comparison vector and per-feature match weights alongside the decision so links are explainable and reversible. (4) For clustering many records into one entity, links are transitive (connected components) - but beware that low-confidence edges can chain unrelated records together, so cluster at a higher threshold than pairwise. Splink stores model parameters (m/u per comparison level) separately from the link decisions, which keeps the model auditable. +- **Dedup + matching heuristics:** BLOCKING/INDEXING (avoid O(n^2)): full pairwise comparison "scales quadratic" and is infeasible at scale (Python Record Linkage Toolkit, Christen). Options, with concrete params: (a) STANDARD BLOCKING - only compare records sharing an exact blocking key (e.g. same amount, or same rounded amount + same day); reduces comparisons drastically but misses pairs that disagree on the key. (b) SORTED NEIGHBOURHOOD - sort by a key, slide a window of width w over the sort order, compare records within the window; toolkit default window=3, "window of 1 returns the blocking index," larger window = more pairs/recall. Good when the key has spelling/value variation. (c) Q-GRAM / BIGRAM INDEXING and CANOPY CLUSTERING (cheap TF-IDF distance to form overlapping canopies, then expensive compare only within shared canopy) - Baxter/Christen found bigram indexing and canopy/TF-IDF give "large performance speed-ups and better accuracy" vs standard blocking and sorted-neighbourhood. (d) Multiple blocking passes (a disjunction of keys) to recover recall lost by any single key. For transactions, a natural cheap block = exact amount + date bucket, since true duplicates almost always share amount. + +SIMILARITY for short noisy strings (merchant/description): LEVENSHTEIN - insert/delete/substitute, best for general single-char typos, but does NOT handle adjacent transpositions. DAMERAU-LEVENSHTEIN adds transpositions ("CAKE"->"ACKE" = 1 op vs 2), "particularly useful for names." JARO - position-tolerant char matching, good when all chars equally important "e.g. ID numbers." JARO-WINKLER adds a prefix bonus (up to 4 chars), best for names/strings where the prefix is reliable; example MARTHA vs MARHTA = 0.961 (Jaro 0.944) - but "use with caution... especially with short strings" where no common prefix exists. JACCARD / TOKEN-SET - set overlap of tokens, ignores word order, best for multi-word strings (addresses, merchant descriptors with reordered/extra tokens). TF-IDF COSINE - token vectors weighted so rare tokens dominate, good for longer noisy text and as the cheap canopy distance. EMBEDDINGS (Sentence-BERT) - capture SEMANTIC not just syntactic similarity ("AMZN" ~ "Amazon"); string methods "can only capture syntactic not semantic similarity." Practical guidance: use exact/numeric comparators on amount and date (the high-signal fields), string comparators only on the merchant/description text. + +SCORING / CLASSIFICATION: weighted-sum rules (assign confidence by how many criteria relaxed) are simplest; logistic regression learns weights from labels; FELLEGI-SUNTER probabilistic matching is the principled standard - per feature define m = P(agreement | true match) (data quality, e.g. ~0.98 for DOB) and u = P(agreement | non-match) (coincidence/cardinality, e.g. ~0.005 for a rare surname). Match weight per feature = log2(m/u); total match weight = log2(lambda/(1-lambda)) + sum of feature weights (assumes feature independence, so weights are ADDITIVE). Convert to probability: P(match) = 2^M / (1 + 2^M). Reference points: weight 0 -> p=0.5, weight 4 -> p~0.95, weight 7 -> p~0.99. m/u can be estimated unsupervised via EM (no labels needed), which is why Splink/fastLink are attractive when you lack labeled duplicates. +- **Reconciliation / UX:** Threshold/auto-vs-suggest design from Fellegi-Sunter and Splink: classify each pair into THREE zones using an upper threshold T_lambda and lower threshold T_mu: above upper = auto-MATCH, below lower = auto-NON-MATCH, between = INDETERMINATE -> send to human/clerical review. Crucially, "thresholds are policy choices, not statistical facts" - Fellegi-Sunter lets you set the two thresholds to directly control the false-positive rate (mu) and false-negative rate (lambda), trading review volume against error rate (tightening the band sends more pairs to review but reduces auto-merge errors). Validation method: clerically label a sample of pairs across thresholds (including below cutoff) to estimate the actual FP/FN at each candidate threshold, then pick the threshold that hits your error budget. For a consumer app this maps cleanly to: auto-merge only very high confidence (e.g. exact amount + exact date + high description similarity = the "100% / amount+date+reference exact" tier in reconciliation writeups), SUGGEST/ask the user for the middle tier (amount + date +/-1 day + name similarity ~90%), and never auto-merge the relaxed tier (amount +/-10%, date +/-3 days). Always make merges reversible (store the comparison vector + weights so a merge can be explained and undone), since indeterminate auto-decisions will sometimes be wrong. Specific numeric auto-merge thresholds for named consumer aggregators are NOT publicly documented. +- **Takeaways:** + - Always block before scoring: full O(n^2) comparison is infeasible. For transactions the cheapest high-precision block is exact amount + a date bucket, because genuine duplicates almost always agree on amount; add a second blocking pass on rounded-amount + wider date window to recover near-duplicates (pending vs posted). + - Match the comparator to the field: use exact/numeric comparison on amount and date (your highest-signal fields) and reserve fuzzy string comparators for the merchant/description text only. Do not waste fuzzy logic where exact equality is the right test. + - For short noisy merchant strings, Damerau-Levenshtein (handles typos + transpositions) or Jaro-Winkler (prefix-weighted, good when the start of the name is reliable) beat plain Levenshtein; use token-set Jaccard when descriptors have reordered/extra tokens, and TF-IDF cosine when rare tokens should dominate. Jaro-Winkler is unreliable on very short strings with no shared prefix. + - Fellegi-Sunter gives the cleanest mental model: per feature, match weight = log2(m/u), total weight is additive across features, and P(match) = 2^M/(1+2^M). Anchor your UI thresholds to the probability mapping: weight 4 ~ 0.95, weight 7 ~ 0.99. m/u can be learned unsupervised via EM, so you do not need a labeled duplicate set to start. + - Adopt three zones, not a single cutoff: auto-merge above an upper threshold, ignore below a lower threshold, and route the indeterminate middle to a user review/suggest step. The thresholds are policy choices that directly trade review volume against false merges. + - Validate thresholds with a small clerically-labeled sample of pairs spanning the score range, then pick the cutoffs that meet your false-positive budget. Do not hardcode magic numbers without measuring. + - Distinguish three different linkage problems and do not collapse them: 1:1 duplicate (same sign, same source) -> dedup; opposite-sign equal-amount across two owned accounts -> transfer link; one large payment = sum of many purchases -> one-to-many aggregate link. Same scoring machinery, different candidate generation and different output (merge vs link vs annotate). + - Link rather than collapse for provisional rows: pending/hold and posted/settled versions of the same charge should be linked and reconciled, not silently merged into one row, because amount or date can shift on settlement. + - Hebrew/transliteration is hard for edit-distance and phonetic methods: Soundex/Metaphone are English-pronunciation-bound and 'disastrous' on non-Latin scripts; transliteration explodes one name into thousands of variants. Prefer matching in the original script and/or aggressive normalization (strip RTL marks, niqqud/diacritics, unify final-letter forms) before comparison, and lean on amount+date signals rather than the Hebrew string when possible. + - Embeddings (Sentence-BERT) add SEMANTIC matching that string metrics cannot (e.g. 'AMZN' ~ 'Amazon Marketplace'); they are most useful as a fast vector-similarity BLOCKER to generate candidates (Ditto reported ~3.8x pipeline speedup) and as one more comparator feeding the score, not as a wholesale replacement for the transparent, auditable weighted/Fellegi-Sunter scorer. + - Every auto-decision must be explainable and reversible: persist the per-pair comparison vector and per-feature weights so a merge can be shown to the user and undone. This is essential because the indeterminate band will produce occasional wrong merges. + +### Quicken Simplifi + Quicken Classic. Official help docs reviewed: Simplifi "How Do Transfers Work?" (support.simplifi.quicken.com/en/articles/3352152), "How Credit Card Payments and Transfers Are Handled in the Spending Plan" (.../5142302), "How to Resolve Duplicate Transactions" (.../4901071), "Amount Matching for Recurring Transactions" (.../9174873), "Reconciling Accounts in Quicken Simplifi" (.../8218379), "Managing Transactions" (.../3348103), "Account to Account Transfers (A2A)" (.../9980256). Quicken Classic docs: "Handling Downloaded Transactions" (quicken.com/support/handling-downloaded-transactions), "Quicken Downloads Transactions Which Are Duplicates of Existing Register Entries" (quicken.com/support/quicken-downloads-transactions-which-are-duplicates-existing-register-entries), "Duplicate Transactions or Accounts Downloaded in Quicken Mac" (quicken.com/support/duplicate-transactions-or-accounts-downloaded-quicken-mac), "Match your transactions" (info.quicken.com/mac/match-your-transactions). Date-window figure ("one week in the past, two weeks in the future") corroborated by Quicken community threads (community.quicken.com) but not on an official help page. + +- **Transfer detection:** Both products auto-detect transfers between owned accounts, but the mechanics differ. + +SIMPLIFI: "Quicken Simplifi automatically detects transfers between accounts." When a transfer is detected/categorized, the Category field shows the OTHER account's name (not a normal expense category), and a blue "Go to other side" link jumps to the matched transaction in the paired account. A "Linked Transfer" connects two real transactions: one showing money leaving account A and one showing it arriving in account B. If a user manually categorizes a transaction's Category as a destination account, Simplifi "will try to link it to the matching transaction in that account"; if none exists, it "will automatically create the transaction" on the other side. The exact auto-detection criteria (amount tolerance, date window, payee rules) are NOT publicly documented in Simplifi's help center. + +QUICKEN CLASSIC: Transfers are modeled by putting a bracketed account name in the Category field (e.g., [Checking]). When both sides of an inter-account transfer download, Quicken links them so the pair is counted once. Auto-match uses date + payee + amount within a date window (see dedupPrevention). + +DATA-MODEL NOTE for our design: Simplifi keeps TWO separate transaction rows (one per account) joined by a link/pointer ("Go to other side"), rather than a single shared row. Each side can be independently included/excluded from the Spending Plan, which requires both rows to exist and carry their own flags. +- **Credit-card payment matching:** Simplifi's core principle: "Each dollar spent should be counted once - when you make the purchase, not when you pay for it later." A credit-card payment is treated as a Transfer (moving funds from checking to the card), so it is "neither an income nor an expense" and is "excluded from the Spending Plan by default... because you're simply moving funds from one account to another, which does not change your net worth." This is what prevents double-counting: the individual purchases on the card register count as spending; the later bill payment is a transfer and does NOT re-count. + +Two mechanisms exist: +1. Built-in special categories "Credit Card Payment" and "Transfer" - keep a transaction neutral WITHOUT requiring a matching transaction in another account (useful when the other account is not tracked in Simplifi). These are excluded from the Spending Plan and shown greyed-out in an "Excluded this month" section. +2. Linked Transfer - when both the checking account and the credit card account are tracked in Simplifi, the payment is linked between them. For linked transfers the user can selectively include just one side, or both, in the Spending Plan for flexibility. + +Quicken Classic relies on the same transfer concept: a payment categorized as a transfer to the credit-card account nets to zero across accounts so the bill payment is not counted on top of the purchases. +- **Internal linking / data model:** SIMPLIFI data model: a transfer = two linked transaction records (one per account), bound by a link that powers the "Go to other side" navigation. Linking happens by (a) auto-detection, or (b) user setting one transaction's Category to the other account, which triggers a search for a matching transaction in that account and, if none found, auto-creation of the counterpart. Both sides carry independent "Exclude from Spending Plan" flags. Crucially, state propagates across the link: "If you apply a split to a pending transfer transaction in one account, the corresponding transfer in the other account will also be marked as Pending" - i.e., status/edits on one side cascade to the linked side. + +QUICKEN CLASSIC data model: the transfer is encoded inline via a bracketed account reference in the Category field (e.g., [Savings]). When the counterpart downloads, the bracketed link ties the two register entries together so they net out. Account-to-account integrity relies on this category convention rather than a separate join record. + +Newer "Account to Account Transfers (A2A)" in Simplifi is a feature for initiating real money movement, distinct from the categorization/linking model above. +- **Dedup + matching heuristics:** PRIMARY KEY for dedup is the bank-provided unique ID, not a heuristic. Quicken stores the Financial Institution Transaction ID (FITID), surfaced in the register as the "Downloaded ID" column (FITID column on Mac). "It is used by Quicken to determine which transactions have been downloaded and which ones need to be downloaded." A transaction whose FITID was already downloaded is not re-imported. Duplicates from aggregation occur mainly when this ID breaks: (a) the bank CHANGES its FITID format, so old transactions return with new IDs and "Quicken has no way of knowing that they're actually duplicates"; (b) reactivating an account re-downloads "the most recent 90 to 200 days of transactions (depending on the financial institution)"; (c) mixing download methods (in-app download vs manual file import from the bank site) yields different Downloaded IDs for the same transaction. Guidance: "do not mix your methods for getting transactions from your bank." + +SECONDARY heuristic match (downloaded-vs-register, e.g., user pre-entered a transaction or a scheduled reminder): Quicken matches on "Date + Amount + Payee" - "The payee is the same or similar. The date is within a few days of the scheduled date. The amount matches or is close to the scheduled amount." The exact official window is stated only as "within a few days"; Quicken community threads (not official help) cite the auto-match window as roughly "one week in the past to two weeks in the future" of the downloaded transaction, matching on the downloaded POSTING date which must be the same or earlier than the register date. + +CONFIDENCE / AUTO vs SUGGEST: Downloaded transactions get a status of New (no match), Matched (auto-matched, ready to Accept), or are presented for manual matching when auto-match fails. Quicken does not publish numeric confidence thresholds; matching is binary auto-match-or-not, then user-confirmed via Accept. Mac distinguishes only "Automatic match" vs "Manual match." + +SIMPLIFI: no automatic duplicate prevention/resolution is documented beyond the FITID-style dedup; resolving a leftover duplicate is fully manual. A duplicate that is bank-downloaded shows "Appears on your [account] statement as..." in the Transaction Detail; a manual entry lacks that line - this is the user's signal for which to delete. + +RECURRING/BILL amount matching (separate from raw import dedup): Any Amount (match by payee only), Exact Amount (zero variance), or Limited Range = "a 30% variance (15% above and 15% below) around the set amount." This is a concrete documented tolerance and is the only published numeric matching threshold across either product. +- **Reconciliation / UX:** QUICKEN CLASSIC: Real reconciliation exists. Workflow is "Compare to Register": after download, the user reviews each transaction and chooses Accept / Edit / Delete. "Accept All" bulk-accepts; "Undo Accept All" reverses a batch (available only until Quicken is closed or the next account update, and NOT usable for individually-accepted transactions). Cleared/reconciled state is tracked in the Clr column (c = cleared, R = reconciled). If the Quicken balance matches the online balance, accepted transactions can be auto-reconciled by stamping "R" in the Clr column. Manual matching fixes (right-click, Transactions menu, or drag-and-drop) handle cases where auto-match failed or matched the wrong entry; for an N-to-1 match Quicken creates a split with a line per selected transaction plus a difference line. + +SIMPLIFI: No traditional statement reconciliation. Instead there is a "reviewed" column (and a separate "flagged" column) - the user marks downloaded transactions as reviewed while comparing against the bank statement to confirm transactions and balance match. Balance discrepancies are handled via a dedicated "Resolve Balance Discrepancies" guide. + +CORRECTION / EDIT / SPLIT UX (Simplifi): inline edit of Date/Payee/Category/Amount directly from the list; full edit via the 3-dot Transaction Detail; bulk edit via checkboxes + pencil ("choose properties to change" then Apply); split into multiple categories/tags via the "Split" button with "Divide among Splits" for even distribution (splits cannot have separate dates); per-transaction and per-split "Exclude from Spending Plan" and "Exclude from Reports" flags. Duplicate resolution: merge a manual + downloaded pair (select both, click merge icon, confirm - WEB APP ONLY; mobile users must delete one). Support can only investigate duplicate issues "back 30 days." +- **Takeaways:** + - Make the bank's unique transaction ID (FITID-equivalent) the primary dedup key, not a fuzzy heuristic. Quicken's whole import-dedup model rests on a stored Downloaded ID/FITID; heuristic matching is only the fallback for user-entered or scheduled transactions. + - Plan for the FITID breaking. The dominant real-world duplicate cause is the institution changing its ID format or re-sending old transactions on reactivation (90-200 days). Add a secondary heuristic dedup layer (amount + date-window + normalized payee) as a safety net behind the ID match. + - Model a transfer as TWO linked records (one per account) joined by a pointer, not one shared row. This is what lets Simplifi show 'Go to other side' and lets each side be independently included/excluded from budgets. Cascade status changes across the link (Simplifi marks both sides Pending together). + - Treat credit-card payments as transfers and exclude them from spending by default, on the principle 'each dollar counted once - at purchase, not at payment.' This single rule prevents the most common double-count without special-casing. + - Support 'neutral' categories (Transfer / Credit Card Payment) that zero-out a transaction WITHOUT requiring a matching record. Essential when the counterparty account is not tracked - avoids forcing users to add every account just to silence a transfer. + - Use a small asymmetric date window for heuristic matching, biased toward the future (Quicken community figure: ~1 week past to ~2 weeks future), and match against the bank's POSTING date which should be on/after the user-entered date. Pending-to-posted timing makes the future side wider. + - Offer explicit amount-matching modes rather than one fixed tolerance: Any Amount (payee-only, for variable bills like utilities/CC), Exact Amount (zero variance), and a bounded range (Simplifi uses +/-15%, a 30% band). Let the user pick per recurring payee. + - Status-driven review beats silent auto-merge: classify each import as New / Matched and require an Accept step, with bulk Accept-All plus an Undo-All escape hatch scoped to the current session. Users want to see and reverse matches. + - Give a clear provenance signal for each transaction (Simplifi's 'Appears on your statement as...' shows bank origin vs manual entry). When a duplicate must be resolved manually, this tells the user which record is safe to delete. + - Provide a merge action for manual+downloaded pairs (not just delete), so a pre-entered transaction inherits the downloaded ID and stops re-duplicating. Quicken's failure mode is exactly when a manual entry never gets linked to its download. + - Decide reconciliation depth deliberately: full reconcile (cleared 'c' / reconciled 'R' states, auto-reconcile when computed balance == online balance) like Quicken Classic, or lightweight 'reviewed/flagged' columns like Simplifi. Even the light version needs a balance-discrepancy resolution path. + - Numeric matching thresholds are mostly undocumented in both products - only the recurring-amount +/-15% band is published. Don't assume there is a sophisticated confidence score; the observable behavior is binary auto-match-or-suggest, then human confirm. + +### Copilot Money (copilot.money). primary sources are the official Help Center (help.copilot.money) plus credible reviews. Pages used: Transaction Types (https://help.copilot.money/en/articles/3971267-transaction-types), Credit Card Payment Transactions (https://help.copilot.money/en/articles/10671434-credit-card-payment-transactions), Creating Manual Internal Transfer Payments (https://help.copilot.money/en/articles/4235839), Creating Manual Transactions (https://help.copilot.money/en/articles/4038706), Excluding Transactions (https://help.copilot.money/en/articles/9718801), Transactions FAQ (https://help.copilot.money/en/articles/10761907), Troubleshooting Account Duplicates (https://help.copilot.money/en/articles/8663179), Clearing Local Cache (https://help.copilot.money/en/articles/9922978), Apple Card/Cash/Savings (https://help.copilot.money/en/articles/9038131), Copilot Intelligence for Spending (https://help.copilot.money/en/articles/8182433), and Money with Katie review (moneywithkatie.com). + +- **Transfer detection:** Copilot has a first-class transaction TYPE for transfers called "Internal Transfer" (one of three types: Regular, Income, Internal Transfer). Per docs: "Money you move between accounts, such as paying a credit card bill, is considered an Internal Transfer. These transactions are also excluded from your spending budgets." Detection: "Internal Transfer transactions should automatically be captured in your account transaction data", i.e. when BOTH legs come from linked/connected accounts, Copilot classifies them as Internal Transfer automatically (and tells users to contact support "if you are missing transactions"). KEY DATA-MODEL FINDING: Copilot does NOT model a transfer as one linked record with two legs. It treats each side as a separate, independent transaction, each typed as an Internal Transfer (an "Outgoing Internal Transfer" on the source account and an "Incoming Internal Transfer" on the destination). For manual accounts the user must create both: "You would need to post two manual transactions: 1. Outgoing Transfer from the checking account. 2. Incoming Transfer to the savings account." Copilot does NOT auto-create an offsetting leg. Users can reclassify via the transaction menu ("Transfer" -> "Incoming"/"Outgoing" + choose the destination/source account), and can opt to "apply the same change to similar transactions." There is NO publicly documented amount/date matching window, fuzzy-match heuristic, or confidence threshold for PAIRING the two legs, the mechanism for how (or whether) it relationally links the two records is not publicly documented. The exclusion-from-budget effect is achieved purely by the TYPE on each transaction, not by detecting a matched pair. The Venmo integration is a special case where Copilot deterministically marks all bank-recorded incoming/outgoing Venmo transactions as Internal Transfers after Venmo setup. +- **Credit-card payment matching:** There is no automatic "matching" of a CC bill payment to its underlying purchases, and crucially that matching is unnecessary in Copilot's model because of how the types work. A credit card bill payment is just a money movement between two owned accounts, so both legs are typed Internal Transfer and thus excluded from budgets, while the individual card PURCHASES (Regular transactions) are what count toward spending. Per docs, a CC payment is two transactions: "the outgoing transaction from the checking account" and "the incoming transaction to your credit card," and "If these credit card payments are paying off your monthly balance, you'll want to leave the transactions from your checking account and credit card account as Internal Transfers, because that money is already accounted for in your budget with your individual transactions as they occur." Special historic-balance handling: if you're paying off an OLD balance (purchases predating Copilot, never budgeted), you change the checking outflow to "Regular" so it counts as spend, but "You should still leave the incoming transaction to your credit card as an Internal Transfer" to avoid double counting. Whether Copilot AUTO-types CC payments as Internal Transfer vs requiring the user to set it is not explicitly documented (the article only says what users "should" do; Copilot Intelligence may suggest the type but there's no documented auto-detect-and-pair of the two payment legs). No matching window/amount tolerance documented for CC payment legs. +- **Internal linking / data model:** Account-linking is via a third-party data aggregator (Plaid-style connections) per account; accounts are also supported as fully MANUAL accounts. Transfers are NOT represented as a single double-entry record linking two accounts. Instead each account holds its own transaction rows, and a transfer is two independent rows each carrying the Internal Transfer type. There is no documented foreign-key/relational pairing between the two legs that the user-facing docs expose, pairing as a data-model concept is "not publicly documented." Transaction-type taxonomy is explicit and small: Regular, Income, Internal Transfer; list view shows badges [R] Recurring, [I] Income, [T] Internal Transfer. "Excluded" is a separate orthogonal concept implemented via categories (a category can be flagged type "Excluded"), so exclusion is a property of the assigned category, not a boolean flag on the leg itself. ML categorization runs a PER-USER private model (each user gets their own model; data never leaves Copilot's systems). +- **Dedup + matching heuristics:** Copilot has NO publicly documented automatic at-import dedup/fuzzy-matching engine (amount + date-window + name similarity). Duplicates are treated as an exceptional, mostly aggregator-driven problem rather than something silently deduped. Documented causes: (1) deleting and reconnecting the same account where old transactions weren't removed ("a data caching issue ... usually caused by deleting and reconnecting the same account, but the transactions were not removed properly from the deleted copy"); (2) the aggregator rotating account IDs, "When details of an account change ... the data aggregator marks the original account ID as closed and generates a new account ID," producing a duplicate account; (3) re-issued cards with new last-4 digits; (4) user accidentally creating a duplicate connection; (5) manual+connected overlap (esp. Apple Cash importing historic data). Remedies are largely manual/support-driven, NOT algorithmic: clear the local cache (Settings > Advanced/Account > Clear local cache), and for account-level dupes "without making any edits to the accounts, please contact the Copilot team via the in-app chat" (support does a merge that preserves history). For Apple Cash overlap, the fix is upstream: set iPhone Settings > Privacy & Security > Wallet > Copilot > Activity to "Starting Today" or "Starting 30 Days Ago." The only "smart"/confidence-thresholded ML is for CATEGORY (and suggested type) prediction, not for dedup or transfer-pairing. +- **Reconciliation / UX:** User correction is via the transaction detail/menu: change the transaction TYPE (Regular / Income / Internal Transfer, with Outgoing/Incoming + account selection for transfers), change CATEGORY (tap category -> pick or create one, including toggling a category to type "Excluded"), and a "Exclude" option directly in the category list. When changing type/category, Copilot offers to "apply the same change to similar transactions" (a bulk/rule-style action), and you can create exact or partial transaction-NAME matching RULES so future matching transactions get the same treatment, this is the closest thing to a documented matching heuristic, and it's name-based and user-defined, not automatic. Excluded transactions: removed from spending totals across the app by default, but Cash Flow has a toggle to "add your excluded transactions in your total spend amount" (and then they also show on Dashboard); you can filter to view Excluded in the Transactions tab. Limitation: "We don't currently support marking a Recurring transaction as excluded." Manual transactions can be created/edited; deleting/reconnecting accounts is discouraged as a fix (it causes duplicates), support-assisted merge is preferred. Copilot Intelligence learns from each correction (model improves as you review more; surfaces top-2 guesses for quick re-categorization). No explicit documented "undo" command beyond editing the transaction back. +- **Takeaways:** + - Model transfers as a TYPE on each leg, not as a matched pair. Copilot's whole approach is to type both sides 'Internal Transfer' and exclude that type from budgets, this sidesteps the hard problem of pairing legs and is robust even when only one side is linked. Consider an enum transaction type {Regular, Income, Transfer} plus an orthogonal 'excluded' flag (Copilot puts excluded on the CATEGORY, which is a clean way to make exclusion reusable via rules). + - Avoid CC double-counting by typing the BILL PAYMENT as a transfer (excluded) while keeping the individual PURCHASES as spend. The purchases are the source of truth for budgeting; the payment is just money movement. No need to reconcile a payment against N purchases. + - Have an explicit escape hatch for the historic-balance edge case: paying an old, never-budgeted balance should let the checking outflow count as spend (Regular) while the card-side credit stays a transfer. Build a per-leg type override, not just a pair-level toggle. + - Make type/category changes propagate via user-defined name-matching RULES (exact or partial) and an 'apply to similar transactions' prompt. This gives users dedup/classification control without a risky fully-automatic engine. + - Copilot does NOT auto-dedup at import; it treats duplicates as aggregator/reconnect artifacts and leans on a manual merge + 'clear local cache.' Lesson: design your ingest to be idempotent (stable composite hash, like this project already does) so reconnects don't spawn dupes, that's exactly the failure mode Copilot suffers from. + - Beware aggregator account-ID churn (a connection's ID rotating creates a phantom duplicate account). Key off a stable account identity, not the raw provider account ID, and provide a merge flow that preserves history. + - Beware manual+connected overlap producing duplicate transactions (Copilot's Apple Cash case). If you support both manual and scraped versions of the same account, give users a clean cutover (close/keep history on one) and a date-cutoff on imports. + - Separate the ML categorizer from dedup/transfer logic. Copilot's ML only predicts category/type, with a confidence gate ('if not very confident ... it won't apply it'), a warm-up threshold (30 reviewed transactions before predictions turn on), a per-user private model, and a suggest-vs-auto UX (Intelligence badge + top-2 guesses). Pairing/dedup is deterministic, not ML, a good separation of concerns. + - Use clear, lightweight UI signifiers for type so users can audit at a glance (Copilot's [R]/[I]/[T] badges). Transparency builds trust that auto-classification didn't silently hide money. + - Pairing the two legs of a transfer (relational linking with amount/date-window/fuzzy matching) is NOT publicly documented in Copilot, meaning even a polished commercial app gets away WITHOUT true leg-pairing by relying on per-leg typing + user rules. You don't have to build hard two-sided matching to ship something good; start with per-leg typing and rules, add suggested-pair detection later. + +### YNAB (You Need A Budget) official docs. Primary sources: YNAB API / OpenAPI Transaction schema (https://api.ynab.com/v1 and api.ynab.com/papi/open_api_spec.yaml; field mirror https://github.com/dmlerner/ynab-api/blob/master/docs/TransactionDetail.md); Transfer Transactions guide (https://support.ynab.com/en_us/transfer-transactions-a-guide-HJOsZz4Jj); Credit Card Payments guide (https://support.ynab.com/en_us/credit-card-payments-a-guide-r1_506Q1j) and Handling Credit Cards overview (https://support.ynab.com/en_us/handling-credit-cards-overview-ry7cNub1s); blog "How to Manage Credit Cards" (https://www.ynab.com/blog/how-to-manage-credit-cards-in-ynab); Approving & Matching Transactions (https://support.ynab.com/en_us/approving-and-matching-transactions-a-guide-ByYNZaQ1i) and blog "Manually Match Transactions" (https://www.ynab.com/blog/matchmaker-matchmaker-make-me-a-match); Reconciling Accounts guide (https://support.ynab.com/en_us/reconciling-accounts-a-guide-BJFE3fHys) and blog "8 Myths About Reconciliation" (https://www.ynab.com/blog/8-myths-about-reconciliation-in-ynab). NOTE: the support.ynab.com article bodies are JS-rendered and could not be fetched verbatim; details below come from the API spec (authoritative for data model), the official blog (static), and search snippets quoting the guides. Where a precise value is not in those, it is flagged "not publicly documented." + +- **Transfer detection:** A transfer is a first-class, explicitly-modeled concept, not inferred by a heuristic. The user (or import) picks a payee of the form "Transfer: [Account Name]". This creates TWO paired transaction records, one in each account (equal and opposite milliunit amounts). The data model makes the pairing explicit: every TransactionDetail has `transfer_account_id` ("If a transfer transaction, the account to which it transfers") and `transfer_transaction_id` ("If a transfer transaction, the id of transaction on the other side of the transfer"). So a transfer = two rows cross-linked by id, each pointing at the counterpart account. Editing/deleting one side updates/removes the other (they are kept in sync because they are linked, not independent). KEY DESIGN POINT for double-counting: a transfer between two ON-BUDGET ("Budget") accounts is NOT given a category and is invisible to the budget/spending math, because the money has not entered or left the budget. Per YNAB: "you still have a total of $800. You don't have any additional nor any fewer dollars; nothing is happening in the Budget portion of YNAB" (eshmoneycoach quoting YNAB principle); it's "like taking $20 out of your wallet and putting it in your pants pocket." Conversely, a transfer involving a TRACKING (off-budget) account DOES require a category, because money is genuinely entering or leaving the budget at that boundary (that category-assigned leg is the legitimate inflow/outflow). On import (Direct Import / linked accounts), both legs can arrive separately; YNAB's matching engine pairs the imported entry to the existing manual transfer transaction on the same account so you don't get duplicates. This avoids double-counting by construction: an on-budget-to-on-budget transfer simply never hits any spending category, so it can never be summed as spending or income. +- **Credit-card payment matching:** A credit card payment is modeled as a TRANSFER from the funding account (e.g. checking) to the on-budget credit card account, NOT as a categorized expense. This is the central anti-double-counting mechanism. Mechanics: (1) When you record a PURCHASE on the credit card, you categorize it normally (e.g. "Dining Out"). YNAB then AUTOMATICALLY moves that same dollar amount out of the spending category and into the auto-created "Credit Card Payments" category for that card ("YNAB automatically moves the dollars in your categories" into the payment category; a $33 Fun Money purchase triggers $33 into the card's payment category). So the spend is counted once (in Dining Out / Fun Money) and the payment category just holds the reserved cash to settle the card. (2) When you PAY the card, you record a payment ("Record Payment" / or enter a transfer from checking with payee "Transfer: [Credit Card]"). "YNAB updates both your checking account and credit card account screens" - it writes the paired transfer (linked via transfer_account_id/transfer_transaction_id). Because the payment is a transfer (no spending category), it is never summed as new spending. Double-counting is impossible: the real expense was categorized at purchase time; the payment merely moves already-set-aside money to retire the card balance. On import, the bank's outflow on checking and the bank's payment-credit on the credit card account are matched to the single user-entered transfer (one on each side), preventing the payment from appearing twice. +- **Internal linking / data model:** Data model (from official YNAB API / OpenAPI TransactionDetail schema): each transaction row carries `account_id`, `payee_id`/`payee_name`, `category_id` (nullable - transfers between budget accounts have no category), `amount` (milliunits, signed), `cleared` (enum: cleared / uncleared / reconciled), `approved`, `deleted`. Linking fields: `transfer_account_id` = "If a transfer transaction, the account to which it transfers"; `transfer_transaction_id` = "If a transfer transaction, the id of transaction on the other side of the transfer"; `matched_transaction_id` = "If transaction is matched, the id of the matched transaction" (used for import-to-manual matching, distinct from transfer pairing). So YNAB has TWO distinct linking relationships: (a) transfer pairing (two real, persisted rows, one per account, mutually referencing) and (b) import matching (an imported row matched/merged to a pre-existing user-entered row). Credit cards are modeled as a special on-budget account type that auto-spawns a dedicated "Credit Card Payments" budget category; the transfer-payee + payment-category pair is what ties spending, reserved funds, and payment together. Off-budget/"Tracking" accounts are the escape hatch: transfers crossing the budget boundary into/out of them must be categorized. +- **Dedup + matching heuristics:** Two layers, both documented in the API. (1) DETERMINISTIC dedup via `import_id`: "If specified, a new transaction will be assigned this import_id and considered 'imported'." File-Based / Direct Import assign import_id in the format `YNAB:[milliunit_amount]:[iso_date]:[occurrence]` - e.g. a -$294.23 txn on 2015-12-30 becomes `YNAB:-294230:2015-12-30:1`, and a second identical-amount-same-date txn becomes `...:2`. If a transaction with an import_id already present on that account is sent again, "it will be skipped to prevent duplication," and the skipped import_id(s) are returned in a `duplicate_import_ids` list. Important caveat: if you send two transactions with the SAME import_id, the latter is ignored as a duplicate "even if the data are different" - so the occurrence counter is what disambiguates legitimate same-amount/same-day repeats. (2) FUZZY matching to user-entered rows: an imported transaction is matched to an existing "user-entered" transaction "on the same account, with the same amount, and with a date +/-10 days from the imported transaction date." This is the documented matching window: SAME ACCOUNT + EXACT AMOUNT + DATE WITHIN +/-10 DAYS. Auto-vs-suggest: when criteria are met, YNAB auto-matches (merges) and surfaces it for one-click approval ("approve the transaction using the normal flow we've always had for automatically matched transactions"); when auto-match misses, the user can MANUALLY match by selecting exactly two transactions and Edit > Match. Constraints on matching: amounts must be equal ("transactions with different amounts really don't match"); "Two imported transactions can't be matched" and "two user-entered transactions can't be matched either" - matching is strictly imported<->user-entered. No fuzzy-amount or payee-similarity threshold is publicly documented; matching is exact-amount + 10-day-window based, not a confidence score. +- **Reconciliation / UX:** Reconciliation = comparing YNAB's CLEARED balance to the bank's cleared balance so they "match to the penny." Three transaction states, surfaced as a status column: gray "c" = uncleared (not yet seen/cleared at bank); green "c" = cleared but not yet reconciled; green LOCK = cleared AND included in a prior reconciliation (locked). Locking is the integrity mechanism but is not rigid: "You can edit locked transactions if you need to, just be sure to re-reconcile after." When YNAB's cleared balance does not equal the bank's, YNAB offers to create a RECONCILIATION BALANCE ADJUSTMENT transaction: "A balance adjustment will represent those missing or incorrect transactions for you" so the user can move on without hunting every discrepancy. Recommended cadence: weekly (at least every other week) so discrepancies stay small/findable. Approval workflow is separate from clearing: imported transactions land as "unapproved by default" (`approved=false`) and require user Approve; matched imports are presented for approval too. Correction/undo: edits to locked reconciled rows are allowed (then re-reconcile); transfers and CC payments can be corrected by editing either paired row (the linked side updates); deletion is soft - the API notes "Deleted transactions will only be included in delta requests," implying tombstoning rather than hard delete (supports sync/undo). +- **Takeaways:** + - Model transfers as TWO explicit, mutually-linked rows (one per account) rather than a single record or a post-hoc heuristic. YNAB stores transfer_account_id + transfer_transaction_id on each side. Editing/deleting one side propagates to the other. This makes transfers a first-class type, not a guess. + - Kill double-counting at the schema level, not in reporting: a transfer between two owned (on-budget) accounts gets NO spending category, so it is structurally impossible to sum it as spend or income. Only transfers that cross the budget boundary (to/from off-budget 'tracking' accounts) get categorized. + - Model a credit-card payment as a transfer (checking -> card account), never as an expense. The expense is counted ONCE at purchase time (categorized); the payment just moves already-reserved money. For an Israeli aggregator this directly solves the 'CC bill payment vs the underlying card purchases' double-count problem. + - Auto-reserve funds for liabilities: when a card purchase is categorized, automatically move that amount into a dedicated 'Credit Card Payment' bucket. This keeps the spend visible in its real category while tracking what is owed - a clean separation worth mirroring for Israeli credit (e.g. Isracard/CAL monthly billing). + - Use a deterministic, idempotent import key for dedup: YNAB's import_id = `YNAB:[milliunit_amount]:[iso_date]:[occurrence]`. The occurrence counter disambiguates legitimate identical same-day/same-amount charges (very common with cards). Re-sending a known import_id is skipped and reported back in duplicate_import_ids. This is essentially YNAB's version of our dedup hash - note our dedup.ts also uses count-based dedup, which aligns well. + - Have a SECOND, fuzzy layer to merge imported rows into pre-existing manual ones: YNAB's documented rule is same account + EXACT amount + date within +/-10 days. Exact amount is required (no fuzzy-amount matching is documented); the slack is in the DATE, not the amount - good guidance for choosing a matching window. + - Separate 'clearing' (matches the bank) from 'approval' (user accepted the import) from 'categorization'. Imports arrive unapproved by default and need a one-click approve; auto-matches are merged but still surfaced for approval. Three independent states (uncleared/cleared/reconciled-locked) give a clean reconciliation UX. + - Provide a reconciliation escape hatch: when computed balance != bank balance, offer a single 'balance adjustment' transaction instead of forcing the user to find every discrepancy. Lock reconciled rows but keep them editable (with re-reconcile). Soft-delete (tombstone) rows so sync/undo and delta queries work. + - Matching is strictly imported<->user-entered: never auto-merge two imported rows or two manual rows. This prevents the matcher from collapsing genuinely distinct transactions and bounds the blast radius of bad matches. + - Offer manual match as the fallback when auto-match fails: let the user pick exactly two transactions and merge them, preserving their chosen category/memo. Don't rely on auto-match alone - bank date/amount quirks (very common with Israeli scrapers) will defeat any fixed window. + +### Bank data aggregators (Plaid, MX, Finicity/Mastercard, Akoya) + PFM apps (Monarch, Copilot, Wave). Key official docs: +- Plaid Enrich API: https://plaid.com/docs/api/products/enrich/ and intro https://plaid.com/docs/enrich/ +- Plaid Transactions states / dedup: https://plaid.com/docs/transactions/transactions-data/ +- Plaid PFC taxonomy: https://plaid.com/documents/transactions-personal-finance-category-taxonomy.csv and blog https://plaid.com/blog/transactions-categorization-taxonomy/ , AI update https://plaid.com/blog/ai-enhanced-transaction-categorization/ +- Plaid parsing engine: https://plaid.com/blog/how-plaid-parses-transaction-data/ , https://plaid.com/blog/transaction-enrichment-engine/ +- Plaid Transfer reconciliation (money-movement product, distinct from PFM transfer detection): https://plaid.com/docs/transfer/reconciling-transfers/ , flow of funds https://plaid.com/docs/transfer/flow-of-funds/ , reading events https://plaid.com/docs/api/products/transfer/reading-transfers/ +- MX Data Enhancement: https://www.mx.com/products/data-enhancement/ , off-platform API https://docs.mx.com/api-reference/more-apis/data-enhancement-off-platform/ +- Finicity/Mastercard Data Enrichment: https://www.finicity.com/manage/transactions/ , https://developer.mastercard.com/open-banking-us/documentation/products/manage/data-enrichment/api/ +- Akoya Transactions / FDX: https://akoya.com/products/transactions , https://docs.akoya.com/guides/transactions +- PFM apps: Monarch https://help.monarch.com/hc/en-us/articles/360048393292-Transfers-and-Credit-Card-Payments ; Copilot https://help.copilot.money/en/articles/3971267-transaction-types and https://help.copilot.money/en/articles/10671434-credit-card-payment-transactions ; Wave transfer/CC-payment help. + +- **Transfer detection:** Two distinct senses of "transfer" exist in this space; keep them separate. + +(1) Aggregator transaction classification (closest to our need): Plaid's Personal Finance Category (PFC) taxonomy classifies transfers via top-level primary categories TRANSFER_IN and TRANSFER_OUT (plus LOAN_PAYMENTS, BANK_FEES). Detailed examples: TRANSFER_OUT_ACCOUNT_TRANSFER, TRANSFER_IN_ACCOUNT_TRANSFER, TRANSFER_IN_DEPOSIT, TRANSFER_IN_CASH_ADVANCES_AND_LOANS, TRANSFER_IN_INVESTMENT_AND_RETIREMENT_FUNDS. Taxonomy is 16 primary / 104 detailed (PFCv2, Dec 2025 added a dozen+ subcategories for income types, repayments/disbursements, fees, transfers). Each personal_finance_category object has primary, detailed, and confidence_level. IMPORTANT: Plaid classifies a transaction as a transfer-type at the single-transaction level (semantic intent) but does NOT, in public docs, pair the two legs across the user's owned accounts. MX similarly has an "internal transfer" category meaning a transaction between two accounts of the same entity, again as a label, not a documented cross-account pairing. + +(2) Actual leg-pairing between two owned accounts is done by the PFM layer, not the aggregator. Documented matching parameters across the industry (per transaction-matching literature and PFM apps): match on amount equal-and-opposite-sign, date proximity (a small window to absorb settlement lag), and description/reference. The exact window and thresholds are NOT publicly documented by Plaid/MX/Finicity/Akoya; PFM apps (Monarch, Copilot) auto-detect candidates and let the user confirm/mark as Internal Transfer. So: aggregators give you a reliable transfer LABEL + stable counterparty; you build the leg-pairing yourself using opposite-sign amount + date-window + same-user-account scoping. + +Plaid Transfer (the money-movement product) reconciliation is a different concept: it matches your bank-statement sweep lines to Plaid records via the first 8 chars of sweep_id appearing on the statement (e.g. "PLAID 6c036ea0 CCD"), with event types like swept, swept_settled, return_swept; no amount tolerance window is documented. +- **Credit-card payment matching:** This is treated as the canonical internal-transfer case by PFM apps, and the consistent rule is: classify both legs as Internal Transfer and EXCLUDE them from spending/income so the underlying purchases (already counted on the card) are not double-counted. + +- Monarch: transactions categorized as "Transfers" are excluded from budget and spending "so that they are not double counted," explicitly citing credit card payments and moving money between accounts. +- Copilot: detects when funds move from checking to pay a credit card; both the checking-side debit and the card-side credit are left as Internal Transfers, "because that money is already accounted for in your budget with individual transactions as they occur." Internal Transfers are excluded from spending budgets. Copilot auto-marks CC payments as Internal Transfer by default; user can override one leg to Regular if they want to budget the payment itself (rare). +- Wave: a transfer cannot exist without both sides recorded; if you categorize one side as a transfer without a match, Wave auto-CREATES the missing offsetting transaction. + +Design implication: the CC bill payment (checking -$X out, card +$X in) must NOT be categorized as spend. Only the individual card purchases count. So the matcher's job is to (a) recognize the bill-payment pair, (b) flag both legs as transfer/excluded, NOT to net them against purchases. Aggregators help by labeling the payment leg TRANSFER_OUT / LOAN_PAYMENTS and providing a financial_institution counterparty, but do not net it for you. +- **Internal linking / data model:** Counterparty / account-linking data model from the aggregators: + +- Plaid Enrich counterparties[]: each has name, type (enum: merchant, financial_institution, payment_app, marketplace, payment_terminal, income_source), entity_id (a unique, STABLE, Plaid-generated ID mapping to the counterparty), plus logo_url, website, phone_number. For European institutions only, counterparty can include account_numbers (IBAN, BIC). The financial_institution and payment_app types are the signal that a transaction is account-to-account / P2P movement rather than merchant spend. +- entity_id is the key linking primitive: stable across transactions, so you can group all transactions to/from the same counterparty (e.g., the same brokerage or the same person via a payment app) without string matching. Plaid markets entity_id for "building custom rules or logic for specific merchants/counterparties." +- Akoya/FDX: standardized transaction fields per the Financial Data Exchange spec; enrichment adds three levels of categorization, merchant name/logo/website/location, regularity (recurring), and payment-processor info. FDX provides standard fields (transaction id, status, amount, memo, payee, checkNumber) useful for identifying account movements. + +Data-model takeaway for us: model accounts as first-class, and model a "transfer link" as an explicit join between two transaction rows (leg A in account X, leg B in account Y) rather than mutating either row. Store a stable counterparty/entity id per transaction so transfers and recurring streams can be grouped. Scope pairing to accounts owned by the same user. +- **Dedup + matching heuristics:** The strongest, most concrete documented dedup pattern is Plaid's pending-vs-posted model (directly relevant to our overlapping-pull problem): + +- Plaid does NOT mutate a pending transaction into a posted one. Instead the posted transaction is a NEW row carrying pending_transaction_id = the transaction_id of the original pending row. Consumers dedup by matching on pending_transaction_id. +- /transactions/sync returns three arrays: added, modified, removed, plus a cursor. When a pending posts, Plaid sends the pending in removed AND the posted in added (with pending_transaction_id set). You must apply all three in order to stay consistent and avoid duplicates. +- Caveats Plaid documents: some institutions (Capital One, USAA) provide no pending data so pending_transaction_id is null; Plaid may also FAIL to match pending↔posted, leaving it absent; and the pending and posted versions "may not necessarily share the same details: their name and amount may change" (e.g., restaurant tip added on settle, gas-station auth hold that vanishes). So amount/name equality alone is unreliable for matching legs. +- Confidence/auto-vs-suggest: Plaid Enrich attaches confidence_level to BOTH personal_finance_category and counterparty, enum VERY_HIGH (>98%), HIGH (>90%), MEDIUM, LOW, UNKNOWN. This is the documented signal to drive auto-apply vs review: auto-apply on VERY_HIGH/HIGH, suggest-for-review on MEDIUM/LOW/UNKNOWN. (Plaid does not publicly document a numeric threshold for transfer LEG-PAIRING specifically, that is our heuristic.) +- MX/Finicity/Akoya use ML + multi-stage cleansing + rule-based logic; specific dedup windows/thresholds are NOT publicly documented. + +For our cross-pull dedup: combine a stable composite hash (account + date + amount + normalized description, our existing approach) with the pending/posted insight, since posted versions can differ in name/amount from pending, a pure exact-hash will create duplicates when a pending settles; allow a small reconciliation step (amount-and-date-window) to retire a pending when its posted version arrives. +- **Reconciliation / UX:** User-correction and undo patterns observed: + +- Copilot: user selects a transaction and chooses "Mark as Internal Transfer" (or reverts to Regular); is offered to apply the change to similar transactions and/or create a name-based rule (exact or partial match) so future imports auto-classify. Manual internal-transfer pairs can be created explicitly. +- Monarch: duplicates are removed via "Edit multiple"; account-level duplicates are resolved by merging accounts; a transfer tool moves transaction/balance history between accounts (move transactions, balances, or both) when switching data providers for the same institution. +- Wave: if you mark one side as a transfer with no match, it auto-creates the offsetting transaction; both sides must exist for a transfer to be valid. +- Plaid does not provide PFM reconciliation UX; it provides the data (states, confidence) and leaves correction to the app. + +Design implications for us: (1) make transfer pairing reversible, an explicit link row that can be unlinked without deleting either transaction; (2) when the user confirms/corrects a classification, offer "apply to similar" + create a persistent rule keyed on normalized description/counterparty; (3) provide a merge/move tool for the data-provider-switch case (same institution re-linked produces parallel accounts/duplicates); (4) treat auto-detection as suggestions for low-confidence matches and only auto-apply high-confidence ones, since amounts/names can legitimately change between pending and posted. +- **Takeaways:** + - Separate two concepts: aggregators classify a single transaction as transfer-type (Plaid PFC TRANSFER_IN/TRANSFER_OUT, MX 'internal transfer' label), but they do NOT pair the two legs across owned accounts. Leg-pairing is the PFM layer's job and is our responsibility. + - Use a confidence-driven auto-vs-suggest gate. Plaid Enrich's documented enum is VERY_HIGH(>98%), HIGH(>90%), MEDIUM, LOW, UNKNOWN on both category and counterparty. Mirror this: auto-apply transfer pairing / categorization at high confidence, surface for user review otherwise. + - Pair transfer legs with opposite-sign equal amount + a date-proximity window + same-user account scope; reference/description as a tiebreaker. The exact window is not publicly documented by any aggregator, so pick a small window (commonly a few days) to absorb settlement lag and make it tunable. + - Credit card bill payments are the canonical internal transfer: classify BOTH legs as transfer and EXCLUDE from spending/income. Do not net them against purchases. The card purchases are the spend; the payment is just money movement (Monarch and Copilot both do exactly this to prevent double-counting). + - Model transfer pairing as an explicit, reversible LINK row joining two transaction rows, not by mutating/deleting either leg. This makes unlink/undo and audit trivial and keeps both account ledgers intact. + - Do not assume the two legs are byte-identical. Plaid documents that pending vs posted versions can differ in name AND amount (tips, auth holds), so dedup and pairing must tolerate small amount/name drift rather than require exact equality. + - Adopt Plaid's pending/posted dedup mental model: a posted transaction is a NEW record that references the pending one (pending_transaction_id); reconcile by retiring the pending when its posted version arrives. For our count-based hash dedup, add a settlement-reconciliation pass so a settled posted txn doesn't become a duplicate of its pending. + - Store a STABLE counterparty/entity id per transaction (Plaid entity_id model) so transfers, recurring streams, and 'apply to similar' rules can group by entity instead of brittle string matching. Counterparty.type (financial_institution / payment_app) is a strong transfer signal. + - Make corrections sticky: when a user confirms or fixes a transfer/category, offer 'apply to similar' and persist a rule keyed on normalized description/counterparty so future imports auto-classify (Copilot's exact/partial name-rule pattern). + - Provide a merge/move tool for the re-link/provider-switch case. Re-linking the same institution produces parallel accounts and duplicate history; offer moving transactions and/or balance history between accounts and de-duping at the account level (Monarch's pattern). + +### Monarch Money official Help Center (help.monarch.com, formerly help.monarchmoney.com) plus credible third-party guides. Primary pages used: +- Transfers and Credit Card Payments: https://help.monarch.com/hc/en-us/articles/360048393292-Transfers-and-Credit-Card-Payments (Last Updated Nov 14, 2025) +- Hiding or Unhiding Transactions: https://help.monarch.com/hc/en-us/articles/4405041904916-Hiding-or-Unhiding-Transactions (Last Updated Dec 16, 2025) +- Creating Transaction Rules: https://help.monarch.com/hc/en-us/articles/360048393372-Creating-Transaction-Rules (Last Updated Jun 4, 2026) +- Troubleshooting Duplicate Transactions: https://help.monarch.com/hc/en-us/articles/32110313427604-Troubleshooting-Duplicate-Transactions (Last Updated Mar 20, 2025) +- Tips & Tricks for Using Monarch: https://help.monarch.com/hc/en-us/articles/28953573066260-Tips-Tricks-for-Using-Monarch +- Third-party guide: https://www.evolvingmoneycoaching.com/avoid-these-mistakes-in-monarch-money-and-an-explanation-on-transfers/ +Note: help.monarch.com is behind Cloudflare bot protection (WebFetch returned 403); content was retrieved via a headed browser session, so quotes below are verbatim from the live official pages. + +- **Transfer detection:** Monarch's transfer handling is CATEGORY-BASED EXCLUSION, not a two-sided pairing/matching engine. Official definition: "A Transfer in Monarch is when you move money from one of your accounts to another" (e.g., checking to savings, or paying a credit card from a bank account). The mechanism: any transaction whose category is in the "Transfers" group is excluded from budget, cash flow, and spending totals. "Anything categorized under the 'Transfers' category is excluded from your budget and spending so that it is not double counted." Detection of which transactions ARE transfers happens via Monarch's auto-categorization on sync ("Monarch will automatically apply categories to each transaction as it arrives"), which assigns transfer-like transactions (Zelle, Venmo, checks, internal moves) to a Transfers category. There is NO publicly documented algorithm that automatically PAIRS the two sides of a transfer (the outflow on account A with the inflow on account B) using an amount/date matching window. Each leg is independently categorized as a transfer and independently excluded; the exclusion is per-transaction by category, not by detecting a matched pair. A known weakness called out in the official-adjacent guidance: person-to-person payments (Zelle/Venmo/checks) are "often auto-categorized by Monarch as 'Transfers'" even when the money actually left to a third party, so users must manually recategorize those as expenses. Reverse problem also exists: real internal moves can land in the wrong category and need manual fixing. Bottom line for our design: Monarch leans on category semantics + auto-categorization + user rules rather than a confidence-scored transfer-pair detector. +- **Credit-card payment matching:** This is the clearest, best-documented part. Monarch does NOT "match" a credit card payment to specific underlying purchases. Instead it relies on a clean accounting split: (1) the original purchase is recorded immediately as an EXPENSE on the credit card account when the charge posts ("When you place a charge on your credit card (like $50 at a gas station), Monarch tracks that original charge immediately as an expense"); (2) the later bill payment is categorized as a TRANSFER (the special "Credit Card Payment" transfer category), NOT a new expense ("Monarch treats this as a Credit Card Payment (transfer), not a new expense. This is true whether it's a partial payment or the entire balance"). Because the payment sits in an excluded Transfers category, spending is counted exactly once (on the original purchases), never again on the payment. Official worded warning: "Monarch counts the original purchase as spending. Treating the credit card payment as a transfer ensures it's not counted a second time. Otherwise... it would look like you spent $100 instead of $50!" Important nuance: a single bill payment produces TWO transactions that should BOTH be categorized as Credit Card Payment / transfer: "The payment out of your bank account (usually appears as a debit)" and "The payment into your credit card (usually appears as a credit)." So Monarch double-excludes both legs by category rather than linking them. The official accounting/net-worth explanation makes clear the payment does not erase the expense; it only settles the liability. Caveat surfaced in their Pay Down Goals docs: if you ALSO track the same payment via debt Pay Down contributions while categorizing purchases as expenses, you can re-introduce double counting; Monarch tells users to pick one method. +- **Internal linking / data model:** No automatic two-sided transfer link/pair object is publicly documented. Monarch does NOT (per docs) create a persisted "this outflow is matched to that inflow" linked-pair record for ordinary transfers. The data-model primitives that ARE documented and relevant: +1. Category group "Transfers" containing system categories: "Transfer" (general internal moves), "Credit Card Payment" (CC-specific), and, when investments are enabled, "Buy" and "Sell". All are flagged as excluded-from-budget/spending. Users can also create custom transfer categories. So the "is this a transfer" signal is modeled as a category attribute, not a pairing. +2. A per-transaction "hidden" boolean (excludes from list/reports/budget/cash flow but not balances/net worth). +3. Goal linking is the one place explicit linking exists: rules can "Link to goal", "Link to save up goal", and "Link to pay down goal" ("Link transactions from liability accounts to debt paydown tracking"), which associates an individual transaction with a goal. This is the closest documented "link a transaction to something else" data model, but it links transaction-to-goal, not transaction-to-transaction. +4. Account-level merge: duplicate ACCOUNTS can be merged; there is a "Transfer Balance and/or Transaction History to Another Account" tool that moves history between accounts (old account loses data, new gains it), account-level, not transfer-pair-level. +Implication for our design: Monarch's model is "tag each leg with an exclude-eligible category" rather than "store a transfer_pair (debit_txn_id, credit_txn_id)". If we want true pair linking we'd be going beyond what Monarch documents. +- **Dedup + matching heuristics:** Duplicate prevention is largely delegated to the data source and account hygiene, not a documented fuzzy in-app matcher. Official "Troubleshooting Duplicate Transactions" guidance, in order: (1) verify on the bank's own website that the transaction is not actually duplicated there; (2) if it shows as "Pending" it "may be a simple error with the bank or merchant"; (3) confirm you have not added the same ACCOUNT multiple times (duplicate accounts are the main in-app cause) and merge duplicate accounts; (4) only if none of those apply, contact support with institution name, transaction name, date, and amount. So there is NO publicly documented amount/date-window fuzzy dedup heuristic and NO documented auto-merge of two near-identical transactions, and NO confidence/auto-vs-suggest threshold. Related pending behavior: a Preferences setting "Allow edits to pending transactions" exists, but Monarch warns "matching pending transactions to their posted version is not always possible", i.e., they acknowledge pending-to-posted matching is imperfect and is the source of some duplicates (a hidden pending txn "may reappear once the bank processes it... finalized transactions sometimes come through as a new entry"). Manual dedup path: "Edit Multiple" lets users select and delete duplicates in bulk. Net: amount/date/fuzzy matching window for dedup is NOT publicly documented; Monarch treats most dupes as upstream/account-config problems. +- **Reconciliation / UX:** User-correction and "hide from reports" UX is well documented. Marking/unmarking as transfer = recategorize: change a transaction's category into (or out of) the Transfers group via the transaction detail (web: expand with ">" then edit category; mobile: tap transaction). Bulk: "Edit Multiple" (web) / checkmark icon (mobile) to recategorize many at once. Automation: Rules ("Settings > Rules") can auto-set category to a transfer category and/or hide, matched on original statement, merchant name, amount (debit/credit, equals/range), category, account, owner, business; all criteria are AND-combined; multiple statement/merchant conditions are OR-combined; rules run in listed order and can be reordered; rules can be applied retroactively to existing transactions ("apply to X existing transactions"). "Quick rules" widget auto-pops in the lower-right whenever you edit a transaction, letting you turn a one-off correction into a rule. HIDE FROM REPORTS: a separate, independent control from category. "Hiding a transaction does not delete it. It simply removes the transaction from your main transactions list, cash flow, reports, and budget calculations." Crucially, "Hiding a transaction removes it from your transaction list, reports, and budget, but it does not impact your account balances or net worth calculations." Hide UI: transaction detail "eye" icon (web) / Hide toggle (mobile); bulk via Edit Multiple; or via Rules (Hide transaction toggle). UNDO/unhide: filter Transactions by Other > "Hidden Only", then untoggle the eye/Hide (individually or via Edit Multiple). Caveats: account-level hidden transactions can only be unhidden at the account level; hiding does NOT clear "Needs Review" status, and review filters deliberately include hidden transactions; hidden pending transactions may reappear once posted. Splits: rules and manual edits support splitting a transaction (by % or $), useful for partial reimbursements. +- **Takeaways:** + - Prefer category-based exclusion as the core mechanism. Monarch prevents double-counting by putting transfers/CC-payments in an excluded 'Transfers' category group rather than by detecting and linking two-sided pairs. This is simpler and robust, and it is what a market leader actually ships. + - Treat credit card payments as transfers, never as expenses, and keep the original purchases as the single source of spend. The CC bill payment (and its mirror leg) is excluded; spending is counted once at purchase time. This cleanly avoids the classic '$50 spent looks like $100' bug. + - Expect TWO legs per transfer and exclude BOTH by category. Monarch explicitly tells users both the bank-side debit and the card-side credit should be the 'Credit Card Payment' category. If you exclude only one leg you reintroduce double counting, so dedup/exclusion must be per-leg, not per-pair. + - Separate 'is a transfer' (category) from 'hide from reports' (a boolean). Monarch models these as two orthogonal controls. Hiding removes a txn from list/reports/budget/cash flow but NOT from balances/net worth. We should mirror that split so balances stay accurate even when something is excluded from spend. + - Net worth must be computed from balances, not from the budget/spend ledger. Monarch's own worked example shows the expense (not the transfer legs) is what moves net worth; paying the card settles a liability and nets to zero on balances. Keep expense-tracking and balance/net-worth tracking as distinct calculations. + - There is NO publicly documented amount+date matching window, no fuzzy transfer-pair matcher, and no auto-vs-suggest confidence threshold. Do not assume an industry-standard window exists; Monarch ships without one. If we build true pair-matching it is a differentiator, but we should design it ourselves (e.g., opposite-sign equal amounts within N days) rather than copy a Monarch spec that does not exist publicly. + - Auto-categorization plus user rules carry most of the load. Monarch auto-assigns categories on sync, then lets users codify corrections as rules (match on raw original statement/merchant/amount/account, AND-combined, retroactively applicable, reorderable, with a 'quick rule' prompt on every manual edit). A rule engine that can set category=transfer and/or hide is high-value and reusable. + - Match on the raw bank 'original statement', not the cleaned merchant name, for stable rules. Monarch explicitly recommends this because the original statement rarely changes while merchant display names get re-cleaned. Good guidance for our dedup keys and rule matching too. + - Person-to-person payments are the main false-positive for transfer detection. Zelle/Venmo/checks often auto-classify as transfers but are real outflows; build an easy review/recategorize path and consider not auto-excluding P2P by default. + - Duplicate-transaction handling is mostly upstream/account hygiene, not in-app fuzzy merge. Monarch's documented fixes are: check the bank site, watch pending-vs-posted churn, and de-duplicate ACCOUNTS (a duplicated connection is the top cause). Our dedup should strongly guard against duplicate account connections and pending->posted re-entry, and offer bulk manual delete as the safety valve. + diff --git a/drizzle.config.ts b/drizzle.config.ts new file mode 100644 index 0000000..ea2b62a --- /dev/null +++ b/drizzle.config.ts @@ -0,0 +1,12 @@ +import { defineConfig } from "drizzle-kit"; + +// Drizzle is used as the typed query layer over the existing better-sqlite3 +// connection. Schema is generated by introspecting a fully-migrated database +// (the .sql migration runner in src/server/db/migrate.ts remains the source of +// truth for DDL). Regenerate with: bun run db:pull +export default defineConfig({ + dialect: "sqlite", + schema: "./src/server/db/schema.ts", + out: "./.context/drizzle-introspect", + dbCredentials: { url: ".context/introspect.db" }, +}); diff --git a/eslint.config.mjs b/eslint.config.mjs index 05e726d..642dc2e 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -12,6 +12,8 @@ const eslintConfig = defineConfig([ "out/**", "build/**", "next-env.d.ts", + // Conductor scratch directory (gitignored, not part of the source tree). + ".context/**", ]), ]); diff --git a/knip.json b/knip.json new file mode 100644 index 0000000..c2e288b --- /dev/null +++ b/knip.json @@ -0,0 +1,12 @@ +{ + "$schema": "https://unpkg.com/knip@5/schema.json", + "entry": [ + "src/app/**/{page,layout,route,not-found,loading,error,global-error,template,default,opengraph-image,icon,sitemap,robots}.{ts,tsx}", + "src/i18n/routing.ts", + "scripts/**/*.{mjs,ts}" + ], + "project": ["src/**/*.{ts,tsx}", "scripts/**/*.{mjs,js,ts}"], + "include": ["files", "dependencies", "unlisted", "unresolved", "binaries"], + "ignoreDependencies": ["tailwindcss", "tw-animate-css", "shadcn"], + "ignore": ["src/server/db/queries/excluded-merchants.ts"] +} diff --git a/menubar/README.md b/menubar/README.md deleted file mode 100644 index 8435c0b..0000000 --- a/menubar/README.md +++ /dev/null @@ -1,38 +0,0 @@ -# Spent menu bar / tray app - -Tiny native controller for the always-on Spent server. Shows status in the menu bar (macOS) or notification area (Windows), with one-click Open dashboard / Sync now / Start–Stop service / Quit. - -Both apps poll `http://127.0.0.1:41234/api/health` every 5 seconds and dim the icon when the server is unreachable. The browser action opens `http://spent.localhost:41234`. - -## macOS - -```bash -cd menubar/mac -./build.sh -cp -R build/Spent.app ~/Applications/ -``` - -First launch: right-click `Spent.app` → Open (one-time Gatekeeper prompt). Add to **System Settings → General → Login Items** to auto-start. - -Requires Xcode Command Line Tools (`xcode-select --install`). Builds with Swift, ad-hoc signed. - -## Windows - -```powershell -cd menubar\windows -.\build.ps1 -mkdir $env:LOCALAPPDATA\Programs\Spent -Copy-Item build\Spent.exe $env:LOCALAPPDATA\Programs\Spent\ -``` - -First launch: SmartScreen may say "Windows protected your PC" — click **More info** → **Run anyway** (binary is unsigned). - -To auto-start at login, drop a shortcut to `Spent.exe` into `shell:startup` (run from Win+R). - -Requires .NET 8 SDK (`winget install Microsoft.DotNet.SDK.8`). Builds a self-contained single-file `Spent.exe` (~30 MB) with WPF + `H.NotifyIcon.Wpf`. - -## What both apps assume - -- The Spent server is already installed as a background service via `npm run service:install` (LaunchAgent on Mac, scheduled task on Windows). -- The service binds to `127.0.0.1:41234`. The tray apps will not connect to any other host. -- Start/Stop calls go to `launchctl bootstrap|bootout com.spent.app` (Mac) or `schtasks /Run|/End /TN Spent` (Windows). diff --git a/menubar/mac/.gitignore b/menubar/mac/.gitignore deleted file mode 100644 index 9e58134..0000000 --- a/menubar/mac/.gitignore +++ /dev/null @@ -1,4 +0,0 @@ -.build/ -build/ -.swiftpm/ -Package.resolved diff --git a/menubar/mac/Package.swift b/menubar/mac/Package.swift deleted file mode 100644 index a1ccdd8..0000000 --- a/menubar/mac/Package.swift +++ /dev/null @@ -1,13 +0,0 @@ -// swift-tools-version:5.9 -import PackageDescription - -let package = Package( - name: "Spent", - platforms: [.macOS(.v13)], - targets: [ - .executableTarget( - name: "Spent", - path: "Sources/Spent" - ) - ] -) diff --git a/menubar/mac/Resources/Info.plist b/menubar/mac/Resources/Info.plist deleted file mode 100644 index a3fb439..0000000 --- a/menubar/mac/Resources/Info.plist +++ /dev/null @@ -1,41 +0,0 @@ - - - - - CFBundleExecutable - Spent - CFBundleIdentifier - com.spent.menubar - CFBundleName - Spent - CFBundleDisplayName - Spent - CFBundleVersion - 1 - CFBundleShortVersionString - 0.1.0 - CFBundlePackageType - APPL - CFBundleInfoDictionaryVersion - 6.0 - LSMinimumSystemVersion - 13.0 - LSUIElement - - NSHumanReadableCopyright - Spent — local only. - NSAppTransportSecurity - - NSExceptionDomains - - 127.0.0.1 - - NSExceptionAllowsInsecureHTTPLoads - - NSIncludesSubdomains - - - - - - diff --git a/menubar/mac/Sources/Spent/SpentApp.swift b/menubar/mac/Sources/Spent/SpentApp.swift deleted file mode 100644 index ae6983f..0000000 --- a/menubar/mac/Sources/Spent/SpentApp.swift +++ /dev/null @@ -1,380 +0,0 @@ -import SwiftUI -import AppKit -import Foundation - -// In-app network access is restricted to 127.0.0.1 by NSAppTransportSecurity -// in Info.plist. Do not change healthURL/syncURL/sameOrigin without re-reviewing -// that. openInBrowserURL is handed to NSWorkspace and opens in the user's -// browser, which is not subject to ATS, so it can use the friendly hostname. -private let openInBrowserURL = URL(string: "http://spent.localhost:41234")! -private let healthURL = URL(string: "http://127.0.0.1:41234/api/health")! -private let syncURL = URL(string: "http://127.0.0.1:41234/api/sync")! -private let sameOrigin = "http://127.0.0.1:41234" -private let launchAgentLabel = "com.spent.app" -private let launchAgentPlist = - ("~/Library/LaunchAgents/com.spent.app.plist" as NSString).expandingTildeInPath - -// MARK: - Logo -// Vector copy of public/logo_*.svg (149x184 viewBox, stroke-width 9). -// Drawn in code so the same shape works in the menu bar and the popover. - -private struct LogoShape: Shape { - func path(in rect: CGRect) -> Path { - let s = min(rect.width / 149, rect.height / 184) - let dx = rect.minX + (rect.width - 149 * s) / 2 - let dy = rect.minY + (rect.height - 184 * s) / 2 - - func pt(_ x: CGFloat, _ y: CGFloat) -> CGPoint { - CGPoint(x: dx + x * s, y: dy + y * s) - } - - var p = Path() - - p.addEllipse(in: CGRect( - x: dx + (32 - 27.5) * s, y: dy + (53.6221 - 27.5) * s, - width: 55 * s, height: 55 * s - )) - - p.move(to: pt(32.5, 26.1221)) - p.addCurve(to: pt(70.5, 23.1221), - control1: pt(32.5, 26.1221), control2: pt(53.5, 30.1221)) - p.addCurve(to: pt(100, 3.12207), - control1: pt(87.5, 16.1221), control2: pt(100, 3.12207)) - p.addLine(to: pt(49.5, 138.122)) - - p.move(to: pt(13.5, 158.122)) - p.addCurve(to: pt(75, 179.122), - control1: pt(13.5, 158.122), control2: pt(31, 180.622)) - p.addCurve(to: pt(133.5, 158.122), - control1: pt(119, 177.622), control2: pt(133.5, 158.122)) - - p.addEllipse(in: CGRect( - x: dx + (117 - 27.5) * s, y: dy + (85.6221 - 27.5) * s, - width: 55 * s, height: 55 * s - )) - - return p - } -} - -private struct LogoView: View { - // Height in pt of the rendered logo. Width is derived from the 149:184 aspect. - var height: CGFloat - var strokeWidth: CGFloat - - var body: some View { - LogoShape() - .stroke(style: StrokeStyle(lineWidth: strokeWidth, - lineCap: .butt, lineJoin: .miter)) - .frame(width: height * 149.0 / 184.0, height: height) - } -} - -// Renders the logo as a template NSImage so macOS auto-tints it for the menu bar -// (white on dark menu bars, black on light). SwiftUI's `.foregroundStyle(.primary)` -// is unreliable inside `MenuBarExtra(label:)` for LSUIElement apps — template image -// is the canonical AppKit/macOS approach. -private func makeLogoTemplateImage(height: CGFloat, strokeWidth: CGFloat) -> NSImage { - let size = NSSize(width: height * 149.0 / 184.0, height: height) - let image = NSImage(size: size, flipped: true) { rect in - guard let ctx = NSGraphicsContext.current?.cgContext else { return false } - let path = LogoShape().path(in: rect).cgPath - ctx.setLineWidth(strokeWidth) - ctx.setLineCap(.butt) - ctx.setLineJoin(.miter) - ctx.setStrokeColor(NSColor.black.cgColor) - ctx.addPath(path) - ctx.strokePath() - return true - } - image.isTemplate = true - return image -} - -// MARK: - Status model - -@MainActor -final class StatusModel: ObservableObject { - @Published var isOnline = false - @Published var version = "" - - private var pollTask: Task? - - init() { - pollTask = Task { await pollLoop() } - } - - deinit { - pollTask?.cancel() - } - - private func pollLoop() async { - while !Task.isCancelled { - await pollOnce() - try? await Task.sleep(for: .seconds(5)) - } - } - - func pollOnce() async { - var req = URLRequest(url: healthURL) - req.timeoutInterval = 2.0 - req.cachePolicy = .reloadIgnoringLocalCacheData - do { - let (data, response) = try await URLSession.shared.data(for: req) - guard let http = response as? HTTPURLResponse, http.statusCode == 200 else { - isOnline = false - return - } - if let json = try JSONSerialization.jsonObject(with: data) as? [String: Any] { - isOnline = (json["ok"] as? Bool) ?? false - version = (json["version"] as? String) ?? "" - } - } catch { - isOnline = false - } - } -} - -// MARK: - Actions - -private func openSpent() { - NSWorkspace.shared.open(openInBrowserURL) -} - -private func syncNow() { - var req = URLRequest(url: syncURL) - req.httpMethod = "POST" - req.setValue(sameOrigin, forHTTPHeaderField: "Origin") - req.timeoutInterval = 5.0 - URLSession.shared.dataTask(with: req).resume() -} - -private func runLaunchctl(_ args: [String]) -> Int32 { - let task = Process() - task.launchPath = "/bin/launchctl" - task.arguments = args - task.standardOutput = Pipe() - task.standardError = Pipe() - do { - try task.run() - task.waitUntilExit() - return task.terminationStatus - } catch { - return -1 - } -} - -private func startService() { - _ = runLaunchctl(["bootstrap", "gui/\(getuid())", launchAgentPlist]) -} - -private func stopService() { - _ = runLaunchctl(["bootout", "gui/\(getuid())/\(launchAgentLabel)"]) -} - -private func restartService() { - _ = runLaunchctl(["kickstart", "-k", "gui/\(getuid())/\(launchAgentLabel)"]) -} - -// Deactivating the app closes the MenuBarExtra(.window) popover. -private func dismissPopover() { - NSApp.deactivate() -} - -// MARK: - Menu row - -private struct MenuRow: View { - let icon: String - let title: String - var shortcut: String? = nil - var tint: Color = .primary - var isEnabled: Bool = true - let action: () -> Void - - @State private var isHovered = false - - var body: some View { - Button(action: action) { - HStack(spacing: 10) { - Image(systemName: icon) - .font(.system(size: 13, weight: .medium)) - .frame(width: 16, alignment: .center) - Text(title) - .font(.system(size: 13, weight: .regular)) - Spacer(minLength: 8) - if let shortcut = shortcut { - Text(shortcut) - .font(.system(size: 12, weight: .regular)) - .foregroundStyle(.secondary) - } - } - .foregroundStyle(tint) - .padding(.horizontal, 10) - .padding(.vertical, 6) - .frame(maxWidth: .infinity, alignment: .leading) - .background( - RoundedRectangle(cornerRadius: 6, style: .continuous) - .fill(isHovered && isEnabled - ? Color.primary.opacity(0.08) - : Color.clear) - ) - .contentShape(Rectangle()) - .opacity(isEnabled ? 1.0 : 0.4) - } - .buttonStyle(.plain) - .disabled(!isEnabled) - .onHover { hovering in - isHovered = hovering && isEnabled - } - } -} - -// MARK: - App - -@main -struct SpentMenuBarApp: App { - @StateObject private var model = StatusModel() - - private static let menuBarIcon = makeLogoTemplateImage(height: 18, strokeWidth: 1.6) - - var body: some Scene { - MenuBarExtra { - PopoverContent(model: model) - } label: { - Image(nsImage: Self.menuBarIcon) - .opacity(model.isOnline ? 1.0 : 0.45) - } - .menuBarExtraStyle(.window) - } -} - -private struct PopoverContent: View { - @ObservedObject var model: StatusModel - - var body: some View { - VStack(spacing: 0) { - header - Divider() - actionSection - Divider() - serviceSection - Divider() - quitSection - } - .frame(width: 264) - } - - // MARK: header - - private var header: some View { - HStack(spacing: 12) { - LogoView(height: 30, strokeWidth: 1.8) - .foregroundStyle(.primary) - VStack(alignment: .leading, spacing: 3) { - Text("Spent") - .font(.system(size: 14, weight: .semibold)) - HStack(spacing: 5) { - Circle() - .fill(model.isOnline ? Color.green : Color.secondary) - .frame(width: 6, height: 6) - Text(statusText) - .font(.system(size: 11)) - .foregroundStyle(.secondary) - } - } - Spacer() - } - .padding(.horizontal, 14) - .padding(.top, 14) - .padding(.bottom, 12) - } - - // MARK: sections - - private var actionSection: some View { - VStack(spacing: 1) { - MenuRow( - icon: "arrow.up.right.square", - title: "Open dashboard", - shortcut: "⌘O", - isEnabled: model.isOnline, - action: { - openSpent() - dismissPopover() - } - ) - .keyboardShortcut("o") - - MenuRow( - icon: "arrow.triangle.2.circlepath", - title: "Sync now", - shortcut: "⌘S", - isEnabled: model.isOnline, - action: { - syncNow() - dismissPopover() - } - ) - .keyboardShortcut("s") - } - .padding(6) - } - - @ViewBuilder - private var serviceSection: some View { - VStack(spacing: 1) { - if model.isOnline { - MenuRow( - icon: "arrow.clockwise", - title: "Restart service", - action: { - restartService() - Task { await model.pollOnce() } - } - ) - MenuRow( - icon: "stop.fill", - title: "Stop service", - tint: .red, - action: { - stopService() - Task { await model.pollOnce() } - } - ) - } else { - MenuRow( - icon: "play.fill", - title: "Start service", - tint: .accentColor, - action: { - startService() - Task { await model.pollOnce() } - } - ) - } - } - .padding(6) - } - - private var quitSection: some View { - VStack(spacing: 1) { - MenuRow( - icon: "power", - title: "Quit menu bar", - shortcut: "⌘Q", - action: { - NSApplication.shared.terminate(nil) - } - ) - .keyboardShortcut("q") - } - .padding(6) - } - - private var statusText: String { - if model.isOnline { - return model.version.isEmpty ? "Running" : "Running · v\(model.version)" - } - return "Stopped" - } -} diff --git a/menubar/mac/build.sh b/menubar/mac/build.sh deleted file mode 100755 index 22029b3..0000000 --- a/menubar/mac/build.sh +++ /dev/null @@ -1,57 +0,0 @@ -#!/usr/bin/env bash -set -euo pipefail - -# Build the Spent menu bar app: a tiny SwiftUI MenuBarExtra controller -# for the always-on Next.js server. Network access is restricted to -# 127.0.0.1 only (see Resources/Info.plist). - -cd "$(dirname "$0")" - -if ! command -v swift >/dev/null 2>&1; then - echo "error: swift not found. Install Xcode Command Line Tools:" - echo " xcode-select --install" - exit 1 -fi - -OUT_DIR="build" -APP_DIR="$OUT_DIR/Spent.app" - -if [[ "${SPENT_UNIVERSAL:-}" == "1" ]]; then - SWIFT_ARGS=(-c release --arch x86_64 --arch arm64) - echo "Building release binary (universal)..." -else - SWIFT_ARGS=(-c release) - echo "Building release binary..." -fi - -swift build "${SWIFT_ARGS[@]}" - -BIN_DIR=$(swift build "${SWIFT_ARGS[@]}" --show-bin-path) -BIN_PATH="$BIN_DIR/Spent" - -if [[ ! -f "$BIN_PATH" ]]; then - echo "error: expected binary at $BIN_PATH not found" - exit 1 -fi - -echo "Assembling $APP_DIR" -rm -rf "$APP_DIR" -mkdir -p "$APP_DIR/Contents/MacOS" -mkdir -p "$APP_DIR/Contents/Resources" - -cp "$BIN_PATH" "$APP_DIR/Contents/MacOS/Spent" -chmod +x "$APP_DIR/Contents/MacOS/Spent" -cp Resources/Info.plist "$APP_DIR/Contents/Info.plist" - -# Ad-hoc sign so Gatekeeper at least recognizes the binary as signed by -# the current user. This is NOT a real Developer ID signature; users will -# still see the first-launch Gatekeeper prompt and need to right-click → Open. -if command -v codesign >/dev/null 2>&1; then - codesign --force --sign - "$APP_DIR" >/dev/null 2>&1 || true -fi - -echo "" -echo "Built: $APP_DIR" -echo "" -echo "To install: cp -R $APP_DIR ~/Applications/" -echo "First launch: right-click Spent.app → Open (one-time Gatekeeper prompt)." diff --git a/menubar/windows/.gitignore b/menubar/windows/.gitignore deleted file mode 100644 index 19bd3e3..0000000 --- a/menubar/windows/.gitignore +++ /dev/null @@ -1,4 +0,0 @@ -bin/ -obj/ -build/ -*.user diff --git a/menubar/windows/Constants.cs b/menubar/windows/Constants.cs deleted file mode 100644 index 78f5f42..0000000 --- a/menubar/windows/Constants.cs +++ /dev/null @@ -1,14 +0,0 @@ -namespace Spent; - -// Network access is intentionally limited to 127.0.0.1 for health/sync. -// OpenInBrowserUrl is handed to ShellExecute so the user's browser opens -// the friendly hostname; that's not subject to this app's HTTP client. -internal static class Constants -{ - public const string OpenInBrowserUrl = "http://spent.localhost:41234"; - public const string HealthUrl = "http://127.0.0.1:41234/api/health"; - public const string SyncUrl = "http://127.0.0.1:41234/api/sync"; - public const string SameOrigin = "http://127.0.0.1:41234"; - public const string TaskName = "Spent"; - public const int PopupWidth = 264; -} diff --git a/menubar/windows/Program.cs b/menubar/windows/Program.cs deleted file mode 100644 index 20505f5..0000000 --- a/menubar/windows/Program.cs +++ /dev/null @@ -1,14 +0,0 @@ -using System.Windows.Forms; - -namespace Spent; - -internal static class Program -{ - [STAThread] - private static void Main() - { - ApplicationConfiguration.Initialize(); - using var tray = new SpentTray(); - Application.Run(); - } -} diff --git a/menubar/windows/Resources/spent.ico b/menubar/windows/Resources/spent.ico deleted file mode 100644 index b34ec0e..0000000 Binary files a/menubar/windows/Resources/spent.ico and /dev/null differ diff --git a/menubar/windows/Services.cs b/menubar/windows/Services.cs deleted file mode 100644 index a71d2b4..0000000 --- a/menubar/windows/Services.cs +++ /dev/null @@ -1,79 +0,0 @@ -using System.Diagnostics; -using System.Net.Http; - -namespace Spent; - -// Service controls. Mirror the Mac app: openSpent / syncNow / startService / stopService. -// Start/stop drive the Spent scheduled task created by `npm run service:install`. -internal static class Services -{ - public static void OpenSpent() - { - try - { - Process.Start(new ProcessStartInfo - { - FileName = Constants.OpenInBrowserUrl, - UseShellExecute = true, - }); - } - catch - { - // Browser launch failed; nothing useful to show in the tray UI. - } - } - - public static async void SyncNow() - { - using var http = new HttpClient { Timeout = TimeSpan.FromSeconds(5) }; - using var req = new HttpRequestMessage(HttpMethod.Post, Constants.SyncUrl); - req.Headers.TryAddWithoutValidation("Origin", Constants.SameOrigin); - try - { - // Fire-and-forget; the server streams progress over SSE which the - // tray app doesn't render. The dashboard is the right place for that. - using var _ = await http.SendAsync(req, HttpCompletionOption.ResponseHeadersRead).ConfigureAwait(false); - } - catch - { - // Best effort; the icon polling will reflect failures. - } - } - - public static void StartService() - { - RunSchtasks("/Run", "/TN", Constants.TaskName); - } - - public static void StopService() - { - RunSchtasks("/End", "/TN", Constants.TaskName); - } - - public static void RestartService() - { - RunSchtasks("/End", "/TN", Constants.TaskName); - RunSchtasks("/Run", "/TN", Constants.TaskName); - } - - private static void RunSchtasks(params string[] args) - { - try - { - using var p = Process.Start(new ProcessStartInfo - { - FileName = "schtasks.exe", - Arguments = string.Join(" ", args), - UseShellExecute = false, - CreateNoWindow = true, - RedirectStandardOutput = true, - RedirectStandardError = true, - }); - p?.WaitForExit(5000); - } - catch - { - // The task may not be installed yet (run `npm run service:install`). - } - } -} diff --git a/menubar/windows/Spent.csproj b/menubar/windows/Spent.csproj deleted file mode 100644 index 5ac1b7d..0000000 --- a/menubar/windows/Spent.csproj +++ /dev/null @@ -1,31 +0,0 @@ - - - - WinExe - net8.0-windows - true - enable - latest - enable - app.manifest - Resources\spent.ico - Spent - Spent - 0.1.0 - Spent - Spent tray controller. Local-only personal finance. - - - - true - true - win-x64 - true - true - - - - - - - diff --git a/menubar/windows/SpentTray.cs b/menubar/windows/SpentTray.cs deleted file mode 100644 index f713b83..0000000 --- a/menubar/windows/SpentTray.cs +++ /dev/null @@ -1,73 +0,0 @@ -using System.Drawing; -using System.Reflection; -using System.Windows.Forms; - -namespace Spent; - -internal sealed class SpentTray : IDisposable -{ - private readonly NotifyIcon _notify; - private readonly StatusModel _status; - - public SpentTray() - { - _notify = new NotifyIcon - { - Icon = LoadIcon(), - Text = "Spent", - ContextMenuStrip = BuildMenu(), - Visible = true, - }; - - _notify.MouseClick += (_, e) => - { - if (e.Button == MouseButtons.Left) - { - Services.OpenSpent(); - } - }; - - _status = new StatusModel(); - _status.PropertyChanged += (_, _) => UpdateTooltip(); - _status.Start(); - UpdateTooltip(); - } - - private static Icon LoadIcon() - { - var asm = Assembly.GetExecutingAssembly(); - using var stream = asm.GetManifestResourceStream("Spent.Resources.spent.ico") - ?? throw new InvalidOperationException("Embedded resource 'Spent.Resources.spent.ico' not found."); - return new Icon(stream); - } - - private static ContextMenuStrip BuildMenu() - { - var menu = new ContextMenuStrip(); - menu.Items.Add("Open dashboard", null, (_, _) => Services.OpenSpent()); - menu.Items.Add("Sync now", null, (_, _) => Services.SyncNow()); - menu.Items.Add(new ToolStripSeparator()); - menu.Items.Add("Start service", null, (_, _) => Services.StartService()); - menu.Items.Add("Restart service", null, (_, _) => Services.RestartService()); - menu.Items.Add("Stop service", null, (_, _) => Services.StopService()); - menu.Items.Add(new ToolStripSeparator()); - menu.Items.Add("Quit", null, (_, _) => Application.Exit()); - return menu; - } - - private void UpdateTooltip() - { - var text = _status.IsOnline - ? string.IsNullOrEmpty(_status.Version) ? "Spent: running" : $"Spent: running v{_status.Version}" - : "Spent: stopped"; - if (text.Length > 127) text = text.Substring(0, 127); - _notify.Text = text; - } - - public void Dispose() - { - _notify.Visible = false; - _notify.Dispose(); - _status.Dispose(); - } -} diff --git a/menubar/windows/StatusModel.cs b/menubar/windows/StatusModel.cs deleted file mode 100644 index bf7588a..0000000 --- a/menubar/windows/StatusModel.cs +++ /dev/null @@ -1,139 +0,0 @@ -using System.ComponentModel; -using System.Net; -using System.Net.Http; -using System.Runtime.CompilerServices; -using System.Text.Json; - -namespace Spent; - -// Polls GET /api/health every 5s; mirrors menubar/mac StatusModel. -internal sealed class StatusModel : INotifyPropertyChanged, IDisposable -{ - private readonly HttpClient _http; - private readonly CancellationTokenSource _cts = new(); - private readonly SynchronizationContext? _uiContext; - private bool _isOnline; - private string _version = string.Empty; - - public StatusModel() - { - // Captured on the UI thread (set up by ApplicationConfiguration.Initialize). - _uiContext = SynchronizationContext.Current; - // Restrict to loopback only. Any non-127.0.0.1 URL is short-circuited. - var handler = new HttpClientHandler - { - UseProxy = false, - AllowAutoRedirect = false, - }; - _http = new HttpClient(handler) { Timeout = TimeSpan.FromSeconds(2) }; - } - - public event PropertyChangedEventHandler? PropertyChanged; - - public bool IsOnline - { - get => _isOnline; - private set - { - if (Set(ref _isOnline, value)) - { - PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(StatusText))); - } - } - } - - public string Version - { - get => _version; - private set - { - if (Set(ref _version, value)) - { - PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(StatusText))); - } - } - } - - public string StatusText => - !_isOnline ? "Stopped" - : string.IsNullOrEmpty(_version) ? "Running" - : $"Running · v{_version}"; - - public void Start() - { - _ = PollLoopAsync(_cts.Token); - } - - public async Task PollOnceAsync() - { - try - { - using var resp = await _http.GetAsync(Constants.HealthUrl, _cts.Token).ConfigureAwait(false); - if (resp.StatusCode != HttpStatusCode.OK) - { - UpdateOnUi(false, string.Empty); - return; - } - var body = await resp.Content.ReadAsStringAsync(_cts.Token).ConfigureAwait(false); - using var doc = JsonDocument.Parse(body); - var root = doc.RootElement; - var ok = root.TryGetProperty("ok", out var okEl) && okEl.ValueKind == JsonValueKind.True; - var ver = root.TryGetProperty("version", out var verEl) && verEl.ValueKind == JsonValueKind.String - ? verEl.GetString() ?? string.Empty - : string.Empty; - UpdateOnUi(ok, ver); - } - catch - { - UpdateOnUi(false, string.Empty); - } - } - - private async Task PollLoopAsync(CancellationToken ct) - { - while (!ct.IsCancellationRequested) - { - await PollOnceAsync().ConfigureAwait(false); - try - { - await Task.Delay(TimeSpan.FromSeconds(5), ct).ConfigureAwait(false); - } - catch (OperationCanceledException) - { - return; - } - } - } - - private void UpdateOnUi(bool online, string version) - { - if (_uiContext is null || SynchronizationContext.Current == _uiContext) - { - IsOnline = online; - Version = version; - } - else - { - _uiContext.Post(_ => - { - IsOnline = online; - Version = version; - }, null); - } - } - - private bool Set(ref T field, T value, [CallerMemberName] string? name = null) - { - if (EqualityComparer.Default.Equals(field, value)) return false; - field = value; - PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name)); - return true; - } - - public void Dispose() - { - _cts.Cancel(); - _cts.Dispose(); - _http.Dispose(); - } -} diff --git a/menubar/windows/app.manifest b/menubar/windows/app.manifest deleted file mode 100644 index ad22b94..0000000 --- a/menubar/windows/app.manifest +++ /dev/null @@ -1,22 +0,0 @@ - - - - - - - - - - - - - - - - - - true/pm - PerMonitorV2 - - - diff --git a/menubar/windows/build.ps1 b/menubar/windows/build.ps1 deleted file mode 100644 index a7be8cc..0000000 --- a/menubar/windows/build.ps1 +++ /dev/null @@ -1,64 +0,0 @@ -#!/usr/bin/env pwsh -$ErrorActionPreference = "Stop" - -# Build the Spent Windows tray app: a tiny WPF NotifyIcon controller for the -# always-on Next.js server. Network access is restricted to 127.0.0.1 only -# (see Constants.cs and StatusModel.cs). - -Set-Location -Path $PSScriptRoot - -function Fail-MissingSdk { - Write-Error @" -.NET 8 SDK not found. (The dotnet launcher can be installed without an SDK.) -Install .NET 8 SDK from: - https://dotnet.microsoft.com/download/dotnet/8.0 -Or via winget: - winget install Microsoft.DotNet.SDK.8 -"@ - exit 1 -} - -if (-not (Get-Command dotnet -ErrorAction SilentlyContinue)) { - Fail-MissingSdk -} - -$sdks = & dotnet --list-sdks 2>$null -if (-not ($sdks | Where-Object { $_ -match '^(8|9|1\d)\.' })) { - Fail-MissingSdk -} - -$outDir = "build" -if (Test-Path $outDir) { - Remove-Item -Recurse -Force $outDir -} - -Write-Host "Building release binary..." -& dotnet publish -c Release -r win-x64 --self-contained true ` - -p:PublishSingleFile=true ` - -p:IncludeNativeLibrariesForSelfExtract=true ` - -p:EnableCompressionInSingleFile=true ` - -o $outDir - -if ($LASTEXITCODE -ne 0) { - Write-Error "dotnet publish failed with exit code $LASTEXITCODE." - exit 1 -} - -$exe = Join-Path $outDir "Spent.exe" -if (-not (Test-Path $exe)) { - Write-Error "Expected $exe but it was not produced." - exit 1 -} - -Write-Host "" -Write-Host "Built: $exe" -Write-Host "" -Write-Host "To install:" -Write-Host " mkdir `$env:LOCALAPPDATA\Programs\Spent" -Write-Host " Copy-Item $exe `$env:LOCALAPPDATA\Programs\Spent\" -Write-Host "" -Write-Host "First launch: SmartScreen may show 'Windows protected your PC'." -Write-Host "Click 'More info' -> 'Run anyway'. (The binary is unsigned.)" -Write-Host "" -Write-Host "To auto-start at login, drop a shortcut to Spent.exe into:" -Write-Host " shell:startup (run from Win+R)" diff --git a/package-lock.json b/package-lock.json deleted file mode 100644 index 0c4c131..0000000 --- a/package-lock.json +++ /dev/null @@ -1,12019 +0,0 @@ -{ - "name": "spent", - "version": "0.1.0", - "lockfileVersion": 3, - "requires": true, - "packages": { - "": { - "name": "spent", - "version": "0.1.0", - "dependencies": { - "@anthropic-ai/sdk": "0.95.2", - "@base-ui/react": "^1.4.1", - "@tanstack/react-query": "^5.100.10", - "better-sqlite3": "12.10.0", - "class-variance-authority": "^0.7.1", - "clsx": "^2.1.1", - "framer-motion": "^12.38.0", - "israeli-bank-scrapers": "6.7.4", - "lucide-react": "^1.14.0", - "next": "16.2.6", - "next-intl": "^4.12.0", - "next-themes": "^0.4.6", - "react": "19.2.4", - "react-dom": "19.2.4", - "recharts": "^3.8.1", - "server-only": "^0.0.1", - "shadcn": "^4.7.0", - "sonner": "^2.0.7", - "tailwind-merge": "^3.6.0", - "tw-animate-css": "^1.4.0" - }, - "devDependencies": { - "@tailwindcss/postcss": "^4", - "@types/better-sqlite3": "^7.6.13", - "@types/node": "^20", - "@types/react": "^19", - "@types/react-dom": "^19", - "eslint": "^9", - "eslint-config-next": "16.2.6", - "tailwindcss": "^4", - "typescript": "^5" - } - }, - "node_modules/@alloc/quick-lru": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz", - "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@anthropic-ai/sdk": { - "version": "0.95.2", - "resolved": "https://registry.npmjs.org/@anthropic-ai/sdk/-/sdk-0.95.2.tgz", - "integrity": "sha512-Egddwo3sheo1PzUrMkZnH6VkQYwS0h/b/i8vSK8Ta9M45UQipAMeDFH57dYuDAfXMEUUGeKw6CMlremgMZgrSQ==", - "license": "MIT", - "dependencies": { - "json-schema-to-ts": "^3.1.1", - "standardwebhooks": "^1.0.0" - }, - "bin": { - "anthropic-ai-sdk": "bin/cli" - }, - "peerDependencies": { - "zod": "^3.25.0 || ^4.0.0" - }, - "peerDependenciesMeta": { - "zod": { - "optional": true - } - } - }, - "node_modules/@babel/code-frame": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz", - "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==", - "license": "MIT", - "dependencies": { - "@babel/helper-validator-identifier": "^7.28.5", - "js-tokens": "^4.0.0", - "picocolors": "^1.1.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/compat-data": { - "version": "7.29.3", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.3.tgz", - "integrity": "sha512-LIVqM46zQWZhj17qA8wb4nW/ixr2y1Nw+r1etiAWgRM6U1IqP+LNhL1yg440jYZR72jCWcWbLWzIosH+uP1fqg==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/core": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz", - "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==", - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.29.0", - "@babel/generator": "^7.29.0", - "@babel/helper-compilation-targets": "^7.28.6", - "@babel/helper-module-transforms": "^7.28.6", - "@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.1.0", - "gensync": "^1.0.0-beta.2", - "json5": "^2.2.3", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/babel" - } - }, - "node_modules/@babel/generator": { - "version": "7.29.1", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz", - "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==", - "license": "MIT", - "dependencies": { - "@babel/parser": "^7.29.0", - "@babel/types": "^7.29.0", - "@jridgewell/gen-mapping": "^0.3.12", - "@jridgewell/trace-mapping": "^0.3.28", - "jsesc": "^3.0.2" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-annotate-as-pure": { - "version": "7.27.3", - "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.27.3.tgz", - "integrity": "sha512-fXSwMQqitTGeHLBC08Eq5yXz2m37E4pJX1qAU1+2cNedz/ifv/bVXft90VeSav5nFO61EcNgwr0aJxbyPaWBPg==", - "license": "MIT", - "dependencies": { - "@babel/types": "^7.27.3" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-compilation-targets": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz", - "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==", - "license": "MIT", - "dependencies": { - "@babel/compat-data": "^7.28.6", - "@babel/helper-validator-option": "^7.27.1", - "browserslist": "^4.24.0", - "lru-cache": "^5.1.1", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-create-class-features-plugin": { - "version": "7.29.3", - "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.29.3.tgz", - "integrity": "sha512-RpLYy2sb51oNLjuu1iD3bwBqCBWUzjO0ocp+iaCP/lJtb2CPLcnC2Fftw+4sAzaMELGeWTgExSKADbdo0GFVzA==", - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.27.3", - "@babel/helper-member-expression-to-functions": "^7.28.5", - "@babel/helper-optimise-call-expression": "^7.27.1", - "@babel/helper-replace-supers": "^7.28.6", - "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1", - "@babel/traverse": "^7.29.0", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/helper-globals": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", - "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-member-expression-to-functions": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.28.5.tgz", - "integrity": "sha512-cwM7SBRZcPCLgl8a7cY0soT1SptSzAlMH39vwiRpOQkJlh53r5hdHwLSCZpQdVLT39sZt+CRpNwYG4Y2v77atg==", - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.28.5", - "@babel/types": "^7.28.5" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-module-imports": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz", - "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==", - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.28.6", - "@babel/types": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-module-transforms": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz", - "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==", - "license": "MIT", - "dependencies": { - "@babel/helper-module-imports": "^7.28.6", - "@babel/helper-validator-identifier": "^7.28.5", - "@babel/traverse": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/helper-optimise-call-expression": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.27.1.tgz", - "integrity": "sha512-URMGH08NzYFhubNSGJrpUEphGKQwMQYBySzat5cAByY1/YgIRkULnIy3tAMeszlL/so2HbeilYloUmSpd7GdVw==", - "license": "MIT", - "dependencies": { - "@babel/types": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-plugin-utils": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz", - "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-replace-supers": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.28.6.tgz", - "integrity": "sha512-mq8e+laIk94/yFec3DxSjCRD2Z0TAjhVbEJY3UQrlwVo15Lmt7C2wAUbK4bjnTs4APkwsYLTahXRraQXhb1WCg==", - "license": "MIT", - "dependencies": { - "@babel/helper-member-expression-to-functions": "^7.28.5", - "@babel/helper-optimise-call-expression": "^7.27.1", - "@babel/traverse": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/helper-skip-transparent-expression-wrappers": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.27.1.tgz", - "integrity": "sha512-Tub4ZKEXqbPjXgWLl2+3JpQAYBJ8+ikpQ2Ocj/q/r0LwE3UhENh7EUabyHjz2kCEsrRY83ew2DQdHluuiDQFzg==", - "license": "MIT", - "dependencies": { - "@babel/traverse": "^7.27.1", - "@babel/types": "^7.27.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-string-parser": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", - "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-validator-identifier": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", - "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-validator-option": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", - "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helpers": { - "version": "7.29.2", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz", - "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==", - "license": "MIT", - "dependencies": { - "@babel/template": "^7.28.6", - "@babel/types": "^7.29.0" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/parser": { - "version": "7.29.3", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.3.tgz", - "integrity": "sha512-b3ctpQwp+PROvU/cttc4OYl4MzfJUWy6FZg+PMXfzmt/+39iHVF0sDfqay8TQM3JA2EUOyKcFZt75jWriQijsA==", - "license": "MIT", - "dependencies": { - "@babel/types": "^7.29.0" - }, - "bin": { - "parser": "bin/babel-parser.js" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@babel/plugin-syntax-jsx": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.28.6.tgz", - "integrity": "sha512-wgEmr06G6sIpqr8YDwA2dSRTE3bJ+V0IfpzfSY3Lfgd7YWOaAdlykvJi13ZKBt8cZHfgH1IXN+CL656W3uUa4w==", - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-syntax-typescript": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.28.6.tgz", - "integrity": "sha512-+nDNmQye7nlnuuHDboPbGm00Vqg3oO8niRRL27/4LYHUsHYh0zJ1xWOz0uRwNFmM1Avzk8wZbc6rdiYhomzv/A==", - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-modules-commonjs": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.28.6.tgz", - "integrity": "sha512-jppVbf8IV9iWWwWTQIxJMAJCWBuuKx71475wHwYytrRGQ2CWiDvYlADQno3tcYpS/T2UUWFQp3nVtYfK/YBQrA==", - "license": "MIT", - "dependencies": { - "@babel/helper-module-transforms": "^7.28.6", - "@babel/helper-plugin-utils": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-typescript": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.28.6.tgz", - "integrity": "sha512-0YWL2RFxOqEm9Efk5PvreamxPME8OyY0wM5wh5lHjF+VtVhdneCWGzZeSqzOfiobVqQaNCd2z0tQvnI9DaPWPw==", - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.27.3", - "@babel/helper-create-class-features-plugin": "^7.28.6", - "@babel/helper-plugin-utils": "^7.28.6", - "@babel/helper-skip-transparent-expression-wrappers": "^7.27.1", - "@babel/plugin-syntax-typescript": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/preset-typescript": { - "version": "7.28.5", - "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.28.5.tgz", - "integrity": "sha512-+bQy5WOI2V6LJZpPVxY+yp66XdZ2yifu0Mc1aP5CQKgjn4QM5IN2i5fAZ4xKop47pr8rpVhiAeu+nDQa12C8+g==", - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1", - "@babel/helper-validator-option": "^7.27.1", - "@babel/plugin-syntax-jsx": "^7.27.1", - "@babel/plugin-transform-modules-commonjs": "^7.27.1", - "@babel/plugin-transform-typescript": "^7.28.5" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/runtime": { - "version": "7.29.2", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz", - "integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/template": { - "version": "7.28.6", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", - "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.28.6", - "@babel/parser": "^7.28.6", - "@babel/types": "^7.28.6" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/traverse": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz", - "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==", - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.29.0", - "@babel/generator": "^7.29.0", - "@babel/helper-globals": "^7.28.0", - "@babel/parser": "^7.29.0", - "@babel/template": "^7.28.6", - "@babel/types": "^7.29.0", - "debug": "^4.3.1" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/types": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", - "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==", - "license": "MIT", - "dependencies": { - "@babel/helper-string-parser": "^7.27.1", - "@babel/helper-validator-identifier": "^7.28.5" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@base-ui/react": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/@base-ui/react/-/react-1.4.1.tgz", - "integrity": "sha512-Ab5/LIhcmL8BQcsBUYiOfkSDRdLpvgUBzMK30cu684JPcLclYlztharvCZyNNgzJtbAiREzI9q0pI5erHCMgCw==", - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.29.2", - "@base-ui/utils": "0.2.8", - "@floating-ui/react-dom": "^2.1.8", - "@floating-ui/utils": "^0.2.11", - "use-sync-external-store": "^1.6.0" - }, - "engines": { - "node": ">=14.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/mui-org" - }, - "peerDependencies": { - "@date-fns/tz": "^1.2.0", - "@types/react": "^17 || ^18 || ^19", - "date-fns": "^4.0.0", - "react": "^17 || ^18 || ^19", - "react-dom": "^17 || ^18 || ^19" - }, - "peerDependenciesMeta": { - "@date-fns/tz": { - "optional": true - }, - "@types/react": { - "optional": true - }, - "date-fns": { - "optional": true - } - } - }, - "node_modules/@base-ui/utils": { - "version": "0.2.8", - "resolved": "https://registry.npmjs.org/@base-ui/utils/-/utils-0.2.8.tgz", - "integrity": "sha512-jvOi+c+ftGlGotNcKnzPVg2IhCaDTB6/6R3JeqdjdXktuAJi3wKH9T7+svuaKh1mmfVU11UWzUZVH74JDfi/wQ==", - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.29.2", - "@floating-ui/utils": "^0.2.11", - "reselect": "^5.1.1", - "use-sync-external-store": "^1.6.0" - }, - "peerDependencies": { - "@types/react": "^17 || ^18 || ^19", - "react": "^17 || ^18 || ^19", - "react-dom": "^17 || ^18 || ^19" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - } - } - }, - "node_modules/@dotenvx/dotenvx": { - "version": "1.65.0", - "resolved": "https://registry.npmjs.org/@dotenvx/dotenvx/-/dotenvx-1.65.0.tgz", - "integrity": "sha512-v4FA/Lw3pTEloLxBqTOaYDX6MNo0Jo7lGBsPZhwnJBqRJp0AzQg1ZZNxrFsh6HVC6QWeWrfIKLn0y2eyIXaVDg==", - "license": "BSD-3-Clause", - "dependencies": { - "commander": "^11.1.0", - "dotenv": "^17.2.1", - "eciesjs": "^0.4.10", - "execa": "^5.1.1", - "fdir": "^6.2.0", - "ignore": "^5.3.0", - "object-treeify": "1.1.33", - "picomatch": "^4.0.4", - "which": "^4.0.0", - "yocto-spinner": "^1.1.0" - }, - "bin": { - "dotenvx": "src/cli/dotenvx.js" - }, - "funding": { - "url": "https://dotenvx.com" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/commander": { - "version": "11.1.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-11.1.0.tgz", - "integrity": "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ==", - "license": "MIT", - "engines": { - "node": ">=16" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/execa": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", - "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", - "license": "MIT", - "dependencies": { - "cross-spawn": "^7.0.3", - "get-stream": "^6.0.0", - "human-signals": "^2.1.0", - "is-stream": "^2.0.0", - "merge-stream": "^2.0.0", - "npm-run-path": "^4.0.1", - "onetime": "^5.1.2", - "signal-exit": "^3.0.3", - "strip-final-newline": "^2.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sindresorhus/execa?sponsor=1" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/fdir": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", - "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", - "license": "MIT", - "engines": { - "node": ">=12.0.0" - }, - "peerDependencies": { - "picomatch": "^3 || ^4" - }, - "peerDependenciesMeta": { - "picomatch": { - "optional": true - } - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/get-stream": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", - "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/human-signals": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", - "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", - "license": "Apache-2.0", - "engines": { - "node": ">=10.17.0" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/is-stream": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", - "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", - "license": "MIT", - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/isexe": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-3.1.5.tgz", - "integrity": "sha512-6B3tLtFqtQS4ekarvLVMZ+X+VlvQekbe4taUkf/rhVO3d/h0M2rfARm/pXLcPEsjjMsFgrFgSrhQIxcSVrBz8w==", - "license": "BlueOak-1.0.0", - "engines": { - "node": ">=18" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/npm-run-path": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", - "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", - "license": "MIT", - "dependencies": { - "path-key": "^3.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/onetime": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", - "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", - "license": "MIT", - "dependencies": { - "mimic-fn": "^2.1.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/picomatch": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", - "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/signal-exit": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", - "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", - "license": "ISC" - }, - "node_modules/@dotenvx/dotenvx/node_modules/strip-final-newline": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", - "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/@dotenvx/dotenvx/node_modules/which": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/which/-/which-4.0.0.tgz", - "integrity": "sha512-GlaYyEb07DPxYCKhKzplCWBJtvxZcZMrL+4UkrTSJHHPyZU4mYYTv3qaOe77H7EODLSSopAUFAc6W8U4yqvscg==", - "license": "ISC", - "dependencies": { - "isexe": "^3.1.1" - }, - "bin": { - "node-which": "bin/which.js" - }, - "engines": { - "node": "^16.13.0 || >=18.0.0" - } - }, - "node_modules/@ecies/ciphers": { - "version": "0.2.6", - "resolved": "https://registry.npmjs.org/@ecies/ciphers/-/ciphers-0.2.6.tgz", - "integrity": "sha512-patgsRPKGkhhoBjETV4XxD0En4ui5fbX0hzayqI3M8tvNMGUoUvmyYAIWwlxBc1KX5cturfqByYdj5bYGRpN9g==", - "license": "MIT", - "engines": { - "bun": ">=1", - "deno": ">=2.7.10", - "node": ">=16" - }, - "peerDependencies": { - "@noble/ciphers": "^1.0.0" - } - }, - "node_modules/@emnapi/core": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", - "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "@emnapi/wasi-threads": "1.2.1", - "tslib": "^2.4.0" - } - }, - "node_modules/@emnapi/runtime": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", - "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", - "license": "MIT", - "optional": true, - "dependencies": { - "tslib": "^2.4.0" - } - }, - "node_modules/@emnapi/wasi-threads": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", - "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "tslib": "^2.4.0" - } - }, - "node_modules/@eslint-community/eslint-utils": { - "version": "4.9.1", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", - "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "eslint-visitor-keys": "^3.4.3" - }, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - }, - "peerDependencies": { - "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" - } - }, - "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", - "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@eslint-community/regexpp": { - "version": "4.12.2", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", - "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^12.0.0 || ^14.0.0 || >=16.0.0" - } - }, - "node_modules/@eslint/config-array": { - "version": "0.21.2", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.2.tgz", - "integrity": "sha512-nJl2KGTlrf9GjLimgIru+V/mzgSK0ABCDQRvxw5BjURL7WfH5uoWmizbH7QB6MmnMBd8cIC9uceWnezL1VZWWw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/object-schema": "^2.1.7", - "debug": "^4.3.1", - "minimatch": "^3.1.5" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/config-helpers": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz", - "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/core": "^0.17.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/core": { - "version": "0.17.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz", - "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@types/json-schema": "^7.0.15" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/eslintrc": { - "version": "3.3.5", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.5.tgz", - "integrity": "sha512-4IlJx0X0qftVsN5E+/vGujTRIFtwuLbNsVUe7TO6zYPDR1O6nFwvwhIKEKSrl6dZchmYBITazxKoUYOjdtjlRg==", - "dev": true, - "license": "MIT", - "dependencies": { - "ajv": "^6.14.0", - "debug": "^4.3.2", - "espree": "^10.0.1", - "globals": "^14.0.0", - "ignore": "^5.2.0", - "import-fresh": "^3.2.1", - "js-yaml": "^4.1.1", - "minimatch": "^3.1.5", - "strip-json-comments": "^3.1.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@eslint/js": { - "version": "9.39.4", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.4.tgz", - "integrity": "sha512-nE7DEIchvtiFTwBw4Lfbu59PG+kCofhjsKaCWzxTpt4lfRjRMqG6uMBzKXuEcyXhOHoUp9riAm7/aWYGhXZ9cw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://eslint.org/donate" - } - }, - "node_modules/@eslint/object-schema": { - "version": "2.1.7", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz", - "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/plugin-kit": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz", - "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@eslint/core": "^0.17.0", - "levn": "^0.4.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@floating-ui/core": { - "version": "1.7.5", - "resolved": "https://registry.npmjs.org/@floating-ui/core/-/core-1.7.5.tgz", - "integrity": "sha512-1Ih4WTWyw0+lKyFMcBHGbb5U5FtuHJuujoyyr5zTaWS5EYMeT6Jb2AuDeftsCsEuchO+mM2ij5+q9crhydzLhQ==", - "license": "MIT", - "dependencies": { - "@floating-ui/utils": "^0.2.11" - } - }, - "node_modules/@floating-ui/dom": { - "version": "1.7.6", - "resolved": "https://registry.npmjs.org/@floating-ui/dom/-/dom-1.7.6.tgz", - "integrity": "sha512-9gZSAI5XM36880PPMm//9dfiEngYoC6Am2izES1FF406YFsjvyBMmeJ2g4SAju3xWwtuynNRFL2s9hgxpLI5SQ==", - "license": "MIT", - "dependencies": { - "@floating-ui/core": "^1.7.5", - "@floating-ui/utils": "^0.2.11" - } - }, - "node_modules/@floating-ui/react-dom": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/@floating-ui/react-dom/-/react-dom-2.1.8.tgz", - "integrity": "sha512-cC52bHwM/n/CxS87FH0yWdngEZrjdtLW/qVruo68qg+prK7ZQ4YGdut2GyDVpoGeAYe/h899rVeOVm6Oi40k2A==", - "license": "MIT", - "dependencies": { - "@floating-ui/dom": "^1.7.6" - }, - "peerDependencies": { - "react": ">=16.8.0", - "react-dom": ">=16.8.0" - } - }, - "node_modules/@floating-ui/utils": { - "version": "0.2.11", - "resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.11.tgz", - "integrity": "sha512-RiB/yIh78pcIxl6lLMG0CgBXAZ2Y0eVHqMPYugu+9U0AeT6YBeiJpf7lbdJNIugFP5SIjwNRgo4DhR1Qxi26Gg==", - "license": "MIT" - }, - "node_modules/@formatjs/fast-memoize": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/@formatjs/fast-memoize/-/fast-memoize-3.1.5.tgz", - "integrity": "sha512-KLi3fan6WnCHmigd9pmEEN8Hid0v4wiFBW576M/d07KMWYecf1CvyMI3n34vCmHT4AoVqG2n702kiHbXjzZX2A==", - "license": "MIT" - }, - "node_modules/@formatjs/icu-messageformat-parser": { - "version": "3.5.9", - "resolved": "https://registry.npmjs.org/@formatjs/icu-messageformat-parser/-/icu-messageformat-parser-3.5.9.tgz", - "integrity": "sha512-PZm6O9JI/gUPtQV9r2eaMuLb4yWqV2vz+ot03ORHWTKO343LSpZi0TqeXLB2ZZGDXLCw2SbfgsQ0GxoxXMl79g==", - "license": "MIT", - "dependencies": { - "@formatjs/icu-skeleton-parser": "2.1.9" - } - }, - "node_modules/@formatjs/icu-skeleton-parser": { - "version": "2.1.9", - "resolved": "https://registry.npmjs.org/@formatjs/icu-skeleton-parser/-/icu-skeleton-parser-2.1.9.tgz", - "integrity": "sha512-rsxswgHMfU1zUgB2byc08fesf83wLGjFnzLCEtuf00mx2doiqc6pYrf67raI37XqdRcGUviQepk2UKGqpng74Q==", - "license": "MIT" - }, - "node_modules/@formatjs/intl-localematcher": { - "version": "0.8.8", - "resolved": "https://registry.npmjs.org/@formatjs/intl-localematcher/-/intl-localematcher-0.8.8.tgz", - "integrity": "sha512-pBr2hVKWvkHVnfXegW+53NT9U2uaVQCc+EgzLPCCwXqBA3nvM5fPbK9IcJlNjV+NMKGyZ2F3ZSG78iGdxAAqbA==", - "license": "MIT", - "dependencies": { - "@formatjs/fast-memoize": "3.1.5" - } - }, - "node_modules/@hono/node-server": { - "version": "1.19.14", - "resolved": "https://registry.npmjs.org/@hono/node-server/-/node-server-1.19.14.tgz", - "integrity": "sha512-GwtvgtXxnWsucXvbQXkRgqksiH2Qed37H9xHZocE5sA3N8O8O8/8FA3uclQXxXVzc9XBZuEOMK7+r02FmSpHtw==", - "license": "MIT", - "engines": { - "node": ">=18.14.1" - }, - "peerDependencies": { - "hono": "^4" - } - }, - "node_modules/@humanfs/core": { - "version": "0.19.2", - "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz", - "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@humanfs/types": "^0.15.0" - }, - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/@humanfs/node": { - "version": "0.16.8", - "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz", - "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@humanfs/core": "^0.19.2", - "@humanfs/types": "^0.15.0", - "@humanwhocodes/retry": "^0.4.0" - }, - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/@humanfs/types": { - "version": "0.15.0", - "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz", - "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/@humanwhocodes/module-importer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", - "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=12.22" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/nzakas" - } - }, - "node_modules/@humanwhocodes/retry": { - "version": "0.4.3", - "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", - "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=18.18" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/nzakas" - } - }, - "node_modules/@img/colour": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@img/colour/-/colour-1.1.0.tgz", - "integrity": "sha512-Td76q7j57o/tLVdgS746cYARfSyxk8iEfRxewL9h4OMzYhbW4TAcppl0mT4eyqXddh6L/jwoM75mo7ixa/pCeQ==", - "license": "MIT", - "optional": true, - "engines": { - "node": ">=18" - } - }, - "node_modules/@img/sharp-darwin-arm64": { - "version": "0.34.5", - "resolved": "https://registry.npmjs.org/@img/sharp-darwin-arm64/-/sharp-darwin-arm64-0.34.5.tgz", - "integrity": "sha512-imtQ3WMJXbMY4fxb/Ndp6HBTNVtWCUI0WdobyheGf5+ad6xX8VIDO8u2xE4qc/fr08CKG/7dDseFtn6M6g/r3w==", - "cpu": [ - "arm64" - ], - "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-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" - ], - "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", - "integrity": "sha512-zqjjo7RatFfFoP0MkQ51jfuFZBnVE2pRiaydKJ1G/rHZvnsrHAOcQALIi9sA5co5xenQdTugCvtb1cuf78Vf4g==", - "cpu": [ - "arm64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "darwin" - ], - "funding": { - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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" - ], - "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/@inquirer/ansi": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@inquirer/ansi/-/ansi-2.0.5.tgz", - "integrity": "sha512-doc2sWgJpbFQ64UflSVd17ibMGDuxO1yKgOgLMwavzESnXjFWJqUeG8saYosqKpHp4kWiM5x1nXvEjbpx90gzw==", - "license": "MIT", - "engines": { - "node": ">=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0" - } - }, - "node_modules/@inquirer/confirm": { - "version": "6.0.13", - "resolved": "https://registry.npmjs.org/@inquirer/confirm/-/confirm-6.0.13.tgz", - "integrity": "sha512-wkGPC7yJ5WJk1DJ5SX7fzk+gfj4BM8cf5dDDi71B/551xHrdsZVRJOC0WyikXd0pEsb/9cLniuE4atbsMqmFkw==", - "license": "MIT", - "dependencies": { - "@inquirer/core": "^11.1.10", - "@inquirer/type": "^4.0.5" - }, - "engines": { - "node": ">=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0" - }, - "peerDependencies": { - "@types/node": ">=18" - }, - "peerDependenciesMeta": { - "@types/node": { - "optional": true - } - } - }, - "node_modules/@inquirer/core": { - "version": "11.1.10", - "resolved": "https://registry.npmjs.org/@inquirer/core/-/core-11.1.10.tgz", - "integrity": "sha512-a4Q5BXHQAHa9eO202sTaFCHFYVB3x5fauDuThEAdZ9gfn76pSxiKU7wWcEH0N1O0XmQvNfQNU6QXpiRxmYQx+A==", - "license": "MIT", - "dependencies": { - "@inquirer/ansi": "^2.0.5", - "@inquirer/figures": "^2.0.5", - "@inquirer/type": "^4.0.5", - "cli-width": "^4.1.0", - "fast-wrap-ansi": "^0.2.0", - "mute-stream": "^3.0.0", - "signal-exit": "^4.1.0" - }, - "engines": { - "node": ">=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0" - }, - "peerDependencies": { - "@types/node": ">=18" - }, - "peerDependenciesMeta": { - "@types/node": { - "optional": true - } - } - }, - "node_modules/@inquirer/figures": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@inquirer/figures/-/figures-2.0.5.tgz", - "integrity": "sha512-NsSs4kzfm12lNetHwAn3GEuH317IzpwrMCbOuMIVytpjnJ90YYHNwdRgYGuKmVxwuIqSgqk3M5qqQt1cDk0tGQ==", - "license": "MIT", - "engines": { - "node": ">=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0" - } - }, - "node_modules/@inquirer/type": { - "version": "4.0.5", - "resolved": "https://registry.npmjs.org/@inquirer/type/-/type-4.0.5.tgz", - "integrity": "sha512-aetVUNeKNc/VriqXlw1NRSW0zhMBB0W4bNbWRJgzRl/3d0QNDQFfk0GO5SDdtjMZVg6o8ZKEiadd7SCCzoOn5Q==", - "license": "MIT", - "engines": { - "node": ">=23.5.0 || ^22.13.0 || ^21.7.0 || ^20.12.0" - }, - "peerDependencies": { - "@types/node": ">=18" - }, - "peerDependenciesMeta": { - "@types/node": { - "optional": true - } - } - }, - "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.13", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", - "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", - "license": "MIT", - "dependencies": { - "@jridgewell/sourcemap-codec": "^1.5.0", - "@jridgewell/trace-mapping": "^0.3.24" - } - }, - "node_modules/@jridgewell/remapping": { - "version": "2.3.5", - "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", - "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", - "license": "MIT", - "dependencies": { - "@jridgewell/gen-mapping": "^0.3.5", - "@jridgewell/trace-mapping": "^0.3.24" - } - }, - "node_modules/@jridgewell/resolve-uri": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", - "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", - "license": "MIT", - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@jridgewell/sourcemap-codec": { - "version": "1.5.5", - "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", - "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", - "license": "MIT" - }, - "node_modules/@jridgewell/trace-mapping": { - "version": "0.3.31", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", - "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", - "license": "MIT", - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, - "node_modules/@modelcontextprotocol/sdk": { - "version": "1.29.0", - "resolved": "https://registry.npmjs.org/@modelcontextprotocol/sdk/-/sdk-1.29.0.tgz", - "integrity": "sha512-zo37mZA9hJWpULgkRpowewez1y6ML5GsXJPY8FI0tBBCd77HEvza4jDqRKOXgHNn867PVGCyTdzqpz0izu5ZjQ==", - "license": "MIT", - "dependencies": { - "@hono/node-server": "^1.19.9", - "ajv": "^8.17.1", - "ajv-formats": "^3.0.1", - "content-type": "^1.0.5", - "cors": "^2.8.5", - "cross-spawn": "^7.0.5", - "eventsource": "^3.0.2", - "eventsource-parser": "^3.0.0", - "express": "^5.2.1", - "express-rate-limit": "^8.2.1", - "hono": "^4.11.4", - "jose": "^6.1.3", - "json-schema-typed": "^8.0.2", - "pkce-challenge": "^5.0.0", - "raw-body": "^3.0.0", - "zod": "^3.25 || ^4.0", - "zod-to-json-schema": "^3.25.1" - }, - "engines": { - "node": ">=18" - }, - "peerDependencies": { - "@cfworker/json-schema": "^4.1.1", - "zod": "^3.25 || ^4.0" - }, - "peerDependenciesMeta": { - "@cfworker/json-schema": { - "optional": true - }, - "zod": { - "optional": false - } - } - }, - "node_modules/@modelcontextprotocol/sdk/node_modules/ajv": { - "version": "8.20.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.20.0.tgz", - "integrity": "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA==", - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.3", - "fast-uri": "^3.0.1", - "json-schema-traverse": "^1.0.0", - "require-from-string": "^2.0.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/@modelcontextprotocol/sdk/node_modules/json-schema-traverse": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", - "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", - "license": "MIT" - }, - "node_modules/@mswjs/interceptors": { - "version": "0.41.9", - "resolved": "https://registry.npmjs.org/@mswjs/interceptors/-/interceptors-0.41.9.tgz", - "integrity": "sha512-VVPPgHyQ6ShqnrmDWuxjmUIsO9gWyOZFmuOfLd9LfBGQJwZfy0gvv9pbHSJuoFNIYC7ZDX9aoFwowjcdSC4E8w==", - "license": "MIT", - "dependencies": { - "@open-draft/deferred-promise": "^2.2.0", - "@open-draft/logger": "^0.3.0", - "@open-draft/until": "^2.0.0", - "is-node-process": "^1.2.0", - "outvariant": "^1.4.3", - "strict-event-emitter": "^0.5.1" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/@mswjs/interceptors/node_modules/@open-draft/deferred-promise": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/@open-draft/deferred-promise/-/deferred-promise-2.2.0.tgz", - "integrity": "sha512-CecwLWx3rhxVQF6V4bAgPS5t+So2sTbPgAzafKkVizyi7tlwpcFpdFqq+wqF2OwNBmqFuu6tOyouTuxgpMfzmA==", - "license": "MIT" - }, - "node_modules/@napi-rs/wasm-runtime": { - "version": "0.2.12", - "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-0.2.12.tgz", - "integrity": "sha512-ZVWUcfwY4E/yPitQJl481FjFo3K22D6qF0DuFH6Y/nbnE11GY5uguDxZMGXPQ8WQ0128MXQD7TnfHyK4oWoIJQ==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "@emnapi/core": "^1.4.3", - "@emnapi/runtime": "^1.4.3", - "@tybys/wasm-util": "^0.10.0" - } - }, - "node_modules/@next/env": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/env/-/env-16.2.6.tgz", - "integrity": "sha512-gd8HoHN4ufj73WmR3JmVolrpJR47ILK6LouP5xElPglaVxir6e1a7VzvTvDWkOoPXT9rkkTzyCxBu4yeZfZwcw==", - "license": "MIT" - }, - "node_modules/@next/eslint-plugin-next": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/eslint-plugin-next/-/eslint-plugin-next-16.2.6.tgz", - "integrity": "sha512-Z8l6o4JWKUl755x4R+wogD86KPeU+Ckw4K+SYG4kHeOJtRenDeK+OSbGcqZpDtbwn9DsJVdir2UxmwXuinUbUw==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-glob": "3.3.1" - } - }, - "node_modules/@next/swc-darwin-arm64": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-darwin-arm64/-/swc-darwin-arm64-16.2.6.tgz", - "integrity": "sha512-ZJGkkcNfYgrrMkqOdZ7zoLa1TOy0qpcMfk/z4Mh/FKUz40gVO+HNQWqmLxf67Z5WB64DRp0dhEbyHfel+6sJUg==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-darwin-x64": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-16.2.6.tgz", - "integrity": "sha512-v/YLBHIY132Ced3puBJ7YJKw1lqsCrgcNo2aRJlCEyQrrCeRJlvGlnmxhPxNQI3KE3N1DN5r9TPNPvka3nq5RQ==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-linux-arm64-gnu": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-16.2.6.tgz", - "integrity": "sha512-RPOvqlYBbcQjkz9VQQDZ2T2bARIjXZV1KFlt+V2Mr6SW/e4I9fcKsaA0hdyf2FHoTlsV2xnBd5Y912rP/1Ce6w==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-linux-arm64-musl": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-16.2.6.tgz", - "integrity": "sha512-URUTu1+dMkxJsPFgm+OeEvq9wf5sujw0EvgYy80TDGHTSLTnIHeqb0Eu8A3sC95IRgjejQL+kC4mw+4yPxiAXA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-linux-x64-gnu": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-16.2.6.tgz", - "integrity": "sha512-DOj182mPV8G3UkrayLoREM5YEYI+Dk5wv7Ox9xl1fFibAELEsFD0lDPfHIeILlutMMfdyhlzYPELG3peuKaurw==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-linux-x64-musl": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-16.2.6.tgz", - "integrity": "sha512-HKQ5SP/V/ub73UvF7n/zeJlxk2kLmtL7Wzrg4WfmkjmNos5onJ2tKu7yZOPdL18A6Svfn3max29ym+ry7NkK4g==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-win32-arm64-msvc": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-16.2.6.tgz", - "integrity": "sha512-LZXpTlPyS5v7HhSmnvsLGP3iIYgYOBnc8r8ArlT55sGHV89bR2HlDdBjWQ+PY6SJMmk8TuVGFuxalnP3k/0Dwg==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@next/swc-win32-x64-msvc": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-16.2.6.tgz", - "integrity": "sha512-F0+4i0h9J6C4eE3EAPWsoCk7UW/dbzOjyzxY0qnDUOYFu6FFmdZ6l97/XdV3/Nz3VYyO7UWjyEJUXkGqcoXfMA==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10" - } - }, - "node_modules/@noble/ciphers": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/@noble/ciphers/-/ciphers-1.3.0.tgz", - "integrity": "sha512-2I0gnIVPtfnMw9ee9h1dJG7tp81+8Ob3OJb3Mv37rx5L40/b0i7djjCVvGOVqc9AEIQyvyu1i6ypKdFw8R8gQw==", - "license": "MIT", - "engines": { - "node": "^14.21.3 || >=16" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - } - }, - "node_modules/@noble/curves": { - "version": "1.9.7", - "resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.9.7.tgz", - "integrity": "sha512-gbKGcRUYIjA3/zCCNaWDciTMFI0dCkvou3TL8Zmy5Nc7sJ47a0jtOeZoTaMxkuqRo9cRhjOdZJXegxYE5FN/xw==", - "license": "MIT", - "dependencies": { - "@noble/hashes": "1.8.0" - }, - "engines": { - "node": "^14.21.3 || >=16" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - } - }, - "node_modules/@noble/hashes": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.8.0.tgz", - "integrity": "sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A==", - "license": "MIT", - "engines": { - "node": "^14.21.3 || >=16" - }, - "funding": { - "url": "https://paulmillr.com/funding/" - } - }, - "node_modules/@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "license": "MIT", - "dependencies": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nolyfill/is-core-module": { - "version": "1.0.39", - "resolved": "https://registry.npmjs.org/@nolyfill/is-core-module/-/is-core-module-1.0.39.tgz", - "integrity": "sha512-nn5ozdjYQpUCZlWGuxcJY/KpxkWQs4DcbMCmKojjyrYDEAGy4Ce19NN4v5MduafTwJlbKc99UA8YhSVqq9yPZA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12.4.0" - } - }, - "node_modules/@open-draft/deferred-promise": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@open-draft/deferred-promise/-/deferred-promise-3.0.0.tgz", - "integrity": "sha512-XW375UK8/9SqUVNVa6M0yEy8+iTi4QN5VZ7aZuRFQmy76LRwI9wy5F4YIBU6T+eTe2/DNDo8tqu8RHlwLHM6RA==", - "license": "MIT" - }, - "node_modules/@open-draft/logger": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/@open-draft/logger/-/logger-0.3.0.tgz", - "integrity": "sha512-X2g45fzhxH238HKO4xbSr7+wBS8Fvw6ixhTDuvLd5mqh6bJJCFAPwU9mPDxbcrRtfxv4u5IHCEH77BmxvXmmxQ==", - "license": "MIT", - "dependencies": { - "is-node-process": "^1.2.0", - "outvariant": "^1.4.0" - } - }, - "node_modules/@open-draft/until": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/@open-draft/until/-/until-2.1.0.tgz", - "integrity": "sha512-U69T3ItWHvLwGg5eJ0n3I62nWuE6ilHlmz7zM0npLBRvPRd7e6NYmg54vvRtP5mZG7kZqZCFVdsTWo7BPtBujg==", - "license": "MIT" - }, - "node_modules/@parcel/watcher": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-2.5.6.tgz", - "integrity": "sha512-tmmZ3lQxAe/k/+rNnXQRawJ4NjxO2hqiOLTHvWchtGZULp4RyFeh6aU4XdOYBFe2KE1oShQTv4AblOs2iOrNnQ==", - "hasInstallScript": true, - "license": "MIT", - "dependencies": { - "detect-libc": "^2.0.3", - "is-glob": "^4.0.3", - "node-addon-api": "^7.0.0", - "picomatch": "^4.0.3" - }, - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - }, - "optionalDependencies": { - "@parcel/watcher-android-arm64": "2.5.6", - "@parcel/watcher-darwin-arm64": "2.5.6", - "@parcel/watcher-darwin-x64": "2.5.6", - "@parcel/watcher-freebsd-x64": "2.5.6", - "@parcel/watcher-linux-arm-glibc": "2.5.6", - "@parcel/watcher-linux-arm-musl": "2.5.6", - "@parcel/watcher-linux-arm64-glibc": "2.5.6", - "@parcel/watcher-linux-arm64-musl": "2.5.6", - "@parcel/watcher-linux-x64-glibc": "2.5.6", - "@parcel/watcher-linux-x64-musl": "2.5.6", - "@parcel/watcher-win32-arm64": "2.5.6", - "@parcel/watcher-win32-ia32": "2.5.6", - "@parcel/watcher-win32-x64": "2.5.6" - } - }, - "node_modules/@parcel/watcher-android-arm64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-android-arm64/-/watcher-android-arm64-2.5.6.tgz", - "integrity": "sha512-YQxSS34tPF/6ZG7r/Ih9xy+kP/WwediEUsqmtf0cuCV5TPPKw/PQHRhueUo6JdeFJaqV3pyjm0GdYjZotbRt/A==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-darwin-arm64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-darwin-arm64/-/watcher-darwin-arm64-2.5.6.tgz", - "integrity": "sha512-Z2ZdrnwyXvvvdtRHLmM4knydIdU9adO3D4n/0cVipF3rRiwP+3/sfzpAwA/qKFL6i1ModaabkU7IbpeMBgiVEA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-darwin-x64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-darwin-x64/-/watcher-darwin-x64-2.5.6.tgz", - "integrity": "sha512-HgvOf3W9dhithcwOWX9uDZyn1lW9R+7tPZ4sug+NGrGIo4Rk1hAXLEbcH1TQSqxts0NYXXlOWqVpvS1SFS4fRg==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-freebsd-x64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-freebsd-x64/-/watcher-freebsd-x64-2.5.6.tgz", - "integrity": "sha512-vJVi8yd/qzJxEKHkeemh7w3YAn6RJCtYlE4HPMoVnCpIXEzSrxErBW5SJBgKLbXU3WdIpkjBTeUNtyBVn8TRng==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm-glibc": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm-glibc/-/watcher-linux-arm-glibc-2.5.6.tgz", - "integrity": "sha512-9JiYfB6h6BgV50CCfasfLf/uvOcJskMSwcdH1PHH9rvS1IrNy8zad6IUVPVUfmXr+u+Km9IxcfMLzgdOudz9EQ==", - "cpu": [ - "arm" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm-musl": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm-musl/-/watcher-linux-arm-musl-2.5.6.tgz", - "integrity": "sha512-Ve3gUCG57nuUUSyjBq/MAM0CzArtuIOxsBdQ+ftz6ho8n7s1i9E1Nmk/xmP323r2YL0SONs1EuwqBp2u1k5fxg==", - "cpu": [ - "arm" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm64-glibc": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm64-glibc/-/watcher-linux-arm64-glibc-2.5.6.tgz", - "integrity": "sha512-f2g/DT3NhGPdBmMWYoxixqYr3v/UXcmLOYy16Bx0TM20Tchduwr4EaCbmxh1321TABqPGDpS8D/ggOTaljijOA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-arm64-musl": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-arm64-musl/-/watcher-linux-arm64-musl-2.5.6.tgz", - "integrity": "sha512-qb6naMDGlbCwdhLj6hgoVKJl2odL34z2sqkC7Z6kzir8b5W65WYDpLB6R06KabvZdgoHI/zxke4b3zR0wAbDTA==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-x64-glibc": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-glibc/-/watcher-linux-x64-glibc-2.5.6.tgz", - "integrity": "sha512-kbT5wvNQlx7NaGjzPFu8nVIW1rWqV780O7ZtkjuWaPUgpv2NMFpjYERVi0UYj1msZNyCzGlaCWEtzc+exjMGbQ==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-linux-x64-musl": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-linux-x64-musl/-/watcher-linux-x64-musl-2.5.6.tgz", - "integrity": "sha512-1JRFeC+h7RdXwldHzTsmdtYR/Ku8SylLgTU/reMuqdVD7CtLwf0VR1FqeprZ0eHQkO0vqsbvFLXUmYm/uNKJBg==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-win32-arm64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-arm64/-/watcher-win32-arm64-2.5.6.tgz", - "integrity": "sha512-3ukyebjc6eGlw9yRt678DxVF7rjXatWiHvTXqphZLvo7aC5NdEgFufVwjFfY51ijYEWpXbqF5jtrK275z52D4Q==", - "cpu": [ - "arm64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-win32-ia32": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-ia32/-/watcher-win32-ia32-2.5.6.tgz", - "integrity": "sha512-k35yLp1ZMwwee3Ez/pxBi5cf4AoBKYXj00CZ80jUz5h8prpiaQsiRPKQMxoLstNuqe2vR4RNPEAEcjEFzhEz/g==", - "cpu": [ - "ia32" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher-win32-x64": { - "version": "2.5.6", - "resolved": "https://registry.npmjs.org/@parcel/watcher-win32-x64/-/watcher-win32-x64-2.5.6.tgz", - "integrity": "sha512-hbQlYcCq5dlAX9Qx+kFb0FHue6vbjlf0FrNzSKdYK2APUf7tGfGxQCk2ihEREmbR6ZMc0MVAD5RIX/41gpUzTw==", - "cpu": [ - "x64" - ], - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 10.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/@parcel/watcher/node_modules/picomatch": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", - "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/@puppeteer/browsers": { - "version": "2.13.2", - "resolved": "https://registry.npmjs.org/@puppeteer/browsers/-/browsers-2.13.2.tgz", - "integrity": "sha512-5EUZSUIc37H6aIXyWO0Z4y8NlF8NnjgmqeQgOGiswAU7pY0HOo16ho4+alIWmSfdZnjqBRawMsP3I5YqLSn6kw==", - "license": "Apache-2.0", - "dependencies": { - "debug": "^4.4.3", - "extract-zip": "^2.0.1", - "progress": "^2.0.3", - "proxy-agent": "^6.5.0", - "semver": "^7.7.4", - "tar-fs": "^3.1.1", - "yargs": "^17.7.2" - }, - "bin": { - "browsers": "lib/cjs/main-cli.js" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/@puppeteer/browsers/node_modules/semver": { - "version": "7.8.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", - "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/@puppeteer/browsers/node_modules/tar-fs": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-3.1.2.tgz", - "integrity": "sha512-QGxxTxxyleAdyM3kpFs14ymbYmNFrfY+pHj7Z8FgtbZ7w2//VAgLMac7sT6nRpIHjppXO2AwwEOg0bPFVRcmXw==", - "license": "MIT", - "dependencies": { - "pump": "^3.0.0", - "tar-stream": "^3.1.5" - }, - "optionalDependencies": { - "bare-fs": "^4.0.1", - "bare-path": "^3.0.0" - } - }, - "node_modules/@puppeteer/browsers/node_modules/tar-stream": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-3.2.0.tgz", - "integrity": "sha512-ojzvCvVaNp6aOTFmG7jaRD0meowIAuPc3cMMhSgKiVWws1GyHbGd/xvnyuRKcKlMpt3qvxx6r0hreCNITP9hIg==", - "license": "MIT", - "dependencies": { - "b4a": "^1.6.4", - "bare-fs": "^4.5.5", - "fast-fifo": "^1.2.0", - "streamx": "^2.15.0" - } - }, - "node_modules/@reduxjs/toolkit": { - "version": "2.11.2", - "resolved": "https://registry.npmjs.org/@reduxjs/toolkit/-/toolkit-2.11.2.tgz", - "integrity": "sha512-Kd6kAHTA6/nUpp8mySPqj3en3dm0tdMIgbttnQ1xFMVpufoj+ADi8pXLBsd4xzTRHQa7t/Jv8W5UnCuW4kuWMQ==", - "license": "MIT", - "dependencies": { - "@standard-schema/spec": "^1.0.0", - "@standard-schema/utils": "^0.3.0", - "immer": "^11.0.0", - "redux": "^5.0.1", - "redux-thunk": "^3.1.0", - "reselect": "^5.1.0" - }, - "peerDependencies": { - "react": "^16.9.0 || ^17.0.0 || ^18 || ^19", - "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0" - }, - "peerDependenciesMeta": { - "react": { - "optional": true - }, - "react-redux": { - "optional": true - } - } - }, - "node_modules/@reduxjs/toolkit/node_modules/immer": { - "version": "11.1.8", - "resolved": "https://registry.npmjs.org/immer/-/immer-11.1.8.tgz", - "integrity": "sha512-/tbkHMW7y10Lx6i1crLjD4/OhNkRG+Fo7byZHtah0547nIeXYcpIXaUh0IAQY6gO5459qpGGYapcEOHtFXkIuA==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/immer" - } - }, - "node_modules/@rtsao/scc": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", - "integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==", - "dev": true, - "license": "MIT" - }, - "node_modules/@schummar/icu-type-parser": { - "version": "1.21.5", - "resolved": "https://registry.npmjs.org/@schummar/icu-type-parser/-/icu-type-parser-1.21.5.tgz", - "integrity": "sha512-bXHSaW5jRTmke9Vd0h5P7BtWZG9Znqb8gSDxZnxaGSJnGwPLDPfS+3g0BKzeWqzgZPsIVZkM7m2tbo18cm5HBw==", - "license": "MIT" - }, - "node_modules/@sec-ant/readable-stream": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/@sec-ant/readable-stream/-/readable-stream-0.4.1.tgz", - "integrity": "sha512-831qok9r2t8AlxLko40y2ebgSDhenenCatLVeW/uBtnHPyhHOvG0C7TvfgecV+wHzIm5KUICgzmVpWS+IMEAeg==", - "license": "MIT" - }, - "node_modules/@sindresorhus/merge-streams": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@sindresorhus/merge-streams/-/merge-streams-4.0.0.tgz", - "integrity": "sha512-tlqY9xq5ukxTUZBmoOp+m61cqwQD5pHJtFY3Mn8CA8ps6yghLH/Hw8UPdqg4OLmFW3IFlcXnQNmo/dh8HzXYIQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/@stablelib/base64": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@stablelib/base64/-/base64-1.0.1.tgz", - "integrity": "sha512-1bnPQqSxSuc3Ii6MhBysoWCg58j97aUjuCSZrGSmDxNqtytIi0k8utUenAwTZN4V5mXXYGsVUI9zeBqy+jBOSQ==", - "license": "MIT" - }, - "node_modules/@standard-schema/spec": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.1.0.tgz", - "integrity": "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w==", - "license": "MIT" - }, - "node_modules/@standard-schema/utils": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/@standard-schema/utils/-/utils-0.3.0.tgz", - "integrity": "sha512-e7Mew686owMaPJVNNLs55PUvgz371nKgwsc4vxE49zsODpJEnxgxRo2y/OKrqueavXgZNMDVj3DdHFlaSAeU8g==", - "license": "MIT" - }, - "node_modules/@swc/core-darwin-arm64": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.15.33.tgz", - "integrity": "sha512-N+L0uXhuO7FIfzqwgxmzv0zIpV0qEp8wPX3QQs2p4atjMoywup2JTeDlXPw+z9pWJGCae3JjM+tZ6myclI+2gA==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-darwin-x64": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.15.33.tgz", - "integrity": "sha512-/Il4QHSOhV4FekbsDtkrNmKbsX26oSysvgrRswa/RYOHXAkwXDbB4jaeKq6PsJLSPkzJ2KzQ061gtBnk0vNHfA==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm-gnueabihf": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.15.33.tgz", - "integrity": "sha512-C64hBnBxq4viOPQ8hlx+2lJ23bzZBGnjw7ryALmS+0Q3zHmwO8lw1/DArLENw4Q18/0w5wdEO1k3m1wWNtKGqQ==", - "cpu": [ - "arm" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-gnu": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.15.33.tgz", - "integrity": "sha512-TRJfnJbX3jqpxRDRoieMzRiCBS5jOmXNb3iQXmcgjFEHKLnAgK1RZRU8Cq1MsPqO4jAJp/ld1G4O3fXuxv85uw==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-musl": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.15.33.tgz", - "integrity": "sha512-il7tYM+CpUNzieQbwAjFT1P8zqAhmGWNAGhQZBnxurXZ0aNn+5nqYFTEUKNZl7QibtT0uQXzTZrNGHCIj6Y1Og==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-ppc64-gnu": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-ppc64-gnu/-/core-linux-ppc64-gnu-1.15.33.tgz", - "integrity": "sha512-ZtNBwN0Z7CFj9Il0FcPaKdjgP7URyKu/3RfH46vq+0paOBqLj4NYldD6Qo//Duif/7IOtAraUfDOmp0PLAufog==", - "cpu": [ - "ppc64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-s390x-gnu": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-s390x-gnu/-/core-linux-s390x-gnu-1.15.33.tgz", - "integrity": "sha512-De1IyajoOmhOYYjw/lx66bKlyDpHZTueqwpDrWgf5O7T6d1ODeJJO9/OqMBmrBQc5C+dNnlmIufHsp4QVCWufA==", - "cpu": [ - "s390x" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-gnu": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.15.33.tgz", - "integrity": "sha512-mGTH0YxmUN+x6vRN/I6NOk5X0ogNktkwPnJ94IMvR7QjhRDwL0O8RXEDhyUM0YtwWrryBOqaJQBX4zruxEPRGw==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-musl": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.15.33.tgz", - "integrity": "sha512-hj628ZkSEJf6zMf5VMbYrG2O6QqyTIp2qwY6VlCjvIa9lAEZ5c2lfPblCLVGYubTeLJDxadLB/CxqQYOQABeEQ==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-arm64-msvc": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.15.33.tgz", - "integrity": "sha512-GV2oohtN2/5+KSccl86VULu3aT+LrISC8uzgSq0FRnikpD+Zwc+sBlXmoKQ+Db6jI57ITUOIB8jRkdGMABC29g==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-ia32-msvc": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.15.33.tgz", - "integrity": "sha512-gtyvzSNR8DHKfFEA2uqb8Ld1myqi6uEg2jyeUq3ikn5ytYs7H8RpZYC8mdy4NXr8hfcdJfCLXPlYaqqfBXpoEQ==", - "cpu": [ - "ia32" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-x64-msvc": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.15.33.tgz", - "integrity": "sha512-d6fRqQSkJI+kmMEBWaDQ7TMl8+YjLYbwRUPZQ9DY0ORBJeTzOrG0twvfvlZ2xgw6jA0ScQKgfBm4vHLSLl5Hqg==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/counter": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", - "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==", - "license": "Apache-2.0" - }, - "node_modules/@swc/helpers": { - "version": "0.5.15", - "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.15.tgz", - "integrity": "sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==", - "license": "Apache-2.0", - "dependencies": { - "tslib": "^2.8.0" - } - }, - "node_modules/@swc/types": { - "version": "0.1.26", - "resolved": "https://registry.npmjs.org/@swc/types/-/types-0.1.26.tgz", - "integrity": "sha512-lyMwd7WGgG79RS7EERZV3T8wMdmPq3xwyg+1nmAM64kIhx5yl+juO2PYIHb7vTiPgPCj8LYjsNV2T5wiQHUEaw==", - "license": "Apache-2.0", - "dependencies": { - "@swc/counter": "^0.1.3" - } - }, - "node_modules/@tailwindcss/node": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/node/-/node-4.3.0.tgz", - "integrity": "sha512-aFb4gUhFOgdh9AXo4IzBEOzBkkAxm9VigwDJnMIYv3lcfXCJVesNfbEaBl4BNgVRyid92AmdviqwBUBRKSeY3g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/remapping": "^2.3.5", - "enhanced-resolve": "^5.21.0", - "jiti": "^2.6.1", - "lightningcss": "1.32.0", - "magic-string": "^0.30.21", - "source-map-js": "^1.2.1", - "tailwindcss": "4.3.0" - } - }, - "node_modules/@tailwindcss/oxide": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide/-/oxide-4.3.0.tgz", - "integrity": "sha512-F7HZGBeN9I0/AuuJS5PwcD8xayx5ri5GhjYUDBEVYUkexyA/giwbDNjRVrxSezE3T250OU2K/wp/ltWx3UOefg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 20" - }, - "optionalDependencies": { - "@tailwindcss/oxide-android-arm64": "4.3.0", - "@tailwindcss/oxide-darwin-arm64": "4.3.0", - "@tailwindcss/oxide-darwin-x64": "4.3.0", - "@tailwindcss/oxide-freebsd-x64": "4.3.0", - "@tailwindcss/oxide-linux-arm-gnueabihf": "4.3.0", - "@tailwindcss/oxide-linux-arm64-gnu": "4.3.0", - "@tailwindcss/oxide-linux-arm64-musl": "4.3.0", - "@tailwindcss/oxide-linux-x64-gnu": "4.3.0", - "@tailwindcss/oxide-linux-x64-musl": "4.3.0", - "@tailwindcss/oxide-wasm32-wasi": "4.3.0", - "@tailwindcss/oxide-win32-arm64-msvc": "4.3.0", - "@tailwindcss/oxide-win32-x64-msvc": "4.3.0" - } - }, - "node_modules/@tailwindcss/oxide-android-arm64": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-android-arm64/-/oxide-android-arm64-4.3.0.tgz", - "integrity": "sha512-TJPiq67tKlLuObP6RkwvVGDoxCMBVtDgKkLfa/uyj7/FyxvQwHS+UOnVrXXgbEsfUaMgiVvC4KbJnRr26ho4Ng==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-darwin-arm64": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.3.0.tgz", - "integrity": "sha512-oMN/WZRb+SO37BmUElEgeEWuU8E/HXRkiODxJxLe1UTHVXLrdVSgfaJV7pSlhRGMSOiXLuxTIjfsF3wYvz8cgQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-darwin-x64": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-x64/-/oxide-darwin-x64-4.3.0.tgz", - "integrity": "sha512-N6CUmu4a6bKVADfw77p+iw6Yd9Q3OBhe0veaDX+QazfuVYlQsHfDgxBrsjQ/IW+zywL8mTrNd0SdJT/zgtvMdA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-freebsd-x64": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-freebsd-x64/-/oxide-freebsd-x64-4.3.0.tgz", - "integrity": "sha512-zDL5hBkQdH5C6MpqbK3gQAgP80tsMwSI26vjOzjJtNCMUo0lFgOItzHKBIupOZNQxt3ouPH7RPhvNhiTfCe5CQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-linux-arm-gnueabihf": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm-gnueabihf/-/oxide-linux-arm-gnueabihf-4.3.0.tgz", - "integrity": "sha512-R06HdNi7A7OEoMsf6d4tjZ71RCWnZQPHj2mnotSFURjNLdBC+cIgXQ7l81CqeoiQftjf6OOblxXMInMgN2VzMA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-linux-arm64-gnu": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-gnu/-/oxide-linux-arm64-gnu-4.3.0.tgz", - "integrity": "sha512-qTJHELX8jetjhRQHCLilkVLmybpzNQAtaI/gaoVoidn/ufbNDbAo8KlK2J+yPoc8wQxvDxCmh/5lr8nC1+lTbg==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-linux-arm64-musl": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-musl/-/oxide-linux-arm64-musl-4.3.0.tgz", - "integrity": "sha512-Z6sukiQsngnWO+l39X4pPbiWT81IC+PLKF+PHxIlyZbGNb9MODfYlXEVlFvej5BOZInWX01kVyzeLvHsXhfczQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-linux-x64-gnu": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-gnu/-/oxide-linux-x64-gnu-4.3.0.tgz", - "integrity": "sha512-DRNdQRpSGzRGfARVuVkxvM8Q12nh19l4BF/G7zGA1oe+9wcC6saFBHTISrpIcKzhiXtSrlSrluCfvMuledoCTQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-linux-x64-musl": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-musl/-/oxide-linux-x64-musl-4.3.0.tgz", - "integrity": "sha512-Z0IADbDo8bh6I7h2IQMx601AdXBLfFpEdUotft86evd/8ZPflZe9COPO8Q1vw+pfLWIUo9zN/JGZvwuAJqduqg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-wasm32-wasi": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-wasm32-wasi/-/oxide-wasm32-wasi-4.3.0.tgz", - "integrity": "sha512-HNZGOUxEmElksYR7S6sC5jTeNGpobAsy9u7Gu0AskJ8/20FR9GqebUyB+HBcU/ax6BHuiuJi+Oda4B+YX6H1yA==", - "bundleDependencies": [ - "@napi-rs/wasm-runtime", - "@emnapi/core", - "@emnapi/runtime", - "@tybys/wasm-util", - "@emnapi/wasi-threads", - "tslib" - ], - "cpu": [ - "wasm32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "@emnapi/core": "^1.10.0", - "@emnapi/runtime": "^1.10.0", - "@emnapi/wasi-threads": "^1.2.1", - "@napi-rs/wasm-runtime": "^1.1.4", - "@tybys/wasm-util": "^0.10.1", - "tslib": "^2.8.1" - }, - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/@tailwindcss/oxide-win32-arm64-msvc": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-arm64-msvc/-/oxide-win32-arm64-msvc-4.3.0.tgz", - "integrity": "sha512-Pe+RPVTi1T+qymuuRpcdvwSVZjnll/f7n8gBxMMh3xLTctMDKqpdfGimbMyioqtLhUYZxdJ9wGNhV7MKHvgZsQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/oxide-win32-x64-msvc": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-x64-msvc/-/oxide-win32-x64-msvc-4.3.0.tgz", - "integrity": "sha512-Mvrf2kXW/yeW/OTezZlCGOirXRcUuLIBx/5Y12BaPM7wJoryG6dfS/NJL8aBPqtTEx/Vm4T4vKzFUcKDT+TKUA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 20" - } - }, - "node_modules/@tailwindcss/postcss": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@tailwindcss/postcss/-/postcss-4.3.0.tgz", - "integrity": "sha512-Jm05Tjx+9yCLGv5qw1c+84Psds8MnyrEQYCB+FFk2lgGiUjlRqdxke4mVTuYrj2xnVZqKim2Apr5ySuQRYAw/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "@alloc/quick-lru": "^5.2.0", - "@tailwindcss/node": "4.3.0", - "@tailwindcss/oxide": "4.3.0", - "postcss": "^8.5.10", - "tailwindcss": "4.3.0" - } - }, - "node_modules/@tanstack/query-core": { - "version": "5.100.10", - "resolved": "https://registry.npmjs.org/@tanstack/query-core/-/query-core-5.100.10.tgz", - "integrity": "sha512-8UR0yJR+GiQ40m3lPhUr0xbfAupe6GSQiksSBSa9SM2NjezFyxXCIA69/lz8cSoNKZLrw1/PktIyQBJcVeMi3w==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/tannerlinsley" - } - }, - "node_modules/@tanstack/react-query": { - "version": "5.100.10", - "resolved": "https://registry.npmjs.org/@tanstack/react-query/-/react-query-5.100.10.tgz", - "integrity": "sha512-FLaZf2RCrA/Zgp4aiu5tG3TyasTRO7aZ99skxQpr3Hg/zXOhu6yq5FZCYQ/tRaJtM9ylnoK8tFK7PolXQadv6Q==", - "license": "MIT", - "dependencies": { - "@tanstack/query-core": "5.100.10" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/tannerlinsley" - }, - "peerDependencies": { - "react": "^18 || ^19" - } - }, - "node_modules/@tootallnate/quickjs-emscripten": { - "version": "0.23.0", - "resolved": "https://registry.npmjs.org/@tootallnate/quickjs-emscripten/-/quickjs-emscripten-0.23.0.tgz", - "integrity": "sha512-C5Mc6rdnsaJDjO3UpGW/CQTHtCKaYlScZTly4JIu97Jxo/odCiH0ITnDXSJPTOrEKk/ycSZ0AOgTmkDtkOsvIA==", - "license": "MIT" - }, - "node_modules/@ts-morph/common": { - "version": "0.27.0", - "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.27.0.tgz", - "integrity": "sha512-Wf29UqxWDpc+i61k3oIOzcUfQt79PIT9y/MWfAGlrkjg6lBC1hwDECLXPVJAhWjiGbfBCxZd65F/LIZF3+jeJQ==", - "license": "MIT", - "dependencies": { - "fast-glob": "^3.3.3", - "minimatch": "^10.0.1", - "path-browserify": "^1.0.1" - } - }, - "node_modules/@ts-morph/common/node_modules/balanced-match": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", - "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", - "license": "MIT", - "engines": { - "node": "18 || 20 || >=22" - } - }, - "node_modules/@ts-morph/common/node_modules/brace-expansion": { - "version": "5.0.6", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", - "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", - "license": "MIT", - "dependencies": { - "balanced-match": "^4.0.2" - }, - "engines": { - "node": "18 || 20 || >=22" - } - }, - "node_modules/@ts-morph/common/node_modules/fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.8" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/@ts-morph/common/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/@ts-morph/common/node_modules/minimatch": { - "version": "10.2.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", - "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", - "license": "BlueOak-1.0.0", - "dependencies": { - "brace-expansion": "^5.0.5" - }, - "engines": { - "node": "18 || 20 || >=22" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/@tybys/wasm-util": { - "version": "0.10.2", - "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz", - "integrity": "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==", - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "tslib": "^2.4.0" - } - }, - "node_modules/@types/better-sqlite3": { - "version": "7.6.13", - "resolved": "https://registry.npmjs.org/@types/better-sqlite3/-/better-sqlite3-7.6.13.tgz", - "integrity": "sha512-NMv9ASNARoKksWtsq/SHakpYAYnhBrQgGD8zkLYk/jaK8jUGn08CfEdTRgYhMypUQAfzSP8W6gNLe0q19/t4VA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@types/d3-array": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/@types/d3-array/-/d3-array-3.2.2.tgz", - "integrity": "sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw==", - "license": "MIT" - }, - "node_modules/@types/d3-color": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/@types/d3-color/-/d3-color-3.1.3.tgz", - "integrity": "sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A==", - "license": "MIT" - }, - "node_modules/@types/d3-ease": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@types/d3-ease/-/d3-ease-3.0.2.tgz", - "integrity": "sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA==", - "license": "MIT" - }, - "node_modules/@types/d3-interpolate": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-interpolate/-/d3-interpolate-3.0.4.tgz", - "integrity": "sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA==", - "license": "MIT", - "dependencies": { - "@types/d3-color": "*" - } - }, - "node_modules/@types/d3-path": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/@types/d3-path/-/d3-path-3.1.1.tgz", - "integrity": "sha512-VMZBYyQvbGmWyWVea0EHs/BwLgxc+MKi1zLDCONksozI4YJMcTt8ZEuIR4Sb1MMTE8MMW49v0IwI5+b7RmfWlg==", - "license": "MIT" - }, - "node_modules/@types/d3-scale": { - "version": "4.0.9", - "resolved": "https://registry.npmjs.org/@types/d3-scale/-/d3-scale-4.0.9.tgz", - "integrity": "sha512-dLmtwB8zkAeO/juAMfnV+sItKjlsw2lKdZVVy6LRr0cBmegxSABiLEpGVmSJJ8O08i4+sGR6qQtb6WtuwJdvVw==", - "license": "MIT", - "dependencies": { - "@types/d3-time": "*" - } - }, - "node_modules/@types/d3-shape": { - "version": "3.1.8", - "resolved": "https://registry.npmjs.org/@types/d3-shape/-/d3-shape-3.1.8.tgz", - "integrity": "sha512-lae0iWfcDeR7qt7rA88BNiqdvPS5pFVPpo5OfjElwNaT2yyekbM0C9vK+yqBqEmHr6lDkRnYNoTBYlAgJa7a4w==", - "license": "MIT", - "dependencies": { - "@types/d3-path": "*" - } - }, - "node_modules/@types/d3-time": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/d3-time/-/d3-time-3.0.4.tgz", - "integrity": "sha512-yuzZug1nkAAaBlBBikKZTgzCeA+k1uy4ZFwWANOfKw5z5LRhV0gNA7gNkKm7HoK+HRN0wX3EkxGk0fpbWhmB7g==", - "license": "MIT" - }, - "node_modules/@types/d3-timer": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@types/d3-timer/-/d3-timer-3.0.2.tgz", - "integrity": "sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw==", - "license": "MIT" - }, - "node_modules/@types/estree": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.9.tgz", - "integrity": "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/json-schema": { - "version": "7.0.15", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", - "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/node": { - "version": "20.19.41", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.41.tgz", - "integrity": "sha512-ECymXOukMnOoVkC2bb1Vc/w/836DXncOg5m8Xj1RH7xSHZJWNYY6Zh7EH477vcnD5egKNNfy2RpNOmuChhFPgQ==", - "license": "MIT", - "dependencies": { - "undici-types": "~6.21.0" - } - }, - "node_modules/@types/react": { - "version": "19.2.14", - "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.14.tgz", - "integrity": "sha512-ilcTH/UniCkMdtexkoCN0bI7pMcJDvmQFPvuPvmEaYA/NSfFTAgdUSLAoVjaRJm7+6PvcM+q1zYOwS4wTYMF9w==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "csstype": "^3.2.2" - } - }, - "node_modules/@types/react-dom": { - "version": "19.2.3", - "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz", - "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "@types/react": "^19.2.0" - } - }, - "node_modules/@types/set-cookie-parser": { - "version": "2.4.10", - "resolved": "https://registry.npmjs.org/@types/set-cookie-parser/-/set-cookie-parser-2.4.10.tgz", - "integrity": "sha512-GGmQVGpQWUe5qglJozEjZV/5dyxbOOZ0LHe/lqyWssB88Y4svNfst0uqBVscdDeIKl5Jy5+aPSvy7mI9tYRguw==", - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@types/statuses": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/@types/statuses/-/statuses-2.0.6.tgz", - "integrity": "sha512-xMAgYwceFhRA2zY+XbEA7mxYbA093wdiW8Vu6gZPGWy9cmOyU9XesH1tNcEWsKFd5Vzrqx5T3D38PWx1FIIXkA==", - "license": "MIT" - }, - "node_modules/@types/use-sync-external-store": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.6.tgz", - "integrity": "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg==", - "license": "MIT" - }, - "node_modules/@types/validate-npm-package-name": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/@types/validate-npm-package-name/-/validate-npm-package-name-4.0.2.tgz", - "integrity": "sha512-lrpDziQipxCEeK5kWxvljWYhUvOiB2A9izZd9B2AFarYAkqZshb4lPbRs7zKEic6eGtH8V/2qJW+dPp9OtF6bw==", - "license": "MIT" - }, - "node_modules/@types/yauzl": { - "version": "2.10.3", - "resolved": "https://registry.npmjs.org/@types/yauzl/-/yauzl-2.10.3.tgz", - "integrity": "sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q==", - "license": "MIT", - "optional": true, - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@typescript-eslint/eslint-plugin": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.59.3.tgz", - "integrity": "sha512-PwFvSKsXGShKGW6n5bZOhGHEcCZXM8HofLK9fNsEwZXzFRjoY+XT1Vsf1zgyXdwTr0ZYz1/2tkZ0DBTT9jZjhw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/regexpp": "^4.12.2", - "@typescript-eslint/scope-manager": "8.59.3", - "@typescript-eslint/type-utils": "8.59.3", - "@typescript-eslint/utils": "8.59.3", - "@typescript-eslint/visitor-keys": "8.59.3", - "ignore": "^7.0.5", - "natural-compare": "^1.4.0", - "ts-api-utils": "^2.5.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "@typescript-eslint/parser": "^8.59.3", - "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { - "version": "7.0.5", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", - "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, - "node_modules/@typescript-eslint/parser": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.59.3.tgz", - "integrity": "sha512-HPwA+hVkfcriajbNvTmZv4VRauibay+cWArYUYq7u7W7PmGShMxbPxLvrwDme55a6d5alG3nrYfhyJ/G28XlLg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/scope-manager": "8.59.3", - "@typescript-eslint/types": "8.59.3", - "@typescript-eslint/typescript-estree": "8.59.3", - "@typescript-eslint/visitor-keys": "8.59.3", - "debug": "^4.4.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/project-service": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.59.3.tgz", - "integrity": "sha512-ECiUWa/KYRGDFUqTNehaRgzDshnJfkTABJxVemHk4ko22gcr0ukloKjWvyQ64g8YCV/UI47kN1dbmjf/GaQYng==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/tsconfig-utils": "^8.59.3", - "@typescript-eslint/types": "^8.59.3", - "debug": "^4.4.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/scope-manager": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.59.3.tgz", - "integrity": "sha512-t2LvZnoEfzKtnPjgeEu41xw5gxq9mQVfYy4OoZ4Vlt0sk3JwxmhCca/AR7DwOiHrjWgjAj6as4AhRLKSDfvZIA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.59.3", - "@typescript-eslint/visitor-keys": "8.59.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/tsconfig-utils": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.59.3.tgz", - "integrity": "sha512-PcIJHjmaREXLgIAIzLnSY9VucEzz8FKXsRgFa1DmdGCK/5tJpW03TKJF01Q6VZd1lLdz2sIKPWaDUZN9dp//dw==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/type-utils": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.59.3.tgz", - "integrity": "sha512-g71d8QD8UaiHGvrJwyIS1hCX5r63w6Jll+4VEYhEAHXTDIqX1JgxhTAbEHtKntL9kuc4jRo7/GWw5xfCepSccQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.59.3", - "@typescript-eslint/typescript-estree": "8.59.3", - "@typescript-eslint/utils": "8.59.3", - "debug": "^4.4.3", - "ts-api-utils": "^2.5.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/types": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.59.3.tgz", - "integrity": "sha512-ePFoH0g4ludssdRFqqDxQePCxU4WQyRa9+XVwjm7yLn0FKhMeoetC+qBEEI1Eyb1pGSDveTIT09Bvw2WhlGayg==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/typescript-estree": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.59.3.tgz", - "integrity": "sha512-CbRjVRAf7Lr9Kr8RopKcbY45p2VfmmHrm0ygOCYFi7oU8q19m0Fs/6iHS7kNOmwpp+ob07ZVcAqlxUod9lYdmg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/project-service": "8.59.3", - "@typescript-eslint/tsconfig-utils": "8.59.3", - "@typescript-eslint/types": "8.59.3", - "@typescript-eslint/visitor-keys": "8.59.3", - "debug": "^4.4.3", - "minimatch": "^10.2.2", - "semver": "^7.7.3", - "tinyglobby": "^0.2.15", - "ts-api-utils": "^2.5.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/balanced-match": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", - "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", - "dev": true, - "license": "MIT", - "engines": { - "node": "18 || 20 || >=22" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { - "version": "5.0.6", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", - "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^4.0.2" - }, - "engines": { - "node": "18 || 20 || >=22" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { - "version": "10.2.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", - "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", - "dev": true, - "license": "BlueOak-1.0.0", - "dependencies": { - "brace-expansion": "^5.0.5" - }, - "engines": { - "node": "18 || 20 || >=22" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { - "version": "7.8.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", - "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/@typescript-eslint/utils": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.59.3.tgz", - "integrity": "sha512-JAvT14goBzRzzzZyqq3P9BLArIxTtQURUtFgQ/V7FO+eU+Gg6ES+5ymOPP1wRxXcxAYeivCk4uS3jCKWI1K8Zg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.9.1", - "@typescript-eslint/scope-manager": "8.59.3", - "@typescript-eslint/types": "8.59.3", - "@typescript-eslint/typescript-estree": "8.59.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/@typescript-eslint/visitor-keys": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.59.3.tgz", - "integrity": "sha512-f1UQF7ggd42YiwI5wGrRaPsa+P0CINBlrkLPmGfpq/u/I/oVtecoEIfFR9ag/oa1sLOsRNZ6xehf6qMZhQGBDg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/types": "8.59.3", - "eslint-visitor-keys": "^5.0.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - } - }, - "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", - "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^20.19.0 || ^22.13.0 || >=24" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/@unrs/resolver-binding-android-arm-eabi": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm-eabi/-/resolver-binding-android-arm-eabi-1.11.1.tgz", - "integrity": "sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@unrs/resolver-binding-android-arm64": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm64/-/resolver-binding-android-arm64-1.11.1.tgz", - "integrity": "sha512-lCxkVtb4wp1v+EoN+HjIG9cIIzPkX5OtM03pQYkG+U5O/wL53LC4QbIeazgiKqluGeVEeBlZahHalCaBvU1a2g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@unrs/resolver-binding-darwin-arm64": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-arm64/-/resolver-binding-darwin-arm64-1.11.1.tgz", - "integrity": "sha512-gPVA1UjRu1Y/IsB/dQEsp2V1pm44Of6+LWvbLc9SDk1c2KhhDRDBUkQCYVWe6f26uJb3fOK8saWMgtX8IrMk3g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@unrs/resolver-binding-darwin-x64": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-x64/-/resolver-binding-darwin-x64-1.11.1.tgz", - "integrity": "sha512-cFzP7rWKd3lZaCsDze07QX1SC24lO8mPty9vdP+YVa3MGdVgPmFc59317b2ioXtgCMKGiCLxJ4HQs62oz6GfRQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ] - }, - "node_modules/@unrs/resolver-binding-freebsd-x64": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-freebsd-x64/-/resolver-binding-freebsd-x64-1.11.1.tgz", - "integrity": "sha512-fqtGgak3zX4DCB6PFpsH5+Kmt/8CIi4Bry4rb1ho6Av2QHTREM+47y282Uqiu3ZRF5IQioJQ5qWRV6jduA+iGw==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] - }, - "node_modules/@unrs/resolver-binding-linux-arm-gnueabihf": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-gnueabihf/-/resolver-binding-linux-arm-gnueabihf-1.11.1.tgz", - "integrity": "sha512-u92mvlcYtp9MRKmP+ZvMmtPN34+/3lMHlyMj7wXJDeXxuM0Vgzz0+PPJNsro1m3IZPYChIkn944wW8TYgGKFHw==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-arm-musleabihf": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-musleabihf/-/resolver-binding-linux-arm-musleabihf-1.11.1.tgz", - "integrity": "sha512-cINaoY2z7LVCrfHkIcmvj7osTOtm6VVT16b5oQdS4beibX2SYBwgYLmqhBjA1t51CarSaBuX5YNsWLjsqfW5Cw==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-arm64-gnu": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-gnu/-/resolver-binding-linux-arm64-gnu-1.11.1.tgz", - "integrity": "sha512-34gw7PjDGB9JgePJEmhEqBhWvCiiWCuXsL9hYphDF7crW7UgI05gyBAi6MF58uGcMOiOqSJ2ybEeCvHcq0BCmQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-arm64-musl": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-musl/-/resolver-binding-linux-arm64-musl-1.11.1.tgz", - "integrity": "sha512-RyMIx6Uf53hhOtJDIamSbTskA99sPHS96wxVE/bJtePJJtpdKGXO1wY90oRdXuYOGOTuqjT8ACccMc4K6QmT3w==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-ppc64-gnu": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-ppc64-gnu/-/resolver-binding-linux-ppc64-gnu-1.11.1.tgz", - "integrity": "sha512-D8Vae74A4/a+mZH0FbOkFJL9DSK2R6TFPC9M+jCWYia/q2einCubX10pecpDiTmkJVUH+y8K3BZClycD8nCShA==", - "cpu": [ - "ppc64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-riscv64-gnu": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-gnu/-/resolver-binding-linux-riscv64-gnu-1.11.1.tgz", - "integrity": "sha512-frxL4OrzOWVVsOc96+V3aqTIQl1O2TjgExV4EKgRY09AJ9leZpEg8Ak9phadbuX0BA4k8U5qtvMSQQGGmaJqcQ==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-riscv64-musl": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-musl/-/resolver-binding-linux-riscv64-musl-1.11.1.tgz", - "integrity": "sha512-mJ5vuDaIZ+l/acv01sHoXfpnyrNKOk/3aDoEdLO/Xtn9HuZlDD6jKxHlkN8ZhWyLJsRBxfv9GYM2utQ1SChKew==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-s390x-gnu": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-s390x-gnu/-/resolver-binding-linux-s390x-gnu-1.11.1.tgz", - "integrity": "sha512-kELo8ebBVtb9sA7rMe1Cph4QHreByhaZ2QEADd9NzIQsYNQpt9UkM9iqr2lhGr5afh885d/cB5QeTXSbZHTYPg==", - "cpu": [ - "s390x" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-x64-gnu": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-gnu/-/resolver-binding-linux-x64-gnu-1.11.1.tgz", - "integrity": "sha512-C3ZAHugKgovV5YvAMsxhq0gtXuwESUKc5MhEtjBpLoHPLYM+iuwSj3lflFwK3DPm68660rZ7G8BMcwSro7hD5w==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-linux-x64-musl": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-musl/-/resolver-binding-linux-x64-musl-1.11.1.tgz", - "integrity": "sha512-rV0YSoyhK2nZ4vEswT/QwqzqQXw5I6CjoaYMOX0TqBlWhojUf8P94mvI7nuJTeaCkkds3QE4+zS8Ko+GdXuZtA==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@unrs/resolver-binding-wasm32-wasi": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-wasm32-wasi/-/resolver-binding-wasm32-wasi-1.11.1.tgz", - "integrity": "sha512-5u4RkfxJm+Ng7IWgkzi3qrFOvLvQYnPBmjmZQ8+szTK/b31fQCnleNl1GgEt7nIsZRIf5PLhPwT0WM+q45x/UQ==", - "cpu": [ - "wasm32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "dependencies": { - "@napi-rs/wasm-runtime": "^0.2.11" - }, - "engines": { - "node": ">=14.0.0" - } - }, - "node_modules/@unrs/resolver-binding-win32-arm64-msvc": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-arm64-msvc/-/resolver-binding-win32-arm64-msvc-1.11.1.tgz", - "integrity": "sha512-nRcz5Il4ln0kMhfL8S3hLkxI85BXs3o8EYoattsJNdsX4YUU89iOkVn7g0VHSRxFuVMdM4Q1jEpIId1Ihim/Uw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@unrs/resolver-binding-win32-ia32-msvc": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-ia32-msvc/-/resolver-binding-win32-ia32-msvc-1.11.1.tgz", - "integrity": "sha512-DCEI6t5i1NmAZp6pFonpD5m7i6aFrpofcp4LA2i8IIq60Jyo28hamKBxNrZcyOwVOZkgsRp9O2sXWBWP8MnvIQ==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@unrs/resolver-binding-win32-x64-msvc": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-x64-msvc/-/resolver-binding-win32-x64-msvc-1.11.1.tgz", - "integrity": "sha512-lrW200hZdbfRtztbygyaq/6jP6AKE8qQN2KvPcJ+x7wiD038YtnYtZ82IMNJ69GJibV7bwL3y9FgK+5w/pYt6g==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/accepts": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", - "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==", - "license": "MIT", - "dependencies": { - "mime-types": "^3.0.0", - "negotiator": "^1.0.0" - }, - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/acorn": { - "version": "8.16.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", - "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", - "dev": true, - "license": "MIT", - "bin": { - "acorn": "bin/acorn" - }, - "engines": { - "node": ">=0.4.0" - } - }, - "node_modules/acorn-jsx": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", - "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", - "dev": true, - "license": "MIT", - "peerDependencies": { - "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" - } - }, - "node_modules/agent-base": { - "version": "7.1.4", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.4.tgz", - "integrity": "sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==", - "license": "MIT", - "engines": { - "node": ">= 14" - } - }, - "node_modules/ajv": { - "version": "6.15.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz", - "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==", - "dev": true, - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/ajv-formats": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-3.0.1.tgz", - "integrity": "sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==", - "license": "MIT", - "dependencies": { - "ajv": "^8.0.0" - }, - "peerDependencies": { - "ajv": "^8.0.0" - }, - "peerDependenciesMeta": { - "ajv": { - "optional": true - } - } - }, - "node_modules/ajv-formats/node_modules/ajv": { - "version": "8.20.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.20.0.tgz", - "integrity": "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA==", - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.3", - "fast-uri": "^3.0.1", - "json-schema-traverse": "^1.0.0", - "require-from-string": "^2.0.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/ajv-formats/node_modules/json-schema-traverse": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", - "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", - "license": "MIT" - }, - "node_modules/ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "license": "MIT", - "dependencies": { - "color-convert": "^2.0.1" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "license": "Python-2.0" - }, - "node_modules/aria-query": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.2.tgz", - "integrity": "sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/array-buffer-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", - "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "is-array-buffer": "^3.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array-includes": { - "version": "3.1.9", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.9.tgz", - "integrity": "sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.24.0", - "es-object-atoms": "^1.1.1", - "get-intrinsic": "^1.3.0", - "is-string": "^1.1.1", - "math-intrinsics": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.findlast": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", - "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.findlastindex": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.6.tgz", - "integrity": "sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "es-shim-unscopables": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flat": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz", - "integrity": "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flatmap": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz", - "integrity": "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.tosorted": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.4.tgz", - "integrity": "sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.3", - "es-errors": "^1.3.0", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/arraybuffer.prototype.slice": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", - "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-buffer-byte-length": "^1.0.1", - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "is-array-buffer": "^3.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/ast-types": { - "version": "0.13.4", - "resolved": "https://registry.npmjs.org/ast-types/-/ast-types-0.13.4.tgz", - "integrity": "sha512-x1FCFnFifvYDDzTaLII71vG5uvDwgtmDTEVWAxrgeiR8VjMONcCXJx7E+USjDtHlwFmt9MysbqgF9b9Vjr6w+w==", - "license": "MIT", - "dependencies": { - "tslib": "^2.0.1" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/ast-types-flow": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.8.tgz", - "integrity": "sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/async-function": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/async-function/-/async-function-1.0.0.tgz", - "integrity": "sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/available-typed-arrays": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", - "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "possible-typed-array-names": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/axe-core": { - "version": "4.11.4", - "resolved": "https://registry.npmjs.org/axe-core/-/axe-core-4.11.4.tgz", - "integrity": "sha512-KunSNx+TVpkAw/6ULfhnx+HWRecjqZGTOyquAoWHYLRSdK1tB5Ihce1ZW+UY3fj33bYAFWPu7W/GRSmmrCGuxA==", - "dev": true, - "license": "MPL-2.0", - "engines": { - "node": ">=4" - } - }, - "node_modules/axobject-query": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-4.1.0.tgz", - "integrity": "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/b4a": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/b4a/-/b4a-1.8.1.tgz", - "integrity": "sha512-aiqre1Nr0B/6DgE2N5vwTc+2/oQZ4Wh1t4NznYY4E00y8LCt6NqdRv81so00oo27D8MVKTpUa/MwUUtBLXCoDw==", - "license": "Apache-2.0", - "peerDependencies": { - "react-native-b4a": "*" - }, - "peerDependenciesMeta": { - "react-native-b4a": { - "optional": true - } - } - }, - "node_modules/balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", - "dev": true, - "license": "MIT" - }, - "node_modules/bare-events": { - "version": "2.8.2", - "resolved": "https://registry.npmjs.org/bare-events/-/bare-events-2.8.2.tgz", - "integrity": "sha512-riJjyv1/mHLIPX4RwiK+oW9/4c3TEUeORHKefKAKnZ5kyslbN+HXowtbaVEqt4IMUB7OXlfixcs6gsFeo/jhiQ==", - "license": "Apache-2.0", - "peerDependencies": { - "bare-abort-controller": "*" - }, - "peerDependenciesMeta": { - "bare-abort-controller": { - "optional": true - } - } - }, - "node_modules/bare-fs": { - "version": "4.7.1", - "resolved": "https://registry.npmjs.org/bare-fs/-/bare-fs-4.7.1.tgz", - "integrity": "sha512-WDRsyVN52eAx/lBamKD6uyw8H4228h/x0sGGGegOamM2cd7Pag88GfMQalobXI+HaEUxpCkbKQUDOQqt9wawRw==", - "license": "Apache-2.0", - "dependencies": { - "bare-events": "^2.5.4", - "bare-path": "^3.0.0", - "bare-stream": "^2.6.4", - "bare-url": "^2.2.2", - "fast-fifo": "^1.3.2" - }, - "engines": { - "bare": ">=1.16.0" - }, - "peerDependencies": { - "bare-buffer": "*" - }, - "peerDependenciesMeta": { - "bare-buffer": { - "optional": true - } - } - }, - "node_modules/bare-os": { - "version": "3.9.1", - "resolved": "https://registry.npmjs.org/bare-os/-/bare-os-3.9.1.tgz", - "integrity": "sha512-6M5XjcnsygQNPMCMPXSK379xrJFiZ/AEMNBmFEmQW8d/789VQATvriyi5r0HYTL9TkQ26rn3kgdTG3aisbrXkQ==", - "license": "Apache-2.0", - "engines": { - "bare": ">=1.14.0" - } - }, - "node_modules/bare-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/bare-path/-/bare-path-3.0.0.tgz", - "integrity": "sha512-tyfW2cQcB5NN8Saijrhqn0Zh7AnFNsnczRcuWODH0eYAXBsJ5gVxAUuNr7tsHSC6IZ77cA0SitzT+s47kot8Mw==", - "license": "Apache-2.0", - "dependencies": { - "bare-os": "^3.0.1" - } - }, - "node_modules/bare-stream": { - "version": "2.13.1", - "resolved": "https://registry.npmjs.org/bare-stream/-/bare-stream-2.13.1.tgz", - "integrity": "sha512-Vp0cnjYyrEC4whYTymQ+YZi6pBpfiICZO3cfRG8sy67ZNWe951urv1x4eW1BKNngw3U+3fPYb5JQvHbCtxH7Ow==", - "license": "Apache-2.0", - "dependencies": { - "streamx": "^2.25.0", - "teex": "^1.0.1" - }, - "peerDependencies": { - "bare-abort-controller": "*", - "bare-buffer": "*", - "bare-events": "*" - }, - "peerDependenciesMeta": { - "bare-abort-controller": { - "optional": true - }, - "bare-buffer": { - "optional": true - }, - "bare-events": { - "optional": true - } - } - }, - "node_modules/bare-url": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/bare-url/-/bare-url-2.4.3.tgz", - "integrity": "sha512-Kccpc7ACfXaxfeInfqKcZtW4pT5YBn1mesc4sCsun6sRwtbJ4h+sNOaksUpYEJUKfN65YWC6Bw2OJEFiKxq8nQ==", - "license": "Apache-2.0", - "dependencies": { - "bare-path": "^3.0.0" - } - }, - "node_modules/base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/baseline-browser-mapping": { - "version": "2.10.29", - "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.29.tgz", - "integrity": "sha512-Asa2krT+XTPZINCS+2QcyS8WTkObE77RwkydwF7h6DmnKqbvlalz93m/dnphUyCa6SWSP51VgtEUf2FN+gelFQ==", - "license": "Apache-2.0", - "bin": { - "baseline-browser-mapping": "dist/cli.cjs" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/basic-ftp": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/basic-ftp/-/basic-ftp-5.3.1.tgz", - "integrity": "sha512-bopVNp6ugyA150DDuZfPFdt1KZ5a94ZDiwX4hMgZDzF+GttD80lEy8kj98kbyhLXnPvhtIo93mdnLIjpCAeeOw==", - "license": "MIT", - "engines": { - "node": ">=10.0.0" - } - }, - "node_modules/better-sqlite3": { - "version": "12.10.0", - "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-12.10.0.tgz", - "integrity": "sha512-CyzaZRQKyHkB2ZInfTTl2nvT33EbDpjkLEbE8/Zck3Ll6O0qqvuGdrJ45HgtH+HykRg88ITY3AdreBGN70aBSQ==", - "hasInstallScript": true, - "license": "MIT", - "dependencies": { - "bindings": "^1.5.0", - "prebuild-install": "^7.1.1" - }, - "engines": { - "node": "20.x || 22.x || 23.x || 24.x || 25.x || 26.x" - } - }, - "node_modules/bindings": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", - "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", - "license": "MIT", - "dependencies": { - "file-uri-to-path": "1.0.0" - } - }, - "node_modules/bl": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", - "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", - "license": "MIT", - "dependencies": { - "buffer": "^5.5.0", - "inherits": "^2.0.4", - "readable-stream": "^3.4.0" - } - }, - "node_modules/body-parser": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz", - "integrity": "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA==", - "license": "MIT", - "dependencies": { - "bytes": "^3.1.2", - "content-type": "^1.0.5", - "debug": "^4.4.3", - "http-errors": "^2.0.0", - "iconv-lite": "^0.7.0", - "on-finished": "^2.4.1", - "qs": "^6.14.1", - "raw-body": "^3.0.1", - "type-is": "^2.0.1" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/brace-expansion": { - "version": "1.1.14", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", - "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/braces": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", - "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", - "license": "MIT", - "dependencies": { - "fill-range": "^7.1.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/browserslist": { - "version": "4.28.2", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz", - "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==", - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/browserslist" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "baseline-browser-mapping": "^2.10.12", - "caniuse-lite": "^1.0.30001782", - "electron-to-chromium": "^1.5.328", - "node-releases": "^2.0.36", - "update-browserslist-db": "^1.2.3" - }, - "bin": { - "browserslist": "cli.js" - }, - "engines": { - "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" - } - }, - "node_modules/buffer": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", - "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "base64-js": "^1.3.1", - "ieee754": "^1.1.13" - } - }, - "node_modules/buffer-crc32": { - "version": "0.2.13", - "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", - "integrity": "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ==", - "license": "MIT", - "engines": { - "node": "*" - } - }, - "node_modules/bundle-name": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/bundle-name/-/bundle-name-4.1.0.tgz", - "integrity": "sha512-tjwM5exMg6BGRI+kNmTntNsvdZS1X8BFYS6tnJ2hdH0kVxM6/eVZ2xy+FqStSWvYmtfFMDLIxurorHwDKfDz5Q==", - "license": "MIT", - "dependencies": { - "run-applescript": "^7.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/bytes": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", - "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/call-bind": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.9.tgz", - "integrity": "sha512-a/hy+pNsFUTR+Iz8TCJvXudKVLAnz/DyeSUo10I5yvFDQJBFU2s9uqQpoSrJlroHUKoKqzg+epxyP9lqFdzfBQ==", - "dev": true, - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/call-bind-apply-helpers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", - "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/call-bound": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", - "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "get-intrinsic": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/caniuse-lite": { - "version": "1.0.30001792", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001792.tgz", - "integrity": "sha512-hVLMUZFgR4JJ6ACt1uEESvQN1/dBVqPAKY0hgrV70eN3391K6juAfTjKZLKvOMsx8PxA7gsY1/tLMMTcfFLLpw==", - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/caniuse-lite" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "CC-BY-4.0" - }, - "node_modules/chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" - } - }, - "node_modules/chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", - "license": "ISC" - }, - "node_modules/chromium-bidi": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/chromium-bidi/-/chromium-bidi-14.0.0.tgz", - "integrity": "sha512-9gYlLtS6tStdRWzrtXaTMnqcM4dudNegMXJxkR0I/CXObHalYeYcAMPrL19eroNZHtJ8DQmu1E+ZNOYu/IXMXw==", - "license": "Apache-2.0", - "dependencies": { - "mitt": "^3.0.1", - "zod": "^3.24.1" - }, - "peerDependencies": { - "devtools-protocol": "*" - } - }, - "node_modules/chromium-bidi/node_modules/zod": { - "version": "3.25.76", - "resolved": "https://registry.npmjs.org/zod/-/zod-3.25.76.tgz", - "integrity": "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/colinhacks" - } - }, - "node_modules/class-variance-authority": { - "version": "0.7.1", - "resolved": "https://registry.npmjs.org/class-variance-authority/-/class-variance-authority-0.7.1.tgz", - "integrity": "sha512-Ka+9Trutv7G8M6WT6SeiRWz792K5qEqIGEGzXKhAE6xOWAY6pPH8U+9IY3oCMv6kqTmLsv7Xh/2w2RigkePMsg==", - "license": "Apache-2.0", - "dependencies": { - "clsx": "^2.1.1" - }, - "funding": { - "url": "https://polar.sh/cva" - } - }, - "node_modules/cli-cursor": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-5.0.0.tgz", - "integrity": "sha512-aCj4O5wKyszjMmDT4tZj93kxyydN/K5zPWSCe6/0AV/AA1pqe5ZBIw0a2ZfPQV7lL5/yb5HsUreJ6UFAF1tEQw==", - "license": "MIT", - "dependencies": { - "restore-cursor": "^5.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/cli-spinners": { - "version": "2.9.2", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", - "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", - "license": "MIT", - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/cli-width": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-4.1.0.tgz", - "integrity": "sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==", - "license": "ISC", - "engines": { - "node": ">= 12" - } - }, - "node_modules/client-only": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/client-only/-/client-only-0.0.1.tgz", - "integrity": "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==", - "license": "MIT" - }, - "node_modules/cliui": { - "version": "8.0.1", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", - "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", - "license": "ISC", - "dependencies": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.1", - "wrap-ansi": "^7.0.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/clsx": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", - "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/code-block-writer": { - "version": "13.0.3", - "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-13.0.3.tgz", - "integrity": "sha512-Oofo0pq3IKnsFtuHqSF7TqBfr71aeyZDVJ0HpmqB7FBM2qEigL0iPONSCZSO9pE9dZTAxANe5XHG9Uy0YMv8cg==", - "license": "MIT" - }, - "node_modules/color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "license": "MIT", - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "license": "MIT" - }, - "node_modules/commander": { - "version": "14.0.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-14.0.3.tgz", - "integrity": "sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw==", - "license": "MIT", - "engines": { - "node": ">=20" - } - }, - "node_modules/concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", - "dev": true, - "license": "MIT" - }, - "node_modules/content-disposition": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.1.0.tgz", - "integrity": "sha512-5jRCH9Z/+DRP7rkvY83B+yGIGX96OYdJmzngqnw2SBSxqCFPd0w2km3s5iawpGX8krnwSGmF0FW5Nhr0Hfai3g==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/content-type": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", - "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/convert-source-map": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", - "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", - "license": "MIT" - }, - "node_modules/cookie": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", - "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/cookie-signature": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz", - "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==", - "license": "MIT", - "engines": { - "node": ">=6.6.0" - } - }, - "node_modules/cors": { - "version": "2.8.6", - "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.6.tgz", - "integrity": "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw==", - "license": "MIT", - "dependencies": { - "object-assign": "^4", - "vary": "^1" - }, - "engines": { - "node": ">= 0.10" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/cosmiconfig": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-9.0.1.tgz", - "integrity": "sha512-hr4ihw+DBqcvrsEDioRO31Z17x71pUYoNe/4h6Z0wB72p7MU7/9gH8Q3s12NFhHPfYBBOV3qyfUxmr/Yn3shnQ==", - "license": "MIT", - "dependencies": { - "env-paths": "^2.2.1", - "import-fresh": "^3.3.0", - "js-yaml": "^4.1.0", - "parse-json": "^5.2.0" - }, - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/d-fischer" - }, - "peerDependencies": { - "typescript": ">=4.9.5" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/cross-spawn": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", - "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", - "license": "MIT", - "dependencies": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/cssesc": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", - "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", - "license": "MIT", - "bin": { - "cssesc": "bin/cssesc" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/csstype": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", - "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", - "devOptional": true, - "license": "MIT" - }, - "node_modules/d3-array": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz", - "integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==", - "license": "ISC", - "dependencies": { - "internmap": "1 - 2" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-color": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz", - "integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-ease": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz", - "integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==", - "license": "BSD-3-Clause", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-format": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.2.tgz", - "integrity": "sha512-AJDdYOdnyRDV5b6ArilzCPPwc1ejkHcoyFarqlPqT7zRYjhavcT3uSrqcMvsgh2CgoPbK3RCwyHaVyxYcP2Arg==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-interpolate": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz", - "integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==", - "license": "ISC", - "dependencies": { - "d3-color": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-path": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz", - "integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-scale": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz", - "integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==", - "license": "ISC", - "dependencies": { - "d3-array": "2.10.0 - 3", - "d3-format": "1 - 3", - "d3-interpolate": "1.2.0 - 3", - "d3-time": "2.1.1 - 3", - "d3-time-format": "2 - 4" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-shape": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz", - "integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==", - "license": "ISC", - "dependencies": { - "d3-path": "^3.1.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-time": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz", - "integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==", - "license": "ISC", - "dependencies": { - "d3-array": "2 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-time-format": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz", - "integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==", - "license": "ISC", - "dependencies": { - "d3-time": "1 - 3" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/d3-timer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz", - "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/damerau-levenshtein": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz", - "integrity": "sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA==", - "dev": true, - "license": "BSD-2-Clause" - }, - "node_modules/data-uri-to-buffer": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-6.0.2.tgz", - "integrity": "sha512-7hvf7/GW8e86rW0ptuwS3OcBGDjIi6SZva7hCyWC0yYry2cOPmLIjXAUHI6DK2HsnwJd9ifmt57i8eV2n4YNpw==", - "license": "MIT", - "engines": { - "node": ">= 14" - } - }, - "node_modules/data-view-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", - "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/data-view-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", - "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/inspect-js" - } - }, - "node_modules/data-view-byte-offset": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", - "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/debug": { - "version": "4.4.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", - "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", - "license": "MIT", - "dependencies": { - "ms": "^2.1.3" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/decimal.js-light": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/decimal.js-light/-/decimal.js-light-2.5.1.tgz", - "integrity": "sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg==", - "license": "MIT" - }, - "node_modules/decompress-response": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", - "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", - "license": "MIT", - "dependencies": { - "mimic-response": "^3.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/dedent": { - "version": "1.7.2", - "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.2.tgz", - "integrity": "sha512-WzMx3mW98SN+zn3hgemf4OzdmyNhhhKz5Ay0pUfQiMQ3e1g+xmTJWp/pKdwKVXhdSkAEGIIzqeuWrL3mV/AXbA==", - "license": "MIT", - "peerDependencies": { - "babel-plugin-macros": "^3.1.0" - }, - "peerDependenciesMeta": { - "babel-plugin-macros": { - "optional": true - } - } - }, - "node_modules/deep-extend": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", - "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", - "license": "MIT", - "engines": { - "node": ">=4.0.0" - } - }, - "node_modules/deep-is": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", - "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/deepmerge": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", - "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/default-browser": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/default-browser/-/default-browser-5.5.0.tgz", - "integrity": "sha512-H9LMLr5zwIbSxrmvikGuI/5KGhZ8E2zH3stkMgM5LpOWDutGM2JZaj460Udnf1a+946zc7YBgrqEWwbk7zHvGw==", - "license": "MIT", - "dependencies": { - "bundle-name": "^4.1.0", - "default-browser-id": "^5.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser-id": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/default-browser-id/-/default-browser-id-5.0.1.tgz", - "integrity": "sha512-x1VCxdX4t+8wVfd1so/9w+vQ4vx7lKd2Qp5tDRutErwmR85OgmfX7RlLRMWafRMY7hbEiXIbudNrjOAPa/hL8Q==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/define-data-property": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", - "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/define-lazy-prop": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-3.0.0.tgz", - "integrity": "sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/define-properties": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", - "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.0.1", - "has-property-descriptors": "^1.0.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/degenerator": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/degenerator/-/degenerator-5.0.1.tgz", - "integrity": "sha512-TllpMR/t0M5sqCXfj85i4XaAzxmS5tVA16dqvdkMwGmzI+dXLXnw3J+3Vdv7VKw+ThlTMboK6i9rnZ6Nntj5CQ==", - "license": "MIT", - "dependencies": { - "ast-types": "^0.13.4", - "escodegen": "^2.1.0", - "esprima": "^4.0.1" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/depd": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", - "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/detect-libc": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", - "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", - "license": "Apache-2.0", - "engines": { - "node": ">=8" - } - }, - "node_modules/devtools-protocol": { - "version": "0.0.1608973", - "resolved": "https://registry.npmjs.org/devtools-protocol/-/devtools-protocol-0.0.1608973.tgz", - "integrity": "sha512-Tpm17fxYzt+J7VrGdc1k8YdRqS3YV7se/M6KeemEqvUbq/n7At1rWVuXMxQgpWkdwSdIEKYbU//Bve+Shm4YNQ==", - "license": "BSD-3-Clause" - }, - "node_modules/diff": { - "version": "8.0.4", - "resolved": "https://registry.npmjs.org/diff/-/diff-8.0.4.tgz", - "integrity": "sha512-DPi0FmjiSU5EvQV0++GFDOJ9ASQUVFh5kD+OzOnYdi7n3Wpm9hWWGfB/O2blfHcMVTL5WkQXSnRiK9makhrcnw==", - "license": "BSD-3-Clause", - "engines": { - "node": ">=0.3.1" - } - }, - "node_modules/doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/dotenv": { - "version": "17.4.2", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.4.2.tgz", - "integrity": "sha512-nI4U3TottKAcAD9LLud4Cb7b2QztQMUEfHbvhTH09bqXTxnSie8WnjPALV/WMCrJZ6UV/qHJ6L03OqO3LcdYZw==", - "license": "BSD-2-Clause", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://dotenvx.com" - } - }, - "node_modules/dunder-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", - "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/eciesjs": { - "version": "0.4.18", - "resolved": "https://registry.npmjs.org/eciesjs/-/eciesjs-0.4.18.tgz", - "integrity": "sha512-wG99Zcfcys9fZux7Cft8BAX/YrOJLJSZ3jyYPfhZHqN2E+Ffx+QXBDsv3gubEgPtV6dTzJMSQUwk1H98/t/0wQ==", - "license": "MIT", - "dependencies": { - "@ecies/ciphers": "^0.2.5", - "@noble/ciphers": "^1.3.0", - "@noble/curves": "^1.9.7", - "@noble/hashes": "^1.8.0" - }, - "engines": { - "bun": ">=1", - "deno": ">=2", - "node": ">=16" - } - }, - "node_modules/ee-first": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", - "license": "MIT" - }, - "node_modules/electron-to-chromium": { - "version": "1.5.354", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.354.tgz", - "integrity": "sha512-JaBHwWcfIdmSAfWM5l3uwjGd431j8YEMikZ+K/2nXVuBqJKyZ0f+2h4n4JY5AyNiZmnY9qQr2RU3v9DxDmHMNg==", - "license": "ISC" - }, - "node_modules/emoji-regex": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", - "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", - "dev": true, - "license": "MIT" - }, - "node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/end-of-stream": { - "version": "1.4.5", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz", - "integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==", - "license": "MIT", - "dependencies": { - "once": "^1.4.0" - } - }, - "node_modules/enhanced-resolve": { - "version": "5.21.3", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.21.3.tgz", - "integrity": "sha512-QyL119InA+XXEkNLNTPCXPugSvOfhwv0JOlGNzvxs0hZaiHLNvXSpudUWsOlsXGWJh8G6ckCScEkVHfX3kw/2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "graceful-fs": "^4.2.4", - "tapable": "^2.3.3" - }, - "engines": { - "node": ">=10.13.0" - } - }, - "node_modules/env-paths": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", - "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/error-ex": { - "version": "1.3.4", - "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", - "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", - "license": "MIT", - "dependencies": { - "is-arrayish": "^0.2.1" - } - }, - "node_modules/es-abstract": { - "version": "1.24.2", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.24.2.tgz", - "integrity": "sha512-2FpH9Q5i2RRwyEP1AylXe6nYLR5OhaJTZwmlcP0dL/+JCbgg7yyEo/sEK6HeGZRf3dFpWwThaRHVApXSkW3xeg==", - "dev": true, - "license": "MIT", - "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.19" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/es-define-property": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", - "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-errors": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", - "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-iterator-helpers": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.3.2.tgz", - "integrity": "sha512-HVLACW1TppGYjJ8H6/jqH/pqOtKRw6wMlrB23xfExmFWxFquAIWCmwoLsOyN96K4a5KbmOf5At9ZUO3GZbetAw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.9", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.24.2", - "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", - "math-intrinsics": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-object-atoms": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", - "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-set-tostringtag": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", - "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-shim-unscopables": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.1.0.tgz", - "integrity": "sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-to-primitive": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", - "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-callable": "^1.2.7", - "is-date-object": "^1.0.5", - "is-symbol": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/es-toolkit": { - "version": "1.46.1", - "resolved": "https://registry.npmjs.org/es-toolkit/-/es-toolkit-1.46.1.tgz", - "integrity": "sha512-5eNtXOs3tbfxXOj04tjjseeWkRWaoCjdEI+96DgwzZoe6c9juL49pXlzAFTI72aWC9Y8p7168g6XIKjh7k6pyQ==", - "license": "MIT", - "workspaces": [ - "docs", - "benchmarks" - ] - }, - "node_modules/escalade": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", - "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/escape-html": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", - "license": "MIT" - }, - "node_modules/escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/escodegen": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.1.0.tgz", - "integrity": "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==", - "license": "BSD-2-Clause", - "dependencies": { - "esprima": "^4.0.1", - "estraverse": "^5.2.0", - "esutils": "^2.0.2" - }, - "bin": { - "escodegen": "bin/escodegen.js", - "esgenerate": "bin/esgenerate.js" - }, - "engines": { - "node": ">=6.0" - }, - "optionalDependencies": { - "source-map": "~0.6.1" - } - }, - "node_modules/eslint": { - "version": "9.39.4", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.4.tgz", - "integrity": "sha512-XoMjdBOwe/esVgEvLmNsD3IRHkm7fbKIUGvrleloJXUZgDHig2IPWNniv+GwjyJXzuNqVjlr5+4yVUZjycJwfQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@eslint-community/eslint-utils": "^4.8.0", - "@eslint-community/regexpp": "^4.12.1", - "@eslint/config-array": "^0.21.2", - "@eslint/config-helpers": "^0.4.2", - "@eslint/core": "^0.17.0", - "@eslint/eslintrc": "^3.3.5", - "@eslint/js": "9.39.4", - "@eslint/plugin-kit": "^0.4.1", - "@humanfs/node": "^0.16.6", - "@humanwhocodes/module-importer": "^1.0.1", - "@humanwhocodes/retry": "^0.4.2", - "@types/estree": "^1.0.6", - "ajv": "^6.14.0", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.6", - "debug": "^4.3.2", - "escape-string-regexp": "^4.0.0", - "eslint-scope": "^8.4.0", - "eslint-visitor-keys": "^4.2.1", - "espree": "^10.4.0", - "esquery": "^1.5.0", - "esutils": "^2.0.2", - "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^8.0.0", - "find-up": "^5.0.0", - "glob-parent": "^6.0.2", - "ignore": "^5.2.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "json-stable-stringify-without-jsonify": "^1.0.1", - "lodash.merge": "^4.6.2", - "minimatch": "^3.1.5", - "natural-compare": "^1.4.0", - "optionator": "^0.9.3" - }, - "bin": { - "eslint": "bin/eslint.js" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://eslint.org/donate" - }, - "peerDependencies": { - "jiti": "*" - }, - "peerDependenciesMeta": { - "jiti": { - "optional": true - } - } - }, - "node_modules/eslint-config-next": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/eslint-config-next/-/eslint-config-next-16.2.6.tgz", - "integrity": "sha512-z2ELYSkyrrJ6cuunTU8vhsT/RpouPkjaSah06nVW6Rg2Hpg0Vs8s497/e5s8G8qtdp4ccsiovz5P1rv+5VSW2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@next/eslint-plugin-next": "16.2.6", - "eslint-import-resolver-node": "^0.3.6", - "eslint-import-resolver-typescript": "^3.5.2", - "eslint-plugin-import": "^2.32.0", - "eslint-plugin-jsx-a11y": "^6.10.0", - "eslint-plugin-react": "^7.37.0", - "eslint-plugin-react-hooks": "^7.0.0", - "globals": "16.4.0", - "typescript-eslint": "^8.46.0" - }, - "peerDependencies": { - "eslint": ">=9.0.0", - "typescript": ">=3.3.1" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/eslint-config-next/node_modules/globals": { - "version": "16.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-16.4.0.tgz", - "integrity": "sha512-ob/2LcVVaVGCYN+r14cnwnoDPUufjiYgSqRhiFD0Q1iI4Odora5RE8Iv1D24hAz5oMophRGkGz+yuvQmmUMnMw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint-import-resolver-node": { - "version": "0.3.10", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.10.tgz", - "integrity": "sha512-tRrKqFyCaKict5hOd244sL6EQFNycnMQnBe+j8uqGNXYzsImGbGUU4ibtoaBmv5FLwJwcFJNeg1GeVjQfbMrDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "debug": "^3.2.7", - "is-core-module": "^2.16.1", - "resolve": "^2.0.0-next.6" - } - }, - "node_modules/eslint-import-resolver-node/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-import-resolver-typescript": { - "version": "3.10.1", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-typescript/-/eslint-import-resolver-typescript-3.10.1.tgz", - "integrity": "sha512-A1rHYb06zjMGAxdLSkN2fXPBwuSaQ0iO5M/hdyS0Ajj1VBaRp0sPD3dn1FhME3c/JluGFbwSxyCfqdSbtQLAHQ==", - "dev": true, - "license": "ISC", - "dependencies": { - "@nolyfill/is-core-module": "1.0.39", - "debug": "^4.4.0", - "get-tsconfig": "^4.10.0", - "is-bun-module": "^2.0.0", - "stable-hash": "^0.0.5", - "tinyglobby": "^0.2.13", - "unrs-resolver": "^1.6.2" - }, - "engines": { - "node": "^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint-import-resolver-typescript" - }, - "peerDependencies": { - "eslint": "*", - "eslint-plugin-import": "*", - "eslint-plugin-import-x": "*" - }, - "peerDependenciesMeta": { - "eslint-plugin-import": { - "optional": true - }, - "eslint-plugin-import-x": { - "optional": true - } - } - }, - "node_modules/eslint-module-utils": { - "version": "2.12.1", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.12.1.tgz", - "integrity": "sha512-L8jSWTze7K2mTg0vos/RuLRS5soomksDPoJLXIslC7c8Wmut3bx7CPpJijDcBZtxQ5lrbUdM+s0OlNbz0DCDNw==", - "dev": true, - "license": "MIT", - "dependencies": { - "debug": "^3.2.7" - }, - "engines": { - "node": ">=4" - }, - "peerDependenciesMeta": { - "eslint": { - "optional": true - } - } - }, - "node_modules/eslint-module-utils/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import": { - "version": "2.32.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.32.0.tgz", - "integrity": "sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA==", - "dev": true, - "license": "MIT", - "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-import-resolver-node": "^0.3.9", - "eslint-module-utils": "^2.12.1", - "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" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9" - } - }, - "node_modules/eslint-plugin-import/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-jsx-a11y": { - "version": "6.10.2", - "resolved": "https://registry.npmjs.org/eslint-plugin-jsx-a11y/-/eslint-plugin-jsx-a11y-6.10.2.tgz", - "integrity": "sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "aria-query": "^5.3.2", - "array-includes": "^3.1.8", - "array.prototype.flatmap": "^1.3.2", - "ast-types-flow": "^0.0.8", - "axe-core": "^4.10.0", - "axobject-query": "^4.1.0", - "damerau-levenshtein": "^1.0.8", - "emoji-regex": "^9.2.2", - "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.0.3", - "string.prototype.includes": "^2.0.1" - }, - "engines": { - "node": ">=4.0" - }, - "peerDependencies": { - "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9" - } - }, - "node_modules/eslint-plugin-react": { - "version": "7.37.5", - "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.37.5.tgz", - "integrity": "sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-includes": "^3.1.8", - "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.1", - "estraverse": "^5.3.0", - "hasown": "^2.0.2", - "jsx-ast-utils": "^2.4.1 || ^3.0.0", - "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" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7" - } - }, - "node_modules/eslint-plugin-react-hooks": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.1.1.tgz", - "integrity": "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/core": "^7.24.4", - "@babel/parser": "^7.24.4", - "hermes-parser": "^0.25.1", - "zod": "^3.25.0 || ^4.0.0", - "zod-validation-error": "^3.5.0 || ^4.0.0" - }, - "engines": { - "node": ">=18" - }, - "peerDependencies": { - "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 || ^10.0.0" - } - }, - "node_modules/eslint-scope": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", - "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "esrecurse": "^4.3.0", - "estraverse": "^5.2.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/espree": { - "version": "10.4.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", - "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "acorn": "^8.15.0", - "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^4.2.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "license": "BSD-2-Clause", - "bin": { - "esparse": "bin/esparse.js", - "esvalidate": "bin/esvalidate.js" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/esquery": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", - "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "estraverse": "^5.1.0" - }, - "engines": { - "node": ">=0.10" - } - }, - "node_modules/esrecurse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", - "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "estraverse": "^5.2.0" - }, - "engines": { - "node": ">=4.0" - } - }, - "node_modules/estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "license": "BSD-2-Clause", - "engines": { - "node": ">=4.0" - } - }, - "node_modules/esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "license": "BSD-2-Clause", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/etag": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", - "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/eventemitter3": { - "version": "5.0.4", - "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-5.0.4.tgz", - "integrity": "sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw==", - "license": "MIT" - }, - "node_modules/events-universal": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/events-universal/-/events-universal-1.0.1.tgz", - "integrity": "sha512-LUd5euvbMLpwOF8m6ivPCbhQeSiYVNb8Vs0fQ8QjXo0JTkEHpz8pxdQf0gStltaPpw0Cca8b39KxvK9cfKRiAw==", - "license": "Apache-2.0", - "dependencies": { - "bare-events": "^2.7.0" - } - }, - "node_modules/eventsource": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-3.0.7.tgz", - "integrity": "sha512-CRT1WTyuQoD771GW56XEZFQ/ZoSfWid1alKGDYMmkt2yl8UXrVR4pspqWNEcqKvVIzg6PAltWjxcSSPrboA4iA==", - "license": "MIT", - "dependencies": { - "eventsource-parser": "^3.0.1" - }, - "engines": { - "node": ">=18.0.0" - } - }, - "node_modules/eventsource-parser": { - "version": "3.0.8", - "resolved": "https://registry.npmjs.org/eventsource-parser/-/eventsource-parser-3.0.8.tgz", - "integrity": "sha512-70QWGkr4snxr0OXLRWsFLeRBIRPuQOvt4s8QYjmUlmlkyTZkRqS7EDVRZtzU3TiyDbXSzaOeF0XUKy8PchzukQ==", - "license": "MIT", - "engines": { - "node": ">=18.0.0" - } - }, - "node_modules/execa": { - "version": "9.6.1", - "resolved": "https://registry.npmjs.org/execa/-/execa-9.6.1.tgz", - "integrity": "sha512-9Be3ZoN4LmYR90tUoVu2te2BsbzHfhJyfEiAVfz7N5/zv+jduIfLrV2xdQXOHbaD6KgpGdO9PRPM1Y4Q9QkPkA==", - "license": "MIT", - "dependencies": { - "@sindresorhus/merge-streams": "^4.0.0", - "cross-spawn": "^7.0.6", - "figures": "^6.1.0", - "get-stream": "^9.0.0", - "human-signals": "^8.0.1", - "is-plain-obj": "^4.1.0", - "is-stream": "^4.0.1", - "npm-run-path": "^6.0.0", - "pretty-ms": "^9.2.0", - "signal-exit": "^4.1.0", - "strip-final-newline": "^4.0.0", - "yoctocolors": "^2.1.1" - }, - "engines": { - "node": "^18.19.0 || >=20.5.0" - }, - "funding": { - "url": "https://github.com/sindresorhus/execa?sponsor=1" - } - }, - "node_modules/execa/node_modules/get-stream": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-9.0.1.tgz", - "integrity": "sha512-kVCxPF3vQM/N0B1PmoqVUqgHP+EeVjmZSQn+1oCRPxd2P21P2F19lIgbR3HBosbB1PUhOAoctJnfEn2GbN2eZA==", - "license": "MIT", - "dependencies": { - "@sec-ant/readable-stream": "^0.4.1", - "is-stream": "^4.0.1" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/expand-template": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", - "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==", - "license": "(MIT OR WTFPL)", - "engines": { - "node": ">=6" - } - }, - "node_modules/express": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/express/-/express-5.2.1.tgz", - "integrity": "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw==", - "license": "MIT", - "dependencies": { - "accepts": "^2.0.0", - "body-parser": "^2.2.1", - "content-disposition": "^1.0.0", - "content-type": "^1.0.5", - "cookie": "^0.7.1", - "cookie-signature": "^1.2.1", - "debug": "^4.4.0", - "depd": "^2.0.0", - "encodeurl": "^2.0.0", - "escape-html": "^1.0.3", - "etag": "^1.8.1", - "finalhandler": "^2.1.0", - "fresh": "^2.0.0", - "http-errors": "^2.0.0", - "merge-descriptors": "^2.0.0", - "mime-types": "^3.0.0", - "on-finished": "^2.4.1", - "once": "^1.4.0", - "parseurl": "^1.3.3", - "proxy-addr": "^2.0.7", - "qs": "^6.14.0", - "range-parser": "^1.2.1", - "router": "^2.2.0", - "send": "^1.1.0", - "serve-static": "^2.2.0", - "statuses": "^2.0.1", - "type-is": "^2.0.1", - "vary": "^1.1.2" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/express-rate-limit": { - "version": "8.5.1", - "resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-8.5.1.tgz", - "integrity": "sha512-5O6KYmyJEpuPJV5hNTXKbAHWRqrzyu+OI3vUnSd2kXFubIVpG7ezpgxQy76Zo5GQZtrQBg86hF+CM/NX+cioiQ==", - "license": "MIT", - "dependencies": { - "ip-address": "^10.2.0" - }, - "engines": { - "node": ">= 16" - }, - "funding": { - "url": "https://github.com/sponsors/express-rate-limit" - }, - "peerDependencies": { - "express": ">= 4.11" - } - }, - "node_modules/extract-zip": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-2.0.1.tgz", - "integrity": "sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg==", - "license": "BSD-2-Clause", - "dependencies": { - "debug": "^4.1.1", - "get-stream": "^5.1.0", - "yauzl": "^2.10.0" - }, - "bin": { - "extract-zip": "cli.js" - }, - "engines": { - "node": ">= 10.17.0" - }, - "optionalDependencies": { - "@types/yauzl": "^2.9.1" - } - }, - "node_modules/fast-deep-equal": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "license": "MIT" - }, - "node_modules/fast-fifo": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/fast-fifo/-/fast-fifo-1.3.2.tgz", - "integrity": "sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==", - "license": "MIT" - }, - "node_modules/fast-glob": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.1.tgz", - "integrity": "sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.4" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/fast-glob/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true, - "license": "MIT" - }, - "node_modules/fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", - "dev": true, - "license": "MIT" - }, - "node_modules/fast-sha256": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/fast-sha256/-/fast-sha256-1.3.0.tgz", - "integrity": "sha512-n11RGP/lrWEFI/bWdygLxhI+pVeo1ZYIVwvvPkW7azl/rOy+F3HYRZ2K5zeE9mmkhQppyv9sQFx0JM9UabnpPQ==", - "license": "Unlicense" - }, - "node_modules/fast-string-truncated-width": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/fast-string-truncated-width/-/fast-string-truncated-width-3.0.3.tgz", - "integrity": "sha512-0jjjIEL6+0jag3l2XWWizO64/aZVtpiGE3t0Zgqxv0DPuxiMjvB3M24fCyhZUO4KomJQPj3LTSUnDP3GpdwC0g==", - "license": "MIT" - }, - "node_modules/fast-string-width": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/fast-string-width/-/fast-string-width-3.0.2.tgz", - "integrity": "sha512-gX8LrtNEI5hq8DVUfRQMbr5lpaS4nMIWV+7XEbXk2b8kiQIizgnlr12B4dA3ZEx3308ze0O4Q1R+cHts8kyUJg==", - "license": "MIT", - "dependencies": { - "fast-string-truncated-width": "^3.0.2" - } - }, - "node_modules/fast-uri": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.1.2.tgz", - "integrity": "sha512-rVjf7ArG3LTk+FS6Yw81V1DLuZl1bRbNrev6Tmd/9RaroeeRRJhAt7jg/6YFxbvAQXUCavSoZhPPj6oOx+5KjQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/fastify" - }, - { - "type": "opencollective", - "url": "https://opencollective.com/fastify" - } - ], - "license": "BSD-3-Clause" - }, - "node_modules/fast-wrap-ansi": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/fast-wrap-ansi/-/fast-wrap-ansi-0.2.0.tgz", - "integrity": "sha512-rLV8JHxTyhVmFYhBJuMujcrHqOT2cnO5Zxj37qROj23CP39GXubJRBUFF0z8KFK77Uc0SukZUf7JZhsVEQ6n8w==", - "license": "MIT", - "dependencies": { - "fast-string-width": "^3.0.2" - } - }, - "node_modules/fastq": { - "version": "1.20.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.20.1.tgz", - "integrity": "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw==", - "license": "ISC", - "dependencies": { - "reusify": "^1.0.4" - } - }, - "node_modules/fd-slicer": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.1.0.tgz", - "integrity": "sha512-cE1qsB/VwyQozZ+q1dGxR8LBYNZeofhEdUNGSMbQD3Gw2lAzX9Zb3uIU6Ebc/Fmyjo9AWWfnn0AUCHqtevs/8g==", - "license": "MIT", - "dependencies": { - "pend": "~1.2.0" - } - }, - "node_modules/fetch-blob": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz", - "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/jimmywarting" - }, - { - "type": "paypal", - "url": "https://paypal.me/jimmywarting" - } - ], - "license": "MIT", - "dependencies": { - "node-domexception": "^1.0.0", - "web-streams-polyfill": "^3.0.3" - }, - "engines": { - "node": "^12.20 || >= 14.13" - } - }, - "node_modules/figures": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/figures/-/figures-6.1.0.tgz", - "integrity": "sha512-d+l3qxjSesT4V7v2fh+QnmFnUWv9lSpjarhShNTgBOfA0ttejbQUAlHLitbjkoRiDulW0OPoQPYIGhIC8ohejg==", - "license": "MIT", - "dependencies": { - "is-unicode-supported": "^2.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/file-entry-cache": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", - "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "flat-cache": "^4.0.0" - }, - "engines": { - "node": ">=16.0.0" - } - }, - "node_modules/file-uri-to-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", - "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", - "license": "MIT" - }, - "node_modules/fill-range": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", - "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "license": "MIT", - "dependencies": { - "to-regex-range": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/finalhandler": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.1.tgz", - "integrity": "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA==", - "license": "MIT", - "dependencies": { - "debug": "^4.4.0", - "encodeurl": "^2.0.0", - "escape-html": "^1.0.3", - "on-finished": "^2.4.1", - "parseurl": "^1.3.3", - "statuses": "^2.0.1" - }, - "engines": { - "node": ">= 18.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/find-up": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", - "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", - "dev": true, - "license": "MIT", - "dependencies": { - "locate-path": "^6.0.0", - "path-exists": "^4.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/flat-cache": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", - "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", - "dev": true, - "license": "MIT", - "dependencies": { - "flatted": "^3.2.9", - "keyv": "^4.5.4" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/flatted": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz", - "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==", - "dev": true, - "license": "ISC" - }, - "node_modules/for-each": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", - "integrity": "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/formdata-polyfill": { - "version": "4.0.10", - "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz", - "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==", - "license": "MIT", - "dependencies": { - "fetch-blob": "^3.1.2" - }, - "engines": { - "node": ">=12.20.0" - } - }, - "node_modules/forwarded": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", - "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/framer-motion": { - "version": "12.38.0", - "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-12.38.0.tgz", - "integrity": "sha512-rFYkY/pigbcswl1XQSb7q424kSTQ8q6eAC+YUsSKooHQYuLdzdHjrt6uxUC+PRAO++q5IS7+TamgIw1AphxR+g==", - "license": "MIT", - "dependencies": { - "motion-dom": "^12.38.0", - "motion-utils": "^12.36.0", - "tslib": "^2.4.0" - }, - "peerDependencies": { - "@emotion/is-prop-valid": "*", - "react": "^18.0.0 || ^19.0.0", - "react-dom": "^18.0.0 || ^19.0.0" - }, - "peerDependenciesMeta": { - "@emotion/is-prop-valid": { - "optional": true - }, - "react": { - "optional": true - }, - "react-dom": { - "optional": true - } - } - }, - "node_modules/fresh": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", - "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/fs-constants": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", - "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", - "license": "MIT" - }, - "node_modules/fs-extra": { - "version": "11.3.5", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.3.5.tgz", - "integrity": "sha512-eKpRKAovdpZtR1WopLHxlBWvAgPny3c4gX1G5Jhwmmw4XJj0ifSD5qB5TOo8hmA0wlRKDAOAhEE1yVPgs6Fgcg==", - "license": "MIT", - "dependencies": { - "graceful-fs": "^4.2.0", - "jsonfile": "^6.0.1", - "universalify": "^2.0.0" - }, - "engines": { - "node": ">=14.14" - } - }, - "node_modules/function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/function.prototype.name": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", - "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "functions-have-names": "^1.2.3", - "hasown": "^2.0.2", - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/fuzzysort": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/fuzzysort/-/fuzzysort-3.1.0.tgz", - "integrity": "sha512-sR9BNCjBg6LNgwvxlBd0sBABvQitkLzoVY9MYYROQVX/FvfJ4Mai9LsGhDgd8qYdds0bY77VzYd5iuB+v5rwQQ==", - "license": "MIT" - }, - "node_modules/generator-function": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/generator-function/-/generator-function-2.0.1.tgz", - "integrity": "sha512-SFdFmIJi+ybC0vjlHN0ZGVGHc3lgE0DxPAT0djjVg+kjOnSqclqmj0KQ7ykTOLP6YxoqOvuAODGdcHJn+43q3g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/gensync": { - "version": "1.0.0-beta.2", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", - "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "license": "ISC", - "engines": { - "node": "6.* || 8.* || >= 10.*" - } - }, - "node_modules/get-east-asian-width": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/get-east-asian-width/-/get-east-asian-width-1.6.0.tgz", - "integrity": "sha512-QRbvDIbx6YklUe6RxeTeleMR0yv3cYH6PsPZHcnVn7xv7zO1BHN8r0XETu8n6Ye3Q+ahtSarc3WgtNWmehIBfA==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/get-intrinsic": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", - "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "function-bind": "^1.1.2", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "math-intrinsics": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-own-enumerable-keys": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-own-enumerable-keys/-/get-own-enumerable-keys-1.0.0.tgz", - "integrity": "sha512-PKsK2FSrQCyxcGHsGrLDcK0lx+0Ke+6e8KFFozA9/fIQLhQzPaRvJFdcz7+Axg3jUH/Mq+NI4xa5u/UT2tQskA==", - "license": "MIT", - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/get-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", - "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/get-stream": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", - "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", - "license": "MIT", - "dependencies": { - "pump": "^3.0.0" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/get-symbol-description": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", - "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-tsconfig": { - "version": "4.14.0", - "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.14.0.tgz", - "integrity": "sha512-yTb+8DXzDREzgvYmh6s9vHsSVCHeC0G3PI5bEXNBHtmshPnO+S5O7qgLEOn0I5QvMy6kpZN8K1NKGyilLb93wA==", - "dev": true, - "license": "MIT", - "dependencies": { - "resolve-pkg-maps": "^1.0.0" - }, - "funding": { - "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" - } - }, - "node_modules/get-uri": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/get-uri/-/get-uri-6.0.5.tgz", - "integrity": "sha512-b1O07XYq8eRuVzBNgJLstU6FYc1tS6wnMtF1I1D9lE8LxZSOGZ7LhxN54yPP6mGw5f2CkXY2BQUL9Fx41qvcIg==", - "license": "MIT", - "dependencies": { - "basic-ftp": "^5.0.2", - "data-uri-to-buffer": "^6.0.2", - "debug": "^4.3.4" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/github-from-package": { - "version": "0.0.0", - "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", - "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==", - "license": "MIT" - }, - "node_modules/glob-parent": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", - "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.3" - }, - "engines": { - "node": ">=10.13.0" - } - }, - "node_modules/globals": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", - "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/globalthis": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", - "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-properties": "^1.2.1", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/graceful-fs": { - "version": "4.2.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", - "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", - "license": "ISC" - }, - "node_modules/graphql": { - "version": "16.14.0", - "resolved": "https://registry.npmjs.org/graphql/-/graphql-16.14.0.tgz", - "integrity": "sha512-BBvQ/406p+4CZbTpCbVPSxfzrZrbnuWSP1ELYgyS6B+hNeKzgrdB4JczCa5VZUBQrDa9hUngm0KnexY6pJRN5Q==", - "license": "MIT", - "engines": { - "node": "^12.22.0 || ^14.16.0 || ^16.0.0 || >=17.0.0" - } - }, - "node_modules/has-bigints": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", - "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/has-property-descriptors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", - "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-proto": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", - "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-symbols": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", - "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-tostringtag": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", - "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-symbols": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/hasown": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.3.tgz", - "integrity": "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg==", - "license": "MIT", - "dependencies": { - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/headers-polyfill": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/headers-polyfill/-/headers-polyfill-5.0.1.tgz", - "integrity": "sha512-1TJ6Fih/b8h5TIcv+1+Hw0PDQWJTKDKzFZzcKOiW1wJza3XoAQlkCuXLbymPYB8+ZQyw8mHvdw560e8zVFIWyA==", - "license": "MIT", - "dependencies": { - "@types/set-cookie-parser": "^2.4.10", - "set-cookie-parser": "^3.0.1" - } - }, - "node_modules/hermes-estree": { - "version": "0.25.1", - "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz", - "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==", - "dev": true, - "license": "MIT" - }, - "node_modules/hermes-parser": { - "version": "0.25.1", - "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz", - "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==", - "dev": true, - "license": "MIT", - "dependencies": { - "hermes-estree": "0.25.1" - } - }, - "node_modules/hono": { - "version": "4.12.18", - "resolved": "https://registry.npmjs.org/hono/-/hono-4.12.18.tgz", - "integrity": "sha512-RWzP96k/yv0PQfyXnWjs6zot20TqfpfsNXhOnev8d1InAxubW93L11/oNUc3tQqn2G0bSdAOBpX+2uDFHV7kdQ==", - "license": "MIT", - "engines": { - "node": ">=16.9.0" - } - }, - "node_modules/http-errors": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", - "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", - "license": "MIT", - "dependencies": { - "depd": "~2.0.0", - "inherits": "~2.0.4", - "setprototypeof": "~1.2.0", - "statuses": "~2.0.2", - "toidentifier": "~1.0.1" - }, - "engines": { - "node": ">= 0.8" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/http-proxy-agent": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", - "integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==", - "license": "MIT", - "dependencies": { - "agent-base": "^7.1.0", - "debug": "^4.3.4" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/https-proxy-agent": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz", - "integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==", - "license": "MIT", - "dependencies": { - "agent-base": "^7.1.2", - "debug": "4" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/human-signals": { - "version": "8.0.1", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-8.0.1.tgz", - "integrity": "sha512-eKCa6bwnJhvxj14kZk5NCPc6Hb6BdsU9DZcOnmQKSnO1VKrfV0zCvtttPZUsBvjmNDn8rpcJfpwSYnHBjc95MQ==", - "license": "Apache-2.0", - "engines": { - "node": ">=18.18.0" - } - }, - "node_modules/iconv-lite": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz", - "integrity": "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw==", - "license": "MIT", - "dependencies": { - "safer-buffer": ">= 2.1.2 < 3.0.0" - }, - "engines": { - "node": ">=0.10.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/icu-minify": { - "version": "4.12.0", - "resolved": "https://registry.npmjs.org/icu-minify/-/icu-minify-4.12.0.tgz", - "integrity": "sha512-zDmM05uav3t3+kxSfRrNlmyXOdj2b+uHA+p04CG32eJabtaHbugXujuL+YfRkwP9joAnf0Uh+RMGCKD5NLa5rQ==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/amannn" - } - ], - "license": "MIT", - "dependencies": { - "@formatjs/icu-messageformat-parser": "^3.4.0" - } - }, - "node_modules/ieee754": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "BSD-3-Clause" - }, - "node_modules/ignore": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", - "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, - "node_modules/immer": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/immer/-/immer-10.2.0.tgz", - "integrity": "sha512-d/+XTN3zfODyjr89gM3mPq1WNX2B8pYsu7eORitdwyA2sBubnTl3laYlBk4sXY5FUa5qTZGBDPJICVbvqzjlbw==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/immer" - } - }, - "node_modules/import-fresh": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", - "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", - "license": "MIT", - "dependencies": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.8.19" - } - }, - "node_modules/inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "license": "ISC" - }, - "node_modules/ini": { - "version": "1.3.8", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", - "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", - "license": "ISC" - }, - "node_modules/internal-slot": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", - "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "hasown": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/internmap": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz", - "integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/intl-messageformat": { - "version": "11.2.6", - "resolved": "https://registry.npmjs.org/intl-messageformat/-/intl-messageformat-11.2.6.tgz", - "integrity": "sha512-afAN2yNN7zjB77G1ZC5L8GtLrEshyBvOQXz88flxCO/ocTIQist98gu0r/O6H/SSiQhQsOOtWPxmCEvtDABXXQ==", - "license": "BSD-3-Clause", - "dependencies": { - "@formatjs/fast-memoize": "3.1.5", - "@formatjs/icu-messageformat-parser": "3.5.9" - } - }, - "node_modules/ip-address": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/ip-address/-/ip-address-10.2.0.tgz", - "integrity": "sha512-/+S6j4E9AHvW9SWMSEY9Xfy66O5PWvVEJ08O0y5JGyEKQpojb0K0GKpz/v5HJ/G0vi3D2sjGK78119oXZeE0qA==", - "license": "MIT", - "engines": { - "node": ">= 12" - } - }, - "node_modules/ipaddr.js": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", - "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", - "license": "MIT", - "engines": { - "node": ">= 0.10" - } - }, - "node_modules/is-array-buffer": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", - "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", - "license": "MIT" - }, - "node_modules/is-async-function": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.1.1.tgz", - "integrity": "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "async-function": "^1.0.0", - "call-bound": "^1.0.3", - "get-proto": "^1.0.1", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-bigint": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", - "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-bigints": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-boolean-object": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.2.tgz", - "integrity": "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-bun-module": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-bun-module/-/is-bun-module-2.0.0.tgz", - "integrity": "sha512-gNCGbnnnnFAUGKeZ9PdbyeGYJqewpmc2aKHUEMO5nQPWU9lOmv7jcmQIv+qHD8fXW6W7qfuCwX4rY9LNRjXrkQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "semver": "^7.7.1" - } - }, - "node_modules/is-bun-module/node_modules/semver": { - "version": "7.8.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", - "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-core-module": { - "version": "2.16.2", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.2.tgz", - "integrity": "sha512-evOr8xfXKxE6qSR0hSXL2r3sd7ALj8+7jQEUvPYcm5sgZFdJ+AYzT6yNmJenvIYQBgIGwfwz08sL8zoL7yq2BA==", - "dev": true, - "license": "MIT", - "dependencies": { - "hasown": "^2.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-data-view": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", - "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "is-typed-array": "^1.1.13" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-date-object": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", - "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-docker": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-3.0.0.tgz", - "integrity": "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ==", - "license": "MIT", - "bin": { - "is-docker": "cli.js" - }, - "engines": { - "node": "^12.20.0 || ^14.13.1 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-finalizationregistry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", - "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/is-generator-function": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.1.2.tgz", - "integrity": "sha512-upqt1SkGkODW9tsGNG5mtXTXtECizwtS2kA161M+gJPc1xdb/Ax629af6YrTwcOeQHbewrPNlE5Dx7kzvXTizA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.4", - "generator-function": "^2.0.0", - "get-proto": "^1.0.1", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-glob": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", - "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", - "license": "MIT", - "dependencies": { - "is-extglob": "^2.1.1" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-in-ssh": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-in-ssh/-/is-in-ssh-1.0.0.tgz", - "integrity": "sha512-jYa6Q9rH90kR1vKB6NM7qqd1mge3Fx4Dhw5TVlK1MUBqhEOuCagrEHMevNuCcbECmXZ0ThXkRm+Ymr51HwEPAw==", - "license": "MIT", - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-inside-container": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-inside-container/-/is-inside-container-1.0.0.tgz", - "integrity": "sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA==", - "license": "MIT", - "dependencies": { - "is-docker": "^3.0.0" - }, - "bin": { - "is-inside-container": "cli.js" - }, - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-interactive": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-2.0.0.tgz", - "integrity": "sha512-qP1vozQRI+BMOPcjFzrjXuQvdak2pHNUMZoeG2eRbiSqyvbEf/wQtEOTOX1guk6E3t36RkaqiSt8A/6YElNxLQ==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", - "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-negative-zero": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", - "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-node-process": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/is-node-process/-/is-node-process-1.2.0.tgz", - "integrity": "sha512-Vg4o6/fqPxIjtxgUH5QLJhwZ7gW5diGCVlXpuUfELC62CuxM1iHcRe51f2W1FDy04Ai4KJkagKjx3XaqyfRKXw==", - "license": "MIT" - }, - "node_modules/is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "license": "MIT", - "engines": { - "node": ">=0.12.0" - } - }, - "node_modules/is-number-object": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", - "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-obj": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-3.0.0.tgz", - "integrity": "sha512-IlsXEHOjtKhpN8r/tRFj2nDyTmHvcfNeu/nrRIcXE17ROeatXchkojffa1SpdqW4cr/Fj6QkEf/Gn4zf6KKvEQ==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-plain-obj": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz", - "integrity": "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-promise": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", - "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", - "license": "MIT" - }, - "node_modules/is-regex": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", - "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-regexp": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/is-regexp/-/is-regexp-3.1.0.tgz", - "integrity": "sha512-rbku49cWloU5bSMI+zaRaXdQHXnthP6DZ/vLnfdSKyL4zUzuWnomtOEiZZOd+ioQ+avFo/qau3KPTc7Fjy1uPA==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-set": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", - "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-shared-array-buffer": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", - "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-stream": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-4.0.1.tgz", - "integrity": "sha512-Dnz92NInDqYckGEUJv689RbRiTSEHCQ7wOVeALbkOz999YpqT46yMRIGtSNl2iCL1waAZSx40+h59NV/EwzV/A==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-string": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", - "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-symbol": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", - "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-symbols": "^1.1.0", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-typed-array": { - "version": "1.1.15", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", - "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-unicode-supported": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-2.1.0.tgz", - "integrity": "sha512-mE00Gnza5EEB3Ds0HfMyllZzbBrmLOX3vfWoj9A9PEnTfratQ/BcaJOuMhnkhjXvb2+FkY3VuHqtAGpTPmglFQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-weakmap": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", - "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakref": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.1.tgz", - "integrity": "sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakset": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", - "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-wsl": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-3.1.1.tgz", - "integrity": "sha512-e6rvdUCiQCAuumZslxRJWR/Doq4VpPR82kqclvcS0efgt430SlGIk05vdCN58+VrzgtIcfNODjozVielycD4Sw==", - "license": "MIT", - "dependencies": { - "is-inside-container": "^1.0.0" - }, - "engines": { - "node": ">=16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true, - "license": "MIT" - }, - "node_modules/isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "license": "ISC" - }, - "node_modules/israeli-bank-scrapers": { - "version": "6.7.4", - "resolved": "https://registry.npmjs.org/israeli-bank-scrapers/-/israeli-bank-scrapers-6.7.4.tgz", - "integrity": "sha512-M6aEws/6Pxks8/INtDYCbLPDWeoZLStNF9ulTalUg89pAyFemNznl0SkcniIDqIQIZN/5vhtr79HojgvRpxN5w==", - "license": "MIT", - "dependencies": { - "debug": "^4.3.2", - "moment": "^2.22.2", - "moment-timezone": "^0.5.37", - "puppeteer": "^24.40.0" - }, - "engines": { - "node": ">= 22.13.0" - } - }, - "node_modules/iterator.prototype": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.5.tgz", - "integrity": "sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.6", - "get-proto": "^1.0.0", - "has-symbols": "^1.1.0", - "set-function-name": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/jiti": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.7.0.tgz", - "integrity": "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ==", - "dev": true, - "license": "MIT", - "bin": { - "jiti": "lib/jiti-cli.mjs" - } - }, - "node_modules/jose": { - "version": "6.2.3", - "resolved": "https://registry.npmjs.org/jose/-/jose-6.2.3.tgz", - "integrity": "sha512-YYVDInQKFJfR/xa3ojUTl8c2KoTwiL1R5Wg9YCydwH0x0B9grbzlg5HC7mMjCtUJjbQ/YnGEZIhI5tCgfTb4Hw==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/panva" - } - }, - "node_modules/js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "license": "MIT" - }, - "node_modules/js-yaml": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", - "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", - "license": "MIT", - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, - "node_modules/jsesc": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", - "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", - "license": "MIT", - "bin": { - "jsesc": "bin/jsesc" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/json-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", - "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/json-parse-even-better-errors": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", - "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", - "license": "MIT" - }, - "node_modules/json-schema-to-ts": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/json-schema-to-ts/-/json-schema-to-ts-3.1.1.tgz", - "integrity": "sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g==", - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.18.3", - "ts-algebra": "^2.0.0" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true, - "license": "MIT" - }, - "node_modules/json-schema-typed": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/json-schema-typed/-/json-schema-typed-8.0.2.tgz", - "integrity": "sha512-fQhoXdcvc3V28x7C7BMs4P5+kNlgUURe2jmUT1T//oBRMDrqy1QPelJimwZGo7Hg9VPV3EQV5Bnq4hbFy2vetA==", - "license": "BSD-2-Clause" - }, - "node_modules/json-stable-stringify-without-jsonify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", - "dev": true, - "license": "MIT" - }, - "node_modules/json5": { - "version": "2.2.3", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", - "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", - "license": "MIT", - "bin": { - "json5": "lib/cli.js" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/jsonfile": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.2.1.tgz", - "integrity": "sha512-zwOTdL3rFQ/lRdBnntKVOX6k5cKJwEc1HdilT71BWEu7J41gXIB2MRp+vxduPSwZJPWBxEzv4yH1wYLJGUHX4Q==", - "license": "MIT", - "dependencies": { - "universalify": "^2.0.0" - }, - "optionalDependencies": { - "graceful-fs": "^4.1.6" - } - }, - "node_modules/jsx-ast-utils": { - "version": "3.3.5", - "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz", - "integrity": "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-includes": "^3.1.6", - "array.prototype.flat": "^1.3.1", - "object.assign": "^4.1.4", - "object.values": "^1.1.6" - }, - "engines": { - "node": ">=4.0" - } - }, - "node_modules/keyv": { - "version": "4.5.4", - "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", - "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", - "dev": true, - "license": "MIT", - "dependencies": { - "json-buffer": "3.0.1" - } - }, - "node_modules/kleur": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", - "integrity": "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/language-subtag-registry": { - "version": "0.3.23", - "resolved": "https://registry.npmjs.org/language-subtag-registry/-/language-subtag-registry-0.3.23.tgz", - "integrity": "sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ==", - "dev": true, - "license": "CC0-1.0" - }, - "node_modules/language-tags": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/language-tags/-/language-tags-1.0.9.tgz", - "integrity": "sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA==", - "dev": true, - "license": "MIT", - "dependencies": { - "language-subtag-registry": "^0.3.20" - }, - "engines": { - "node": ">=0.10" - } - }, - "node_modules/levn": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", - "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "prelude-ls": "^1.2.1", - "type-check": "~0.4.0" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/lightningcss": { - "version": "1.32.0", - "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", - "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", - "dev": true, - "license": "MPL-2.0", - "dependencies": { - "detect-libc": "^2.0.3" - }, - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - }, - "optionalDependencies": { - "lightningcss-android-arm64": "1.32.0", - "lightningcss-darwin-arm64": "1.32.0", - "lightningcss-darwin-x64": "1.32.0", - "lightningcss-freebsd-x64": "1.32.0", - "lightningcss-linux-arm-gnueabihf": "1.32.0", - "lightningcss-linux-arm64-gnu": "1.32.0", - "lightningcss-linux-arm64-musl": "1.32.0", - "lightningcss-linux-x64-gnu": "1.32.0", - "lightningcss-linux-x64-musl": "1.32.0", - "lightningcss-win32-arm64-msvc": "1.32.0", - "lightningcss-win32-x64-msvc": "1.32.0" - } - }, - "node_modules/lightningcss-android-arm64": { - "version": "1.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", - "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", - "cpu": [ - "arm64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", - "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", - "cpu": [ - "arm64" - ], - "dev": 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-darwin-x64": { - "version": "1.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", - "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", - "cpu": [ - "x64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", - "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", - "cpu": [ - "x64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", - "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", - "cpu": [ - "arm" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", - "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", - "cpu": [ - "arm64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", - "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", - "cpu": [ - "arm64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", - "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", - "cpu": [ - "x64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", - "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", - "cpu": [ - "x64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", - "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", - "cpu": [ - "arm64" - ], - "dev": 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.32.0", - "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", - "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "MPL-2.0", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">= 12.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/parcel" - } - }, - "node_modules/lines-and-columns": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", - "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", - "license": "MIT" - }, - "node_modules/locate-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", - "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-locate": "^5.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/lodash.merge": { - "version": "4.6.2", - "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", - "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/log-symbols": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-6.0.0.tgz", - "integrity": "sha512-i24m8rpwhmPIS4zscNzK6MSEhk0DUWa/8iYQWxhffV8jkI4Phvs3F+quL5xvS0gdQR0FyTCMMH33Y78dDTzzIw==", - "license": "MIT", - "dependencies": { - "chalk": "^5.3.0", - "is-unicode-supported": "^1.3.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/log-symbols/node_modules/chalk": { - "version": "5.6.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.6.2.tgz", - "integrity": "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==", - "license": "MIT", - "engines": { - "node": "^12.17.0 || ^14.13 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" - } - }, - "node_modules/log-symbols/node_modules/is-unicode-supported": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-1.3.0.tgz", - "integrity": "sha512-43r2mRvz+8JRIKnWJ+3j8JtjRKZ6GmjzfaE/qiBJnikNnYv/6bagRJ1kUhNk8R5EX/GkobD+r+sfxCPJsiKBLQ==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/loose-envify": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", - "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "js-tokens": "^3.0.0 || ^4.0.0" - }, - "bin": { - "loose-envify": "cli.js" - } - }, - "node_modules/lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "license": "ISC", - "dependencies": { - "yallist": "^3.0.2" - } - }, - "node_modules/lucide-react": { - "version": "1.14.0", - "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-1.14.0.tgz", - "integrity": "sha512-+1mdWcfSJVUsaTIjN9zoezmUhfXo5l0vP7ekBMPo3jcS/aIkxHnXqAPsByszMZx/Y8oQBRJxJx5xg+RH3urzxA==", - "license": "ISC", - "peerDependencies": { - "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/magic-string": { - "version": "0.30.21", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", - "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/sourcemap-codec": "^1.5.5" - } - }, - "node_modules/math-intrinsics": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", - "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/media-typer": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", - "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/merge-descriptors": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz", - "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/merge-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", - "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", - "license": "MIT" - }, - "node_modules/merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/micromatch": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", - "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "license": "MIT", - "dependencies": { - "braces": "^3.0.3", - "picomatch": "^2.3.1" - }, - "engines": { - "node": ">=8.6" - } - }, - "node_modules/mime-db": { - "version": "1.54.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz", - "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/mime-types": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz", - "integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==", - "license": "MIT", - "dependencies": { - "mime-db": "^1.54.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/mimic-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", - "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/mimic-function": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/mimic-function/-/mimic-function-5.0.1.tgz", - "integrity": "sha512-VP79XUPxV2CigYP3jWwAUFSku2aKqBH7uTAapFWCBqutsbmDo96KY5o8uh6U+/YSIn5OxJnXp73beVkpqMIGhA==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/mimic-response": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", - "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==", - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/minimatch": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", - "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, - "node_modules/minimist": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", - "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/mitt": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/mitt/-/mitt-3.0.1.tgz", - "integrity": "sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw==", - "license": "MIT" - }, - "node_modules/mkdirp-classic": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", - "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", - "license": "MIT" - }, - "node_modules/moment": { - "version": "2.30.1", - "resolved": "https://registry.npmjs.org/moment/-/moment-2.30.1.tgz", - "integrity": "sha512-uEmtNhbDOrWPFS+hdjFCBfy9f2YoyzRpwcl+DqpC6taX21FzsTLQVbMV/W7PzNSX6x/bhC1zA3c2UQ5NzH6how==", - "license": "MIT", - "engines": { - "node": "*" - } - }, - "node_modules/moment-timezone": { - "version": "0.5.48", - "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.48.tgz", - "integrity": "sha512-f22b8LV1gbTO2ms2j2z13MuPogNoh5UzxL3nzNAYKGraILnbGc9NEE6dyiiiLv46DGRb8A4kg8UKWLjPthxBHw==", - "license": "MIT", - "dependencies": { - "moment": "^2.29.4" - }, - "engines": { - "node": "*" - } - }, - "node_modules/motion-dom": { - "version": "12.38.0", - "resolved": "https://registry.npmjs.org/motion-dom/-/motion-dom-12.38.0.tgz", - "integrity": "sha512-pdkHLD8QYRp8VfiNLb8xIBJis1byQ9gPT3Jnh2jqfFtAsWUA3dEepDlsWe/xMpO8McV+VdpKVcp+E+TGJEtOoA==", - "license": "MIT", - "dependencies": { - "motion-utils": "^12.36.0" - } - }, - "node_modules/motion-utils": { - "version": "12.36.0", - "resolved": "https://registry.npmjs.org/motion-utils/-/motion-utils-12.36.0.tgz", - "integrity": "sha512-eHWisygbiwVvf6PZ1vhaHCLamvkSbPIeAYxWUuL3a2PD/TROgE7FvfHWTIH4vMl798QLfMw15nRqIaRDXTlYRg==", - "license": "MIT" - }, - "node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "license": "MIT" - }, - "node_modules/msw": { - "version": "2.14.6", - "resolved": "https://registry.npmjs.org/msw/-/msw-2.14.6.tgz", - "integrity": "sha512-ALe+N10S72cyx94cMcy3Zs4HhXCj35sgeAL4c+WTvKi0zWnbd8/h0lcFqv0mb2P+aSgAdD7p9HzvA0DiUPxsyg==", - "hasInstallScript": true, - "license": "MIT", - "dependencies": { - "@inquirer/confirm": "^6.0.11", - "@mswjs/interceptors": "^0.41.3", - "@open-draft/deferred-promise": "^3.0.0", - "@types/statuses": "^2.0.6", - "cookie": "^1.1.1", - "graphql": "^16.13.2", - "headers-polyfill": "^5.0.1", - "is-node-process": "^1.2.0", - "outvariant": "^1.4.3", - "path-to-regexp": "^6.3.0", - "picocolors": "^1.1.1", - "rettime": "^0.11.11", - "statuses": "^2.0.2", - "strict-event-emitter": "^0.5.1", - "tough-cookie": "^6.0.1", - "type-fest": "^5.5.0", - "until-async": "^3.0.2", - "yargs": "^17.7.2" - }, - "bin": { - "msw": "cli/index.js" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/mswjs" - }, - "peerDependencies": { - "typescript": ">= 4.8.x" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/msw/node_modules/cookie": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.1.1.tgz", - "integrity": "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/mute-stream": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-3.0.0.tgz", - "integrity": "sha512-dkEJPVvun4FryqBmZ5KhDo0K9iDXAwn08tMLDinNdRBNPcYEDiWYysLcc6k3mjTMlbP9KyylvRpd4wFtwrT9rw==", - "license": "ISC", - "engines": { - "node": "^20.17.0 || >=22.9.0" - } - }, - "node_modules/nanoid": { - "version": "3.3.12", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz", - "integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "bin": { - "nanoid": "bin/nanoid.cjs" - }, - "engines": { - "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" - } - }, - "node_modules/napi-build-utils": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-2.0.0.tgz", - "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", - "license": "MIT" - }, - "node_modules/napi-postinstall": { - "version": "0.3.4", - "resolved": "https://registry.npmjs.org/napi-postinstall/-/napi-postinstall-0.3.4.tgz", - "integrity": "sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ==", - "dev": true, - "license": "MIT", - "bin": { - "napi-postinstall": "lib/cli.js" - }, - "engines": { - "node": "^12.20.0 || ^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/napi-postinstall" - } - }, - "node_modules/natural-compare": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", - "dev": true, - "license": "MIT" - }, - "node_modules/negotiator": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz", - "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/netmask": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/netmask/-/netmask-2.1.1.tgz", - "integrity": "sha512-eonl3sLUha+S1GzTPxychyhnUzKyeQkZ7jLjKrBagJgPla13F+uQ71HgpFefyHgqrjEbCPkDArxYsjY8/+gLKA==", - "license": "MIT", - "engines": { - "node": ">= 0.4.0" - } - }, - "node_modules/next": { - "version": "16.2.6", - "resolved": "https://registry.npmjs.org/next/-/next-16.2.6.tgz", - "integrity": "sha512-qOVgKJg1+At15NpeUP+eJgCHvTCgXsogweq87Ri/Ix7PkqQHg4sdaXmSFqKlgaIXE4kW0g25LE68W87UANlHtw==", - "license": "MIT", - "dependencies": { - "@next/env": "16.2.6", - "@swc/helpers": "0.5.15", - "baseline-browser-mapping": "^2.9.19", - "caniuse-lite": "^1.0.30001579", - "postcss": "8.4.31", - "styled-jsx": "5.1.6" - }, - "bin": { - "next": "dist/bin/next" - }, - "engines": { - "node": ">=20.9.0" - }, - "optionalDependencies": { - "@next/swc-darwin-arm64": "16.2.6", - "@next/swc-darwin-x64": "16.2.6", - "@next/swc-linux-arm64-gnu": "16.2.6", - "@next/swc-linux-arm64-musl": "16.2.6", - "@next/swc-linux-x64-gnu": "16.2.6", - "@next/swc-linux-x64-musl": "16.2.6", - "@next/swc-win32-arm64-msvc": "16.2.6", - "@next/swc-win32-x64-msvc": "16.2.6", - "sharp": "^0.34.5" - }, - "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_modules/next-intl": { - "version": "4.12.0", - "resolved": "https://registry.npmjs.org/next-intl/-/next-intl-4.12.0.tgz", - "integrity": "sha512-v8KpppWG0yLLlChJ3Of6uoPew9LeRDBAtY6vpJmF7YJmBZlHEzzoEL4w1g1dAU+VleEPNoXNm9hg1eEsKWV5hw==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/amannn" - } - ], - "license": "MIT", - "dependencies": { - "@formatjs/intl-localematcher": "^0.8.1", - "@parcel/watcher": "^2.4.1", - "@swc/core": "^1.15.2", - "icu-minify": "^4.12.0", - "negotiator": "^1.0.0", - "next-intl-swc-plugin-extractor": "^4.12.0", - "po-parser": "^2.1.1", - "use-intl": "^4.12.0" - }, - "peerDependencies": { - "next": "^12.0.0 || ^13.0.0 || ^14.0.0 || ^15.0.0 || ^16.0.0", - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || >=19.0.0-rc <19.0.0 || ^19.0.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/next-intl-swc-plugin-extractor": { - "version": "4.12.0", - "resolved": "https://registry.npmjs.org/next-intl-swc-plugin-extractor/-/next-intl-swc-plugin-extractor-4.12.0.tgz", - "integrity": "sha512-jUxVEu1Nryjt4YgaDktSys7ioOgQfcNPF/SF2dbPNxbVb6U+P1INRgHeCVN+EC59H2rnTFIQwbddmOCrUWFr3g==", - "license": "MIT" - }, - "node_modules/next-intl/node_modules/@swc/core": { - "version": "1.15.33", - "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.15.33.tgz", - "integrity": "sha512-jOlwnFV2xhuuZeAUILGFULeR6vDPfijEJ57evfocwznQldLU3w2cZ9bSDryY9ip+AsM3r1NJKzf47V2NXebkeQ==", - "hasInstallScript": true, - "license": "Apache-2.0", - "dependencies": { - "@swc/counter": "^0.1.3", - "@swc/types": "^0.1.26" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/swc" - }, - "optionalDependencies": { - "@swc/core-darwin-arm64": "1.15.33", - "@swc/core-darwin-x64": "1.15.33", - "@swc/core-linux-arm-gnueabihf": "1.15.33", - "@swc/core-linux-arm64-gnu": "1.15.33", - "@swc/core-linux-arm64-musl": "1.15.33", - "@swc/core-linux-ppc64-gnu": "1.15.33", - "@swc/core-linux-s390x-gnu": "1.15.33", - "@swc/core-linux-x64-gnu": "1.15.33", - "@swc/core-linux-x64-musl": "1.15.33", - "@swc/core-win32-arm64-msvc": "1.15.33", - "@swc/core-win32-ia32-msvc": "1.15.33", - "@swc/core-win32-x64-msvc": "1.15.33" - }, - "peerDependencies": { - "@swc/helpers": ">=0.5.17" - }, - "peerDependenciesMeta": { - "@swc/helpers": { - "optional": true - } - } - }, - "node_modules/next-intl/node_modules/@swc/helpers": { - "version": "0.5.21", - "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.21.tgz", - "integrity": "sha512-jI/VAmtdjB/RnI8GTnokyX7Ug8c+g+ffD6QRLa6XQewtnGyukKkKSk3wLTM3b5cjt1jNh9x0jfVlagdN2gDKQg==", - "license": "Apache-2.0", - "optional": true, - "peer": true, - "dependencies": { - "tslib": "^2.8.0" - } - }, - "node_modules/next-themes": { - "version": "0.4.6", - "resolved": "https://registry.npmjs.org/next-themes/-/next-themes-0.4.6.tgz", - "integrity": "sha512-pZvgD5L0IEvX5/9GWyHMf3m8BKiVQwsCMHfoFosXtXBMnaS0ZnIJ9ST4b4NqLVKDEm8QBxoNNGNaBv2JNF6XNA==", - "license": "MIT", - "peerDependencies": { - "react": "^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc", - "react-dom": "^16.8 || ^17 || ^18 || ^19 || ^19.0.0-rc" - } - }, - "node_modules/next/node_modules/postcss": { - "version": "8.4.31", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz", - "integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==", - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/postcss" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "nanoid": "^3.3.6", - "picocolors": "^1.0.0", - "source-map-js": "^1.0.2" - }, - "engines": { - "node": "^10 || ^12 || >=14" - } - }, - "node_modules/node-abi": { - "version": "3.92.0", - "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.92.0.tgz", - "integrity": "sha512-KdHvFWZjEKDf0cakgFjebl371GPsISX2oZHcuyKqM7DtogIsHrqKeLTo8wBHxaXRAQlY2PsPlZmfo+9ZCxEREQ==", - "license": "MIT", - "dependencies": { - "semver": "^7.3.5" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/node-abi/node_modules/semver": { - "version": "7.8.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", - "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/node-addon-api": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-7.1.1.tgz", - "integrity": "sha512-5m3bsyrjFWE1xf7nz7YXdN4udnVtXK6/Yfgn5qnahL6bCkf2yKt4k3nuTKAtT4r3IG8JNR2ncsIMdZuAzJjHQQ==", - "license": "MIT" - }, - "node_modules/node-domexception": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", - "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", - "deprecated": "Use your platform's native DOMException instead", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/jimmywarting" - }, - { - "type": "github", - "url": "https://paypal.me/jimmywarting" - } - ], - "license": "MIT", - "engines": { - "node": ">=10.5.0" - } - }, - "node_modules/node-exports-info": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/node-exports-info/-/node-exports-info-1.6.0.tgz", - "integrity": "sha512-pyFS63ptit/P5WqUkt+UUfe+4oevH+bFeIiPPdfb0pFeYEu/1ELnJu5l+5EcTKYL5M7zaAa7S8ddywgXypqKCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "array.prototype.flatmap": "^1.3.3", - "es-errors": "^1.3.0", - "object.entries": "^1.1.9", - "semver": "^6.3.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/node-releases": { - "version": "2.0.44", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.44.tgz", - "integrity": "sha512-5WUyunoPMsvvEhS8AxHtRzP+oA8UCkJ7YRxatWKjngndhDGLiqEVAQKWjFAiAiuL8zMRGzGSJxFnLetoa43qGQ==", - "license": "MIT" - }, - "node_modules/npm-run-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-6.0.0.tgz", - "integrity": "sha512-9qny7Z9DsQU8Ou39ERsPU4OZQlSTP47ShQzuKZ6PRXpYLtIFgl/DEBYEXKlvcEa+9tHVcK8CF81Y2V72qaZhWA==", - "license": "MIT", - "dependencies": { - "path-key": "^4.0.0", - "unicorn-magic": "^0.3.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/npm-run-path/node_modules/path-key": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-4.0.0.tgz", - "integrity": "sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/object-inspect": { - "version": "1.13.4", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", - "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object-treeify": { - "version": "1.1.33", - "resolved": "https://registry.npmjs.org/object-treeify/-/object-treeify-1.1.33.tgz", - "integrity": "sha512-EFVjAYfzWqWsBMRHPMAXLCDIJnpMhdWAqR7xG6M6a2cs6PMFpl/+Z20w9zDW4vkxOFfddegBKq9Rehd0bxWE7A==", - "license": "MIT", - "engines": { - "node": ">= 10" - } - }, - "node_modules/object.assign": { - "version": "4.1.7", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", - "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0", - "has-symbols": "^1.1.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.entries": { - "version": "1.1.9", - "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.9.tgz", - "integrity": "sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.fromentries": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", - "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.groupby": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", - "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.values": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.1.tgz", - "integrity": "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/on-finished": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", - "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", - "license": "MIT", - "dependencies": { - "ee-first": "1.1.1" - }, - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", - "license": "ISC", - "dependencies": { - "wrappy": "1" - } - }, - "node_modules/onetime": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-7.0.0.tgz", - "integrity": "sha512-VXJjc87FScF88uafS3JllDgvAm+c/Slfz06lorj2uAY34rlUu0Nt+v8wreiImcrgAjjIHp1rXpTDlLOGw29WwQ==", - "license": "MIT", - "dependencies": { - "mimic-function": "^5.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/open": { - "version": "11.0.0", - "resolved": "https://registry.npmjs.org/open/-/open-11.0.0.tgz", - "integrity": "sha512-smsWv2LzFjP03xmvFoJ331ss6h+jixfA4UUV/Bsiyuu4YJPfN+FIQGOIiv4w9/+MoHkfkJ22UIaQWRVFRfH6Vw==", - "license": "MIT", - "dependencies": { - "default-browser": "^5.4.0", - "define-lazy-prop": "^3.0.0", - "is-in-ssh": "^1.0.0", - "is-inside-container": "^1.0.0", - "powershell-utils": "^0.1.0", - "wsl-utils": "^0.3.0" - }, - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/optionator": { - "version": "0.9.4", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", - "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", - "dev": true, - "license": "MIT", - "dependencies": { - "deep-is": "^0.1.3", - "fast-levenshtein": "^2.0.6", - "levn": "^0.4.1", - "prelude-ls": "^1.2.1", - "type-check": "^0.4.0", - "word-wrap": "^1.2.5" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/ora": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/ora/-/ora-8.2.0.tgz", - "integrity": "sha512-weP+BZ8MVNnlCm8c0Qdc1WSWq4Qn7I+9CJGm7Qali6g44e/PUzbjNqJX5NJ9ljlNMosfJvg1fKEGILklK9cwnw==", - "license": "MIT", - "dependencies": { - "chalk": "^5.3.0", - "cli-cursor": "^5.0.0", - "cli-spinners": "^2.9.2", - "is-interactive": "^2.0.0", - "is-unicode-supported": "^2.0.0", - "log-symbols": "^6.0.0", - "stdin-discarder": "^0.2.2", - "string-width": "^7.2.0", - "strip-ansi": "^7.1.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/ora/node_modules/ansi-regex": { - "version": "6.2.2", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", - "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/ansi-regex?sponsor=1" - } - }, - "node_modules/ora/node_modules/chalk": { - "version": "5.6.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.6.2.tgz", - "integrity": "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==", - "license": "MIT", - "engines": { - "node": "^12.17.0 || ^14.13 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" - } - }, - "node_modules/ora/node_modules/emoji-regex": { - "version": "10.6.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-10.6.0.tgz", - "integrity": "sha512-toUI84YS5YmxW219erniWD0CIVOo46xGKColeNQRgOzDorgBi1v4D71/OFzgD9GO2UGKIv1C3Sp8DAn0+j5w7A==", - "license": "MIT" - }, - "node_modules/ora/node_modules/string-width": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-7.2.0.tgz", - "integrity": "sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ==", - "license": "MIT", - "dependencies": { - "emoji-regex": "^10.3.0", - "get-east-asian-width": "^1.0.0", - "strip-ansi": "^7.1.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/ora/node_modules/strip-ansi": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.2.0.tgz", - "integrity": "sha512-yDPMNjp4WyfYBkHnjIRLfca1i6KMyGCtsVgoKe/z1+6vukgaENdgGBZt+ZmKPc4gavvEZ5OgHfHdrazhgNyG7w==", - "license": "MIT", - "dependencies": { - "ansi-regex": "^6.2.2" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/chalk/strip-ansi?sponsor=1" - } - }, - "node_modules/outvariant": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/outvariant/-/outvariant-1.4.3.tgz", - "integrity": "sha512-+Sl2UErvtsoajRDKCE5/dBz4DIvHXQQnAxtQTF04OJxY0+DyZXSo5P5Bb7XYWOh81syohlYL24hbDwxedPUJCA==", - "license": "MIT" - }, - "node_modules/own-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", - "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "get-intrinsic": "^1.2.6", - "object-keys": "^1.1.1", - "safe-push-apply": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/p-limit": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", - "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "yocto-queue": "^0.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/p-locate": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", - "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-limit": "^3.0.2" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/pac-proxy-agent": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/pac-proxy-agent/-/pac-proxy-agent-7.2.0.tgz", - "integrity": "sha512-TEB8ESquiLMc0lV8vcd5Ql/JAKAoyzHFXaStwjkzpOpC5Yv+pIzLfHvjTSdf3vpa2bMiUQrg9i6276yn8666aA==", - "license": "MIT", - "dependencies": { - "@tootallnate/quickjs-emscripten": "^0.23.0", - "agent-base": "^7.1.2", - "debug": "^4.3.4", - "get-uri": "^6.0.1", - "http-proxy-agent": "^7.0.0", - "https-proxy-agent": "^7.0.6", - "pac-resolver": "^7.0.1", - "socks-proxy-agent": "^8.0.5" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/pac-resolver": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/pac-resolver/-/pac-resolver-7.0.1.tgz", - "integrity": "sha512-5NPgf87AT2STgwa2ntRMr45jTKrYBGkVU36yT0ig/n/GMAa3oPqhZfIQ2kMEimReg0+t9kZViDVZ83qfVUlckg==", - "license": "MIT", - "dependencies": { - "degenerator": "^5.0.0", - "netmask": "^2.0.2" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/parent-module": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", - "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", - "license": "MIT", - "dependencies": { - "callsites": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/parse-json": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", - "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", - "license": "MIT", - "dependencies": { - "@babel/code-frame": "^7.0.0", - "error-ex": "^1.3.1", - "json-parse-even-better-errors": "^2.3.0", - "lines-and-columns": "^1.1.6" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/parse-ms": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse-ms/-/parse-ms-4.0.0.tgz", - "integrity": "sha512-TXfryirbmq34y8QBwgqCVLi+8oA3oWx2eAnSn62ITyEhEYaWRlVZ2DvMM9eZbMs/RfxPu/PK/aBLyGj4IrqMHw==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/parseurl": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", - "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/path-browserify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-1.0.1.tgz", - "integrity": "sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g==", - "license": "MIT" - }, - "node_modules/path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/path-parse": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", - "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", - "dev": true, - "license": "MIT" - }, - "node_modules/path-to-regexp": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-6.3.0.tgz", - "integrity": "sha512-Yhpw4T9C6hPpgPeA28us07OJeqZ5EzQTkbfwuhsUg0c237RomFoETJgmp2sa3F/41gfLE6G5cqcYwznmeEeOlQ==", - "license": "MIT" - }, - "node_modules/pend": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", - "integrity": "sha512-F3asv42UuXchdzt+xXqfW1OGlVBe+mxa2mqI0pg5yAHZPvFmY3Y6drSf/GQ1A86WgWEN9Kzh/WrgKa6iGcHXLg==", - "license": "MIT" - }, - "node_modules/picocolors": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", - "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", - "license": "ISC" - }, - "node_modules/picomatch": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.2.tgz", - "integrity": "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA==", - "license": "MIT", - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/pkce-challenge": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/pkce-challenge/-/pkce-challenge-5.0.1.tgz", - "integrity": "sha512-wQ0b/W4Fr01qtpHlqSqspcj3EhBvimsdh0KlHhH8HRZnMsEa0ea2fTULOXOS9ccQr3om+GcGRk4e+isrZWV8qQ==", - "license": "MIT", - "engines": { - "node": ">=16.20.0" - } - }, - "node_modules/po-parser": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/po-parser/-/po-parser-2.1.1.tgz", - "integrity": "sha512-ECF4zHLbUItpUgE3OTtLKlPjeBN+fKEczj2zYjDfCGOzicNs0GK3Vg2IoAYwx7LH/XYw43fZQP6xnZ4TkNxSLQ==", - "license": "MIT" - }, - "node_modules/possible-typed-array-names": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz", - "integrity": "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/postcss": { - "version": "8.5.14", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.14.tgz", - "integrity": "sha512-SoSL4+OSEtR99LHFZQiJLkT59C5B1amGO1NzTwj7TT1qCUgUO6hxOvzkOYxD+vMrXBM3XJIKzokoERdqQq/Zmg==", - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/postcss/" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/postcss" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "nanoid": "^3.3.11", - "picocolors": "^1.1.1", - "source-map-js": "^1.2.1" - }, - "engines": { - "node": "^10 || ^12 || >=14" - } - }, - "node_modules/postcss-selector-parser": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-7.1.1.tgz", - "integrity": "sha512-orRsuYpJVw8LdAwqqLykBj9ecS5/cRHlI5+nvTo8LcCKmzDmqVORXtOIYEEQuL9D4BxtA1lm5isAqzQZCoQ6Eg==", - "license": "MIT", - "dependencies": { - "cssesc": "^3.0.0", - "util-deprecate": "^1.0.2" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/powershell-utils": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/powershell-utils/-/powershell-utils-0.1.0.tgz", - "integrity": "sha512-dM0jVuXJPsDN6DvRpea484tCUaMiXWjuCn++HGTqUWzGDjv5tZkEZldAJ/UMlqRYGFrD/etByo4/xOuC/snX2A==", - "license": "MIT", - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/prebuild-install": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", - "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", - "deprecated": "No longer maintained. Please contact the author of the relevant native addon; alternatives are available.", - "license": "MIT", - "dependencies": { - "detect-libc": "^2.0.0", - "expand-template": "^2.0.3", - "github-from-package": "0.0.0", - "minimist": "^1.2.3", - "mkdirp-classic": "^0.5.3", - "napi-build-utils": "^2.0.0", - "node-abi": "^3.3.0", - "pump": "^3.0.0", - "rc": "^1.2.7", - "simple-get": "^4.0.0", - "tar-fs": "^2.0.0", - "tunnel-agent": "^0.6.0" - }, - "bin": { - "prebuild-install": "bin.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/prelude-ls": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", - "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/pretty-ms": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/pretty-ms/-/pretty-ms-9.3.0.tgz", - "integrity": "sha512-gjVS5hOP+M3wMm5nmNOucbIrqudzs9v/57bWRHQWLYklXqoXKrVfYW2W9+glfGsqtPgpiz5WwyEEB+ksXIx3gQ==", - "license": "MIT", - "dependencies": { - "parse-ms": "^4.0.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/progress": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", - "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", - "license": "MIT", - "engines": { - "node": ">=0.4.0" - } - }, - "node_modules/prompts": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", - "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", - "license": "MIT", - "dependencies": { - "kleur": "^3.0.3", - "sisteransi": "^1.0.5" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/prompts/node_modules/kleur": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", - "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/prop-types": { - "version": "15.8.1", - "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", - "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", - "dev": true, - "license": "MIT", - "dependencies": { - "loose-envify": "^1.4.0", - "object-assign": "^4.1.1", - "react-is": "^16.13.1" - } - }, - "node_modules/proxy-addr": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", - "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", - "license": "MIT", - "dependencies": { - "forwarded": "0.2.0", - "ipaddr.js": "1.9.1" - }, - "engines": { - "node": ">= 0.10" - } - }, - "node_modules/proxy-agent": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/proxy-agent/-/proxy-agent-6.5.0.tgz", - "integrity": "sha512-TmatMXdr2KlRiA2CyDu8GqR8EjahTG3aY3nXjdzFyoZbmB8hrBsTyMezhULIXKnC0jpfjlmiZ3+EaCzoInSu/A==", - "license": "MIT", - "dependencies": { - "agent-base": "^7.1.2", - "debug": "^4.3.4", - "http-proxy-agent": "^7.0.1", - "https-proxy-agent": "^7.0.6", - "lru-cache": "^7.14.1", - "pac-proxy-agent": "^7.1.0", - "proxy-from-env": "^1.1.0", - "socks-proxy-agent": "^8.0.5" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/proxy-agent/node_modules/lru-cache": { - "version": "7.18.3", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-7.18.3.tgz", - "integrity": "sha512-jumlc0BIUrS3qJGgIkWZsyfAM7NCWiBcCDhnd+3NNM5KbBmLTgHVfWBcg6W+rLUsIpzpERPsvwUP7CckAQSOoA==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/proxy-from-env": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", - "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", - "license": "MIT" - }, - "node_modules/pump": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.4.tgz", - "integrity": "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA==", - "license": "MIT", - "dependencies": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, - "node_modules/punycode": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", - "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/puppeteer": { - "version": "24.43.1", - "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-24.43.1.tgz", - "integrity": "sha512-/FSOViCrqRdb1HDocpsM9Z1giA71gTQPUt3SpHGVRALKAy/rJr1fLFYZW9F23qPxqVxTHQnbh/5B5opJST3kAw==", - "hasInstallScript": true, - "license": "Apache-2.0", - "dependencies": { - "@puppeteer/browsers": "2.13.2", - "chromium-bidi": "14.0.0", - "cosmiconfig": "^9.0.0", - "devtools-protocol": "0.0.1608973", - "puppeteer-core": "24.43.1", - "typed-query-selector": "^2.12.2" - }, - "bin": { - "puppeteer": "lib/cjs/puppeteer/node/cli.js" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/puppeteer-core": { - "version": "24.43.1", - "resolved": "https://registry.npmjs.org/puppeteer-core/-/puppeteer-core-24.43.1.tgz", - "integrity": "sha512-T5ScUMAsmhdNbgDR41AGESYeS6V9MSgetkSnVhhW+gXvzC42VesKCn5ld87gAZDJ6vLHL9GkRvY9WtQWSnwFbw==", - "license": "Apache-2.0", - "dependencies": { - "@puppeteer/browsers": "2.13.2", - "chromium-bidi": "14.0.0", - "debug": "^4.4.3", - "devtools-protocol": "0.0.1608973", - "typed-query-selector": "^2.12.2", - "webdriver-bidi-protocol": "0.4.1", - "ws": "^8.20.0" - }, - "engines": { - "node": ">=18" - } - }, - "node_modules/qs": { - "version": "6.15.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.15.1.tgz", - "integrity": "sha512-6YHEFRL9mfgcAvql/XhwTvf5jKcOiiupt2FiJxHkiX1z4j7WL8J/jRHYLluORvc1XxB5rV20KoeK00gVJamspg==", - "license": "BSD-3-Clause", - "dependencies": { - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">=0.6" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/range-parser": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", - "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/raw-body": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.2.tgz", - "integrity": "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA==", - "license": "MIT", - "dependencies": { - "bytes": "~3.1.2", - "http-errors": "~2.0.1", - "iconv-lite": "~0.7.0", - "unpipe": "~1.0.0" - }, - "engines": { - "node": ">= 0.10" - } - }, - "node_modules/rc": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", - "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", - "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", - "dependencies": { - "deep-extend": "^0.6.0", - "ini": "~1.3.0", - "minimist": "^1.2.0", - "strip-json-comments": "~2.0.1" - }, - "bin": { - "rc": "cli.js" - } - }, - "node_modules/rc/node_modules/strip-json-comments": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", - "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/react": { - "version": "19.2.4", - "resolved": "https://registry.npmjs.org/react/-/react-19.2.4.tgz", - "integrity": "sha512-9nfp2hYpCwOjAN+8TZFGhtWEwgvWHXqESH8qT89AT/lWklpLON22Lc8pEtnpsZz7VmawabSU0gCjnj8aC0euHQ==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/react-dom": { - "version": "19.2.4", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.4.tgz", - "integrity": "sha512-AXJdLo8kgMbimY95O2aKQqsz2iWi9jMgKJhRBAxECE4IFxfcazB2LmzloIoibJI3C12IlY20+KFaLv+71bUJeQ==", - "license": "MIT", - "dependencies": { - "scheduler": "^0.27.0" - }, - "peerDependencies": { - "react": "^19.2.4" - } - }, - "node_modules/react-is": { - "version": "16.13.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", - "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", - "license": "MIT" - }, - "node_modules/react-redux": { - "version": "9.2.0", - "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-9.2.0.tgz", - "integrity": "sha512-ROY9fvHhwOD9ySfrF0wmvu//bKCQ6AeZZq1nJNtbDC+kk5DuSuNX/n6YWYF/SYy7bSba4D4FSz8DJeKY/S/r+g==", - "license": "MIT", - "dependencies": { - "@types/use-sync-external-store": "^0.0.6", - "use-sync-external-store": "^1.4.0" - }, - "peerDependencies": { - "@types/react": "^18.2.25 || ^19", - "react": "^18.0 || ^19", - "redux": "^5.0.0" - }, - "peerDependenciesMeta": { - "@types/react": { - "optional": true - }, - "redux": { - "optional": true - } - } - }, - "node_modules/readable-stream": { - "version": "3.6.2", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", - "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", - "license": "MIT", - "dependencies": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/recast": { - "version": "0.23.11", - "resolved": "https://registry.npmjs.org/recast/-/recast-0.23.11.tgz", - "integrity": "sha512-YTUo+Flmw4ZXiWfQKGcwwc11KnoRAYgzAE2E7mXKCjSviTKShtxBsN6YUUBB2gtaBzKzeKunxhUwNHQuRryhWA==", - "license": "MIT", - "dependencies": { - "ast-types": "^0.16.1", - "esprima": "~4.0.0", - "source-map": "~0.6.1", - "tiny-invariant": "^1.3.3", - "tslib": "^2.0.1" - }, - "engines": { - "node": ">= 4" - } - }, - "node_modules/recast/node_modules/ast-types": { - "version": "0.16.1", - "resolved": "https://registry.npmjs.org/ast-types/-/ast-types-0.16.1.tgz", - "integrity": "sha512-6t10qk83GOG8p0vKmaCr8eiilZwO171AvbROMtvvNiwrTly62t+7XkA8RdIIVbpMhCASAsxgAzdRSwh6nw/5Dg==", - "license": "MIT", - "dependencies": { - "tslib": "^2.0.1" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/recharts": { - "version": "3.8.1", - "resolved": "https://registry.npmjs.org/recharts/-/recharts-3.8.1.tgz", - "integrity": "sha512-mwzmO1s9sFL0TduUpwndxCUNoXsBw3u3E/0+A+cLcrSfQitSG62L32N69GhqUrrT5qKcAE3pCGVINC6pqkBBQg==", - "license": "MIT", - "workspaces": [ - "www" - ], - "dependencies": { - "@reduxjs/toolkit": "^1.9.0 || 2.x.x", - "clsx": "^2.1.1", - "decimal.js-light": "^2.5.1", - "es-toolkit": "^1.39.3", - "eventemitter3": "^5.0.1", - "immer": "^10.1.1", - "react-redux": "8.x.x || 9.x.x", - "reselect": "5.1.1", - "tiny-invariant": "^1.3.3", - "use-sync-external-store": "^1.2.2", - "victory-vendor": "^37.0.2" - }, - "engines": { - "node": ">=18" - }, - "peerDependencies": { - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", - "react-dom": "^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", - "react-is": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/redux": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", - "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==", - "license": "MIT" - }, - "node_modules/redux-thunk": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", - "integrity": "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw==", - "license": "MIT", - "peerDependencies": { - "redux": "^5.0.0" - } - }, - "node_modules/reflect.getprototypeof": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", - "integrity": "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.7", - "get-proto": "^1.0.1", - "which-builtin-type": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regexp.prototype.flags": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", - "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", - "dev": true, - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/require-directory": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/require-from-string": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", - "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/reselect": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/reselect/-/reselect-5.1.1.tgz", - "integrity": "sha512-K/BG6eIky/SBpzfHZv/dd+9JBFiS4SWV7FIujVyJRux6e45+73RaUHXLmIR1f7WOMaQ0U1km6qwklRQxpJJY0w==", - "license": "MIT" - }, - "node_modules/resolve": { - "version": "2.0.0-next.6", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.6.tgz", - "integrity": "sha512-3JmVl5hMGtJ3kMmB3zi3DL25KfkCEyy3Tw7Gmw7z5w8M9WlwoPFnIvwChzu1+cF3iaK3sp18hhPz8ANeimdJfA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "is-core-module": "^2.16.1", - "node-exports-info": "^1.6.0", - "object-keys": "^1.1.1", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - }, - "bin": { - "resolve": "bin/resolve" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/resolve-pkg-maps": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", - "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" - } - }, - "node_modules/restore-cursor": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-5.1.0.tgz", - "integrity": "sha512-oMA2dcrw6u0YfxJQXm342bFKX/E4sG9rbTzO9ptUcR/e8A33cHuvStiYOwH7fszkZlZ1z/ta9AAoPk2F4qIOHA==", - "license": "MIT", - "dependencies": { - "onetime": "^7.0.0", - "signal-exit": "^4.1.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/rettime": { - "version": "0.11.11", - "resolved": "https://registry.npmjs.org/rettime/-/rettime-0.11.11.tgz", - "integrity": "sha512-ILJRqVWBCTlg9r42fFgwVZx1gnFAcQF8mRoMkbgQfIrjEDf9nbBFDFx00oloOa+Q869FUtaYDXZvEfnecQSCoQ==", - "license": "MIT" - }, - "node_modules/reusify": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", - "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", - "license": "MIT", - "engines": { - "iojs": ">=1.0.0", - "node": ">=0.10.0" - } - }, - "node_modules/router": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/router/-/router-2.2.0.tgz", - "integrity": "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==", - "license": "MIT", - "dependencies": { - "debug": "^4.4.0", - "depd": "^2.0.0", - "is-promise": "^4.0.0", - "parseurl": "^1.3.3", - "path-to-regexp": "^8.0.0" - }, - "engines": { - "node": ">= 18" - } - }, - "node_modules/router/node_modules/path-to-regexp": { - "version": "8.4.2", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.4.2.tgz", - "integrity": "sha512-qRcuIdP69NPm4qbACK+aDogI5CBDMi1jKe0ry5rSQJz8JVLsC7jV8XpiJjGRLLol3N+R5ihGYcrPLTno6pAdBA==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/run-applescript": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/run-applescript/-/run-applescript-7.1.0.tgz", - "integrity": "sha512-DPe5pVFaAsinSaV6QjQ6gdiedWDcRCbUuiQfQa2wmWV7+xC9bGulGI8+TdRmoFkAPaBXk8CrAbnlY2ISniJ47Q==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "queue-microtask": "^1.2.2" - } - }, - "node_modules/safe-array-concat": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.4.tgz", - "integrity": "sha512-wtZlHyOje6OZTGqAoaDKxFkgRtkF9CnHAVnCHKfuj200wAgL+bSJhdsCD2l0Qx/2ekEXjPWcyKkfGb5CPboslg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.9", - "call-bound": "^1.0.4", - "get-intrinsic": "^1.3.0", - "has-symbols": "^1.1.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">=0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/safe-push-apply": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", - "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-regex-test": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", - "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-regex": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "license": "MIT" - }, - "node_modules/scheduler": { - "version": "0.27.0", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz", - "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==", - "license": "MIT" - }, - "node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - } - }, - "node_modules/send": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/send/-/send-1.2.1.tgz", - "integrity": "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ==", - "license": "MIT", - "dependencies": { - "debug": "^4.4.3", - "encodeurl": "^2.0.0", - "escape-html": "^1.0.3", - "etag": "^1.8.1", - "fresh": "^2.0.0", - "http-errors": "^2.0.1", - "mime-types": "^3.0.2", - "ms": "^2.1.3", - "on-finished": "^2.4.1", - "range-parser": "^1.2.1", - "statuses": "^2.0.2" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/serve-static": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.2.1.tgz", - "integrity": "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw==", - "license": "MIT", - "dependencies": { - "encodeurl": "^2.0.0", - "escape-html": "^1.0.3", - "parseurl": "^1.3.3", - "send": "^1.2.0" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/server-only": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/server-only/-/server-only-0.0.1.tgz", - "integrity": "sha512-qepMx2JxAa5jjfzxG79yPPq+8BuFToHd1hm7kI+Z4zAq1ftQiP7HcxMhDDItrbtwVeLg/cY2JnKnrcFkmiswNA==", - "license": "MIT" - }, - "node_modules/set-cookie-parser": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-3.1.0.tgz", - "integrity": "sha512-kjnC1DXBHcxaOaOXBHBeRtltsDG2nUiUni+jP92M9gYdW12rsmx92UsfpH7o5tDRs7I1ZZPSQJQGv3UaRfCiuw==", - "license": "MIT" - }, - "node_modules/set-function-length": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", - "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.4", - "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-function-name": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", - "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-proto": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/set-proto/-/set-proto-1.0.0.tgz", - "integrity": "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/setprototypeof": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", - "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", - "license": "ISC" - }, - "node_modules/shadcn": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/shadcn/-/shadcn-4.7.0.tgz", - "integrity": "sha512-70fwnesNrY1GgeD7Kdzn+3SsYeyfibm8immsA5L68+OusoPTvYF01oWExl8/latKpMpvVXcbgdbbE6VFBJQ38w==", - "license": "MIT", - "dependencies": { - "@babel/core": "^7.28.0", - "@babel/parser": "^7.28.0", - "@babel/plugin-transform-typescript": "^7.28.0", - "@babel/preset-typescript": "^7.27.1", - "@dotenvx/dotenvx": "^1.48.4", - "@modelcontextprotocol/sdk": "^1.26.0", - "@types/validate-npm-package-name": "^4.0.2", - "browserslist": "^4.26.2", - "commander": "^14.0.0", - "cosmiconfig": "^9.0.0", - "dedent": "^1.6.0", - "deepmerge": "^4.3.1", - "diff": "^8.0.2", - "execa": "^9.6.0", - "fast-glob": "^3.3.3", - "fs-extra": "^11.3.1", - "fuzzysort": "^3.1.0", - "https-proxy-agent": "^7.0.6", - "kleur": "^4.1.5", - "msw": "^2.10.4", - "node-fetch": "^3.3.2", - "open": "^11.0.0", - "ora": "^8.2.0", - "postcss": "^8.5.6", - "postcss-selector-parser": "^7.1.0", - "prompts": "^2.4.2", - "recast": "^0.23.11", - "stringify-object": "^5.0.0", - "tailwind-merge": "^3.0.1", - "ts-morph": "^26.0.0", - "tsconfig-paths": "^4.2.0", - "validate-npm-package-name": "^7.0.1", - "zod": "^3.24.1", - "zod-to-json-schema": "^3.24.6" - }, - "bin": { - "shadcn": "dist/index.js" - } - }, - "node_modules/shadcn/node_modules/data-uri-to-buffer": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz", - "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==", - "license": "MIT", - "engines": { - "node": ">= 12" - } - }, - "node_modules/shadcn/node_modules/fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "license": "MIT", - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.8" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/shadcn/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/shadcn/node_modules/node-fetch": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.2.tgz", - "integrity": "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==", - "license": "MIT", - "dependencies": { - "data-uri-to-buffer": "^4.0.0", - "fetch-blob": "^3.1.4", - "formdata-polyfill": "^4.0.10" - }, - "engines": { - "node": "^12.20.0 || ^14.13.1 || >=16.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/node-fetch" - } - }, - "node_modules/shadcn/node_modules/tsconfig-paths": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-4.2.0.tgz", - "integrity": "sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg==", - "license": "MIT", - "dependencies": { - "json5": "^2.2.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/shadcn/node_modules/zod": { - "version": "3.25.76", - "resolved": "https://registry.npmjs.org/zod/-/zod-3.25.76.tgz", - "integrity": "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/colinhacks" - } - }, - "node_modules/sharp": { - "version": "0.34.5", - "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.34.5.tgz", - "integrity": "sha512-Ou9I5Ft9WNcCbXrU9cMgPBcCK8LiwLqcbywW3t4oDV37n1pzpuNLsYiAV8eODnjbtQlSDwZ2cUEeQz4E54Hltg==", - "hasInstallScript": true, - "license": "Apache-2.0", - "optional": true, - "dependencies": { - "@img/colour": "^1.0.0", - "detect-libc": "^2.1.2", - "semver": "^7.7.3" - }, - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "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" - } - }, - "node_modules/sharp/node_modules/semver": { - "version": "7.8.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", - "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", - "license": "ISC", - "optional": true, - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/shebang-command": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "license": "MIT", - "dependencies": { - "shebang-regex": "^3.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/side-channel": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", - "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3", - "side-channel-list": "^1.0.0", - "side-channel-map": "^1.0.1", - "side-channel-weakmap": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-list": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.1.tgz", - "integrity": "sha512-mjn/0bi/oUURjc5Xl7IaWi/OJJJumuoJFQJfDDyO46+hBWsfaVM65TBHq2eoZBhzl9EchxOijpkbRC8SVBQU0w==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-map": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", - "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-weakmap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", - "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3", - "side-channel-map": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/signal-exit": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", - "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", - "license": "ISC", - "engines": { - "node": ">=14" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, - "node_modules/simple-concat": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", - "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/simple-get": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", - "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "decompress-response": "^6.0.0", - "once": "^1.3.1", - "simple-concat": "^1.0.0" - } - }, - "node_modules/sisteransi": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", - "license": "MIT" - }, - "node_modules/smart-buffer": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", - "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", - "license": "MIT", - "engines": { - "node": ">= 6.0.0", - "npm": ">= 3.0.0" - } - }, - "node_modules/socks": { - "version": "2.8.9", - "resolved": "https://registry.npmjs.org/socks/-/socks-2.8.9.tgz", - "integrity": "sha512-LJhUYUvItdQ0LkJTmPeaEObWXAqFyfmP85x0tch/ez9cahmhlBBLbIqDFnvBnUJGagb0JbIQrkBs1wJ+yRYpEw==", - "license": "MIT", - "dependencies": { - "ip-address": "^10.1.1", - "smart-buffer": "^4.2.0" - }, - "engines": { - "node": ">= 10.0.0", - "npm": ">= 3.0.0" - } - }, - "node_modules/socks-proxy-agent": { - "version": "8.0.5", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.5.tgz", - "integrity": "sha512-HehCEsotFqbPW9sJ8WVYB6UbmIMv7kUUORIF2Nncq4VQvBfNBLibW9YZR5dlYCSUhwcD628pRllm7n+E+YTzJw==", - "license": "MIT", - "dependencies": { - "agent-base": "^7.1.2", - "debug": "^4.3.4", - "socks": "^2.8.3" - }, - "engines": { - "node": ">= 14" - } - }, - "node_modules/sonner": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/sonner/-/sonner-2.0.7.tgz", - "integrity": "sha512-W6ZN4p58k8aDKA4XPcx2hpIQXBRAgyiWVkYhT7CvK6D3iAu7xjvVyhQHg2/iaKJZ1XVJ4r7XuwGL+WGEK37i9w==", - "license": "MIT", - "peerDependencies": { - "react": "^18.0.0 || ^19.0.0 || ^19.0.0-rc", - "react-dom": "^18.0.0 || ^19.0.0 || ^19.0.0-rc" - } - }, - "node_modules/source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "license": "BSD-3-Clause", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/source-map-js": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", - "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", - "license": "BSD-3-Clause", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/stable-hash": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/stable-hash/-/stable-hash-0.0.5.tgz", - "integrity": "sha512-+L3ccpzibovGXFK+Ap/f8LOS0ahMrHTf3xu7mMLSpEGU0EO9ucaysSylKo9eRDFNhWve/y275iPmIZ4z39a9iA==", - "dev": true, - "license": "MIT" - }, - "node_modules/standardwebhooks": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/standardwebhooks/-/standardwebhooks-1.0.0.tgz", - "integrity": "sha512-BbHGOQK9olHPMvQNHWul6MYlrRTAOKn03rOe4A8O3CLWhNf4YHBqq2HJKKC+sfqpxiBY52pNeesD6jIiLDz8jg==", - "license": "MIT", - "dependencies": { - "@stablelib/base64": "^1.0.0", - "fast-sha256": "^1.3.0" - } - }, - "node_modules/statuses": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", - "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/stdin-discarder": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/stdin-discarder/-/stdin-discarder-0.2.2.tgz", - "integrity": "sha512-UhDfHmA92YAlNnCfhmq0VeNL5bDbiZGg7sZ2IvPsXubGkiNa9EC+tUTsjBRsYUAz87btI6/1wf4XoVvQ3uRnmQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/stop-iteration-iterator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.1.0.tgz", - "integrity": "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "internal-slot": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/streamx": { - "version": "2.25.0", - "resolved": "https://registry.npmjs.org/streamx/-/streamx-2.25.0.tgz", - "integrity": "sha512-0nQuG6jf1w+wddNEEXCF4nTg3LtufWINB5eFEN+5TNZW7KWJp6x87+JFL43vaAUPyCfH1wID+mNVyW6OHtFamg==", - "license": "MIT", - "dependencies": { - "events-universal": "^1.0.0", - "fast-fifo": "^1.3.2", - "text-decoder": "^1.1.0" - } - }, - "node_modules/strict-event-emitter": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/strict-event-emitter/-/strict-event-emitter-0.5.1.tgz", - "integrity": "sha512-vMgjE/GGEPEFnhFub6pa4FmJBRBVOLpIII2hvCZ8Kzb7K0hlHo7mQv6xYrBvCL2LtAIBwFUK8wvuJgTVSQ5MFQ==", - "license": "MIT" - }, - "node_modules/string_decoder": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", - "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", - "license": "MIT", - "dependencies": { - "safe-buffer": "~5.2.0" - } - }, - "node_modules/string-width": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", - "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", - "license": "MIT", - "dependencies": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/string-width/node_modules/emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "license": "MIT" - }, - "node_modules/string.prototype.includes": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.includes/-/string.prototype.includes-2.0.1.tgz", - "integrity": "sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.3" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/string.prototype.matchall": { - "version": "4.0.12", - "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.12.tgz", - "integrity": "sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.6", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.6", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "internal-slot": "^1.1.0", - "regexp.prototype.flags": "^1.5.3", - "set-function-name": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.repeat": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/string.prototype.repeat/-/string.prototype.repeat-1.0.0.tgz", - "integrity": "sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - } - }, - "node_modules/string.prototype.trim": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", - "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-data-property": "^1.1.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-object-atoms": "^1.0.0", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimend": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", - "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimstart": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", - "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/stringify-object": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/stringify-object/-/stringify-object-5.0.0.tgz", - "integrity": "sha512-zaJYxz2FtcMb4f+g60KsRNFOpVMUyuJgA51Zi5Z1DOTC3S59+OQiVOzE9GZt0x72uBGWKsQIuBKeF9iusmKFsg==", - "license": "BSD-2-Clause", - "dependencies": { - "get-own-enumerable-keys": "^1.0.0", - "is-obj": "^3.0.0", - "is-regexp": "^3.1.0" - }, - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/yeoman/stringify-object?sponsor=1" - } - }, - "node_modules/strip-ansi": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", - "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", - "license": "MIT", - "dependencies": { - "ansi-regex": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/strip-final-newline": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-4.0.0.tgz", - "integrity": "sha512-aulFJcD6YK8V1G7iRB5tigAP4TsHBZZrOV8pjV++zdUwmeV8uzbY7yn6h9MswN62adStNZFuCIx4haBnRuMDaw==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/strip-json-comments": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/styled-jsx": { - "version": "5.1.6", - "resolved": "https://registry.npmjs.org/styled-jsx/-/styled-jsx-5.1.6.tgz", - "integrity": "sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA==", - "license": "MIT", - "dependencies": { - "client-only": "0.0.1" - }, - "engines": { - "node": ">= 12.0.0" - }, - "peerDependencies": { - "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 - } - } - }, - "node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/supports-preserve-symlinks-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", - "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/tagged-tag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/tagged-tag/-/tagged-tag-1.0.0.tgz", - "integrity": "sha512-yEFYrVhod+hdNyx7g5Bnkkb0G6si8HJurOoOEgC8B/O0uXLHlaey/65KRv6cuWBNhBgHKAROVpc7QyYqE5gFng==", - "license": "MIT", - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/tailwind-merge": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/tailwind-merge/-/tailwind-merge-3.6.0.tgz", - "integrity": "sha512-uxL7qAVQriqRQPAyK3pj66VqskWqoZ37PW94jwOTwNfq/z9oyu1V+eqrZqtR2+fCiXdYOZe/Modt8GtvqNzu+w==", - "license": "MIT", - "funding": { - "type": "github", - "url": "https://github.com/sponsors/dcastil" - } - }, - "node_modules/tailwindcss": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.3.0.tgz", - "integrity": "sha512-y6nxMGB1nMW9R6k96e5gdIFzcfL/gTJRNaqGes1YvkLnPVXzWgbqFF2yLC0T8G774n24cx3Pe8XrKoniCOAH+Q==", - "dev": true, - "license": "MIT" - }, - "node_modules/tapable": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.3.tgz", - "integrity": "sha512-uxc/zpqFg6x7C8vOE7lh6Lbda8eEL9zmVm/PLeTPBRhh1xCgdWaQ+J1CUieGpIfm2HdtsUpRv+HshiasBMcc6A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - } - }, - "node_modules/tar-fs": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.4.tgz", - "integrity": "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ==", - "license": "MIT", - "dependencies": { - "chownr": "^1.1.1", - "mkdirp-classic": "^0.5.2", - "pump": "^3.0.0", - "tar-stream": "^2.1.4" - } - }, - "node_modules/tar-stream": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", - "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", - "license": "MIT", - "dependencies": { - "bl": "^4.0.3", - "end-of-stream": "^1.4.1", - "fs-constants": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^3.1.1" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/teex": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/teex/-/teex-1.0.1.tgz", - "integrity": "sha512-eYE6iEI62Ni1H8oIa7KlDU6uQBtqr4Eajni3wX7rpfXD8ysFx8z0+dri+KWEPWpBsxXfxu58x/0jvTVT1ekOSg==", - "license": "MIT", - "dependencies": { - "streamx": "^2.12.5" - } - }, - "node_modules/text-decoder": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/text-decoder/-/text-decoder-1.2.7.tgz", - "integrity": "sha512-vlLytXkeP4xvEq2otHeJfSQIRyWxo/oZGEbXrtEEF9Hnmrdly59sUbzZ/QgyWuLYHctCHxFF4tRQZNQ9k60ExQ==", - "license": "Apache-2.0", - "dependencies": { - "b4a": "^1.6.4" - } - }, - "node_modules/tiny-invariant": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/tiny-invariant/-/tiny-invariant-1.3.3.tgz", - "integrity": "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==", - "license": "MIT" - }, - "node_modules/tinyglobby": { - "version": "0.2.16", - "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz", - "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==", - "dev": true, - "license": "MIT", - "dependencies": { - "fdir": "^6.5.0", - "picomatch": "^4.0.4" - }, - "engines": { - "node": ">=12.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/SuperchupuDev" - } - }, - "node_modules/tinyglobby/node_modules/fdir": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", - "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12.0.0" - }, - "peerDependencies": { - "picomatch": "^3 || ^4" - }, - "peerDependenciesMeta": { - "picomatch": { - "optional": true - } - } - }, - "node_modules/tinyglobby/node_modules/picomatch": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", - "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/tldts": { - "version": "7.0.30", - "resolved": "https://registry.npmjs.org/tldts/-/tldts-7.0.30.tgz", - "integrity": "sha512-ELrFxuqsDdHUwoh0XxDbxuLD3Wnz49Z57IFvTtvWy1hJdcMZjXLIuonjilCiWHlT2GbE4Wlv1wKVTzDFnXH1aw==", - "license": "MIT", - "dependencies": { - "tldts-core": "^7.0.30" - }, - "bin": { - "tldts": "bin/cli.js" - } - }, - "node_modules/tldts-core": { - "version": "7.0.30", - "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-7.0.30.tgz", - "integrity": "sha512-uiHN8PIB1VmWyS98eZYja4xzlYqeFZVjb4OuYlJQnZAuJhMw4PbKQOKgHKhBdJR3FE/t5mUQ1Kd80++B+qhD1Q==", - "license": "MIT" - }, - "node_modules/to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "license": "MIT", - "dependencies": { - "is-number": "^7.0.0" - }, - "engines": { - "node": ">=8.0" - } - }, - "node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } - }, - "node_modules/tough-cookie": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-6.0.1.tgz", - "integrity": "sha512-LktZQb3IeoUWB9lqR5EWTHgW/VTITCXg4D21M+lvybRVdylLrRMnqaIONLVb5mav8vM19m44HIcGq4qASeu2Qw==", - "license": "BSD-3-Clause", - "dependencies": { - "tldts": "^7.0.5" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/ts-algebra": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ts-algebra/-/ts-algebra-2.0.0.tgz", - "integrity": "sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw==", - "license": "MIT" - }, - "node_modules/ts-api-utils": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", - "integrity": "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18.12" - }, - "peerDependencies": { - "typescript": ">=4.8.4" - } - }, - "node_modules/ts-morph": { - "version": "26.0.0", - "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-26.0.0.tgz", - "integrity": "sha512-ztMO++owQnz8c/gIENcM9XfCEzgoGphTv+nKpYNM1bgsdOVC/jRZuEBf6N+mLLDNg68Kl+GgUZfOySaRiG1/Ug==", - "license": "MIT", - "dependencies": { - "@ts-morph/common": "~0.27.0", - "code-block-writer": "^13.0.3" - } - }, - "node_modules/tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" - } - }, - "node_modules/tsconfig-paths/node_modules/json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", - "dev": true, - "license": "MIT", - "dependencies": { - "minimist": "^1.2.0" - }, - "bin": { - "json5": "lib/cli.js" - } - }, - "node_modules/tslib": { - "version": "2.8.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", - "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", - "license": "0BSD" - }, - "node_modules/tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", - "license": "Apache-2.0", - "dependencies": { - "safe-buffer": "^5.0.1" - }, - "engines": { - "node": "*" - } - }, - "node_modules/tw-animate-css": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/tw-animate-css/-/tw-animate-css-1.4.0.tgz", - "integrity": "sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/Wombosvideo" - } - }, - "node_modules/type-check": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", - "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", - "dev": true, - "license": "MIT", - "dependencies": { - "prelude-ls": "^1.2.1" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/type-fest": { - "version": "5.6.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-5.6.0.tgz", - "integrity": "sha512-8ZiHFm91orbSAe2PSAiSVBVko18pbhbiB3U9GglSzF/zCGkR+rxpHx6sEMCUm4kxY4LjDIUGgCfUMtwfZfjfUA==", - "license": "(MIT OR CC0-1.0)", - "dependencies": { - "tagged-tag": "^1.0.0" - }, - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/type-is": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz", - "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==", - "license": "MIT", - "dependencies": { - "content-type": "^1.0.5", - "media-typer": "^1.1.0", - "mime-types": "^3.0.0" - }, - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/typed-array-buffer": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", - "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/typed-array-byte-length": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", - "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-byte-offset": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", - "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.15", - "reflect.getprototypeof": "^1.0.9" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-length": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", - "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "is-typed-array": "^1.1.13", - "possible-typed-array-names": "^1.0.0", - "reflect.getprototypeof": "^1.0.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-query-selector": { - "version": "2.12.2", - "resolved": "https://registry.npmjs.org/typed-query-selector/-/typed-query-selector-2.12.2.tgz", - "integrity": "sha512-EOPFbyIub4ngnEdqi2yOcNeDLaX/0jcE1JoAXQDDMIthap7FoN795lc/SHfIq2d416VufXpM8z/lD+WRm2gfOQ==", - "license": "MIT" - }, - "node_modules/typescript": { - "version": "5.9.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", - "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", - "devOptional": true, - "license": "Apache-2.0", - "bin": { - "tsc": "bin/tsc", - "tsserver": "bin/tsserver" - }, - "engines": { - "node": ">=14.17" - } - }, - "node_modules/typescript-eslint": { - "version": "8.59.3", - "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.59.3.tgz", - "integrity": "sha512-KgusgyDgG4LI8Ih/sWaCtZ06tckLAS5CvT5A4D1Q7bYVoAAyzwiZvE4BmwDHkhRVkvhRBepKeASoFzQetha7Fg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@typescript-eslint/eslint-plugin": "8.59.3", - "@typescript-eslint/parser": "8.59.3", - "@typescript-eslint/typescript-estree": "8.59.3", - "@typescript-eslint/utils": "8.59.3" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", - "typescript": ">=4.8.4 <6.1.0" - } - }, - "node_modules/unbox-primitive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", - "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-bigints": "^1.0.2", - "has-symbols": "^1.1.0", - "which-boxed-primitive": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/undici-types": { - "version": "6.21.0", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", - "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", - "license": "MIT" - }, - "node_modules/unicorn-magic": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/unicorn-magic/-/unicorn-magic-0.3.0.tgz", - "integrity": "sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/universalify": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", - "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", - "license": "MIT", - "engines": { - "node": ">= 10.0.0" - } - }, - "node_modules/unpipe": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/unrs-resolver": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/unrs-resolver/-/unrs-resolver-1.11.1.tgz", - "integrity": "sha512-bSjt9pjaEBnNiGgc9rUiHGKv5l4/TGzDmYw3RhnkJGtLhbnnA/5qJj7x3dNDCRx/PJxu774LlH8lCOlB4hEfKg==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "dependencies": { - "napi-postinstall": "^0.3.0" - }, - "funding": { - "url": "https://opencollective.com/unrs-resolver" - }, - "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" - } - }, - "node_modules/until-async": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/until-async/-/until-async-3.0.2.tgz", - "integrity": "sha512-IiSk4HlzAMqTUseHHe3VhIGyuFmN90zMTpD3Z3y8jeQbzLIq500MVM7Jq2vUAnTKAFPJrqwkzr6PoTcPhGcOiw==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/kettanaito" - } - }, - "node_modules/update-browserslist-db": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", - "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", - "funding": [ - { - "type": "opencollective", - "url": "https://opencollective.com/browserslist" - }, - { - "type": "tidelift", - "url": "https://tidelift.com/funding/github/npm/browserslist" - }, - { - "type": "github", - "url": "https://github.com/sponsors/ai" - } - ], - "license": "MIT", - "dependencies": { - "escalade": "^3.2.0", - "picocolors": "^1.1.1" - }, - "bin": { - "update-browserslist-db": "cli.js" - }, - "peerDependencies": { - "browserslist": ">= 4.21.0" - } - }, - "node_modules/uri-js": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", - "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "punycode": "^2.1.0" - } - }, - "node_modules/use-intl": { - "version": "4.12.0", - "resolved": "https://registry.npmjs.org/use-intl/-/use-intl-4.12.0.tgz", - "integrity": "sha512-r+qVb7UI1+kiOhjYsmsNUCY+jrnjVopwGeFlmMyQj4YInlwZzgMeMSv9n8MqnWWy77HL5BVM8K2WgX50SbtcpA==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/amannn" - } - ], - "license": "MIT", - "dependencies": { - "@formatjs/fast-memoize": "^3.1.0", - "@schummar/icu-type-parser": "1.21.5", - "icu-minify": "^4.12.0", - "intl-messageformat": "^11.1.0" - }, - "peerDependencies": { - "react": "^17.0.0 || ^18.0.0 || >=19.0.0-rc <19.0.0 || ^19.0.0" - } - }, - "node_modules/use-sync-external-store": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz", - "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==", - "license": "MIT", - "peerDependencies": { - "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" - } - }, - "node_modules/util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "license": "MIT" - }, - "node_modules/validate-npm-package-name": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/validate-npm-package-name/-/validate-npm-package-name-7.0.2.tgz", - "integrity": "sha512-hVDIBwsRruT73PbK7uP5ebUt+ezEtCmzZz3F59BSr2F6OVFnJ/6h8liuvdLrQ88Xmnk6/+xGGuq+pG9WwTuy3A==", - "license": "ISC", - "engines": { - "node": "^20.17.0 || >=22.9.0" - } - }, - "node_modules/vary": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", - "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/victory-vendor": { - "version": "37.3.6", - "resolved": "https://registry.npmjs.org/victory-vendor/-/victory-vendor-37.3.6.tgz", - "integrity": "sha512-SbPDPdDBYp+5MJHhBCAyI7wKM3d5ivekigc2Dk2s7pgbZ9wIgIBYGVw4zGHBml/qTFbexrofXW6Gu4noGxrOwQ==", - "license": "MIT AND ISC", - "dependencies": { - "@types/d3-array": "^3.0.3", - "@types/d3-ease": "^3.0.0", - "@types/d3-interpolate": "^3.0.1", - "@types/d3-scale": "^4.0.2", - "@types/d3-shape": "^3.1.0", - "@types/d3-time": "^3.0.0", - "@types/d3-timer": "^3.0.0", - "d3-array": "^3.1.6", - "d3-ease": "^3.0.1", - "d3-interpolate": "^3.0.1", - "d3-scale": "^4.0.2", - "d3-shape": "^3.1.0", - "d3-time": "^3.0.0", - "d3-timer": "^3.0.1" - } - }, - "node_modules/web-streams-polyfill": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz", - "integrity": "sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==", - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/webdriver-bidi-protocol": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/webdriver-bidi-protocol/-/webdriver-bidi-protocol-0.4.1.tgz", - "integrity": "sha512-ARrjNjtWRRs2w4Tk7nqrf2gBI0QXWuOmMCx2hU+1jUt6d00MjMxURrhxhGbrsoiZKJrhTSTzbIrc554iKI10qw==", - "license": "Apache-2.0" - }, - "node_modules/which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "license": "ISC", - "dependencies": { - "isexe": "^2.0.0" - }, - "bin": { - "node-which": "bin/node-which" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/which-boxed-primitive": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", - "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-bigint": "^1.1.0", - "is-boolean-object": "^1.2.1", - "is-number-object": "^1.1.1", - "is-string": "^1.1.1", - "is-symbol": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-builtin-type": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", - "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "function.prototype.name": "^1.1.6", - "has-tostringtag": "^1.0.2", - "is-async-function": "^2.0.0", - "is-date-object": "^1.1.0", - "is-finalizationregistry": "^1.1.0", - "is-generator-function": "^1.0.10", - "is-regex": "^1.2.1", - "is-weakref": "^1.0.2", - "isarray": "^2.0.5", - "which-boxed-primitive": "^1.1.0", - "which-collection": "^1.0.2", - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-collection": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", - "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-map": "^2.0.3", - "is-set": "^2.0.3", - "is-weakmap": "^2.0.2", - "is-weakset": "^2.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-typed-array": { - "version": "1.1.20", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.20.tgz", - "integrity": "sha512-LYfpUkmqwl0h9A2HL09Mms427Q1RZWuOHsukfVcKRq9q95iQxdw0ix1JQrqbcDR9PH1QDwf5Qo8OZb5lksZ8Xg==", - "dev": true, - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/word-wrap": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", - "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/wrap-ansi?sponsor=1" - } - }, - "node_modules/wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", - "license": "ISC" - }, - "node_modules/ws": { - "version": "8.20.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.20.1.tgz", - "integrity": "sha512-It4dO0K5v//JtTXuPkfEOaI3uUN87iYPnqo/ZzqCoG3g8uhA66QUMs/SrM0YK7/NAu+r4LMh/9dq2A7k+rHs+w==", - "license": "MIT", - "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 - } - } - }, - "node_modules/wsl-utils": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/wsl-utils/-/wsl-utils-0.3.1.tgz", - "integrity": "sha512-g/eziiSUNBSsdDJtCLB8bdYEUMj4jR7AGeUo96p/3dTafgjHhpF4RiCFPiRILwjQoDXx5MqkBr4fwWtR3Ky4Wg==", - "license": "MIT", - "dependencies": { - "is-wsl": "^3.1.0", - "powershell-utils": "^0.1.0" - }, - "engines": { - "node": ">=20" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/y18n": { - "version": "5.0.8", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", - "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", - "license": "ISC", - "engines": { - "node": ">=10" - } - }, - "node_modules/yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "license": "ISC" - }, - "node_modules/yargs": { - "version": "17.7.2", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", - "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", - "license": "MIT", - "dependencies": { - "cliui": "^8.0.1", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.3", - "y18n": "^5.0.5", - "yargs-parser": "^21.1.1" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/yargs-parser": { - "version": "21.1.1", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", - "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", - "license": "ISC", - "engines": { - "node": ">=12" - } - }, - "node_modules/yauzl": { - "version": "2.10.0", - "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.10.0.tgz", - "integrity": "sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g==", - "license": "MIT", - "dependencies": { - "buffer-crc32": "~0.2.3", - "fd-slicer": "~1.1.0" - } - }, - "node_modules/yocto-queue": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", - "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/yocto-spinner": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/yocto-spinner/-/yocto-spinner-1.2.0.tgz", - "integrity": "sha512-Yw0hUB6UA3o4YUgKy3oSe9a4cxoaZ9sBfYDw+JSxo6Id0KoJGoxzPA24qqUXYKBWABs/zDSGTz9kww7t3F0XGw==", - "license": "MIT", - "dependencies": { - "yoctocolors": "^2.1.1" - }, - "engines": { - "node": ">=18.19" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/yoctocolors": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/yoctocolors/-/yoctocolors-2.1.2.tgz", - "integrity": "sha512-CzhO+pFNo8ajLM2d2IW/R93ipy99LWjtwblvC1RsoSUMZgyLbYFr221TnSNT7GjGdYui6P459mw9JH/g/zW2ug==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/zod": { - "version": "4.4.3", - "resolved": "https://registry.npmjs.org/zod/-/zod-4.4.3.tgz", - "integrity": "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/colinhacks" - } - }, - "node_modules/zod-to-json-schema": { - "version": "3.25.2", - "resolved": "https://registry.npmjs.org/zod-to-json-schema/-/zod-to-json-schema-3.25.2.tgz", - "integrity": "sha512-O/PgfnpT1xKSDeQYSCfRI5Gy3hPf91mKVDuYLUHZJMiDFptvP41MSnWofm8dnCm0256ZNfZIM7DSzuSMAFnjHA==", - "license": "ISC", - "peerDependencies": { - "zod": "^3.25.28 || ^4" - } - }, - "node_modules/zod-validation-error": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz", - "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=18.0.0" - }, - "peerDependencies": { - "zod": "^3.25.0 || ^4.0.0" - } - } - } -} diff --git a/package.json b/package.json index 08ac3d9..cb32701 100644 --- a/package.json +++ b/package.json @@ -2,36 +2,39 @@ "name": "spent", "version": "0.1.0", "private": true, + "packageManager": "bun@1.3.12", "scripts": { - "dev": "next dev -H 127.0.0.1 -p 3000", + "dev": "next dev -p 2412", "build": "next build", - "start": "next start -H 127.0.0.1 -p 41234", - "lint": "eslint", - "security:audit": "npm audit --omit=dev", - "security:outdated": "npm outdated", - "docs:screenshots": "node scripts/capture-docs-screenshots.mjs", - "setup": "node scripts/setup.mjs", - "uninstall": "node scripts/uninstall.mjs", - "service:install": "node scripts/service/install.mjs install", - "service:uninstall": "node scripts/service/install.mjs uninstall", - "service:start": "node scripts/service/install.mjs start", - "service:stop": "node scripts/service/install.mjs stop", - "service:reload": "npm run build && node scripts/service/install.mjs stop && node scripts/service/install.mjs start", - "service:status": "node scripts/service/install.mjs status", - "service:logs": "node scripts/service/install.mjs logs", - "service:open": "node scripts/service/install.mjs open", - "menubar:build": "bash menubar/mac/build.sh", - "menubar:install:mac": "bash menubar/mac/build.sh && cp -R menubar/mac/build/Spent.app ~/Applications/", - "menubar:build:windows": "powershell -ExecutionPolicy Bypass -File menubar/windows/build.ps1", - "menubar:install:windows": "powershell -NoProfile -ExecutionPolicy Bypass -Command \"& './menubar/windows/build.ps1'; if ($LASTEXITCODE -ne 0) { exit $LASTEXITCODE }; $d = Join-Path $env:LOCALAPPDATA 'Programs\\Spent'; New-Item -ItemType Directory -Force -Path $d | Out-Null; Copy-Item -Force 'menubar/windows/build/Spent.exe' (Join-Path $d 'Spent.exe')\"" + "start": "next start -H 127.0.0.1 -p 2412", + "lint": "eslint --max-warnings=0", + "lint:changed": "bun scripts/lint-changed.mjs", + "db:pull": "drizzle-kit introspect", + "format": "biome format --write .", + "format:check": "biome ci .", + "i18n:check": "bun scripts/check-i18n.mjs", + "knip": "knip --no-progress", + "react:doctor": "bunx --bun react-compiler-healthcheck@latest", + "typecheck": "tsc --noEmit", + "security": "bun audit --production", + "security:outdated": "bun outdated", + "test": "bun test --conditions react-server", + "ci": "bun run format:check && bun run lint:changed && bun run typecheck && bun run i18n:check && bun run knip && bun run react:doctor && bun run security && bun run test" }, "dependencies": { + "@ai-sdk/anthropic": "^3.0.79", + "@ai-sdk/google": "^3.0.80", + "@ai-sdk/react": "^3.0.193", "@anthropic-ai/sdk": "0.95.2", "@base-ui/react": "^1.4.1", + "@google/genai": "^2.6.0", "@tanstack/react-query": "^5.100.10", + "ai": "^6.0.191", + "ai-sdk-ollama": "^3.8.4", "better-sqlite3": "12.10.0", "class-variance-authority": "^0.7.1", "clsx": "^2.1.1", + "drizzle-orm": "^0.45.2", "framer-motion": "^12.38.0", "israeli-bank-scrapers": "6.7.4", "lucide-react": "^1.14.0", @@ -40,22 +43,45 @@ "next-themes": "^0.4.6", "react": "19.2.4", "react-dom": "19.2.4", + "react-markdown": "^10.1.0", "recharts": "^3.8.1", + "remark-gfm": "^4.0.1", "server-only": "^0.0.1", - "shadcn": "^4.7.0", "sonner": "^2.0.7", "tailwind-merge": "^3.6.0", - "tw-animate-css": "^1.4.0" + "tw-animate-css": "^1.4.0", + "zod": "^4.4.3" }, "devDependencies": { + "@biomejs/biome": "^2.2.0", "@tailwindcss/postcss": "^4", "@types/better-sqlite3": "^7.6.13", + "@types/bun": "latest", "@types/node": "^20", "@types/react": "^19", "@types/react-dom": "^19", + "drizzle-kit": "^0.31.10", "eslint": "^9", "eslint-config-next": "16.2.6", + "knip": "^5", + "postcss": "^8.5.15", + "puppeteer": "^24", + "shadcn": "^4.7.0", "tailwindcss": "^4", "typescript": "^5" + }, + "trustedDependencies": [ + "@google/genai", + "@parcel/watcher", + "@swc/core", + "better-sqlite3", + "msw", + "protobufjs", + "puppeteer", + "sharp", + "unrs-resolver" + ], + "overrides": { + "postcss": "8.5.15" } } diff --git a/public/file.svg b/public/file.svg deleted file mode 100644 index 004145c..0000000 --- a/public/file.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/globe.svg b/public/globe.svg deleted file mode 100644 index 567f17b..0000000 --- a/public/globe.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/icon-192.png b/public/icon-192.png new file mode 100644 index 0000000..b759318 Binary files /dev/null and b/public/icon-192.png differ diff --git a/public/icon-512.png b/public/icon-512.png new file mode 100644 index 0000000..2428ae2 Binary files /dev/null and b/public/icon-512.png differ diff --git a/public/next.svg b/public/next.svg deleted file mode 100644 index 5174b28..0000000 --- a/public/next.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/vercel.svg b/public/vercel.svg deleted file mode 100644 index 7705396..0000000 --- a/public/vercel.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/public/window.svg b/public/window.svg deleted file mode 100644 index b2b2a44..0000000 --- a/public/window.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/scripts/capture-docs-screenshots.mjs b/scripts/capture-docs-screenshots.mjs deleted file mode 100644 index d26be63..0000000 --- a/scripts/capture-docs-screenshots.mjs +++ /dev/null @@ -1,180 +0,0 @@ -// scripts/capture-docs-screenshots.mjs -// -// Boot Next.js against a tmp data dir seeded with fake data, -// capture docs screenshots into website/src/assets/screenshots/. -// -// Usage: -// npm run docs:screenshots -// -// Requires a built Next app (run `npm run build` first) OR uses dev mode. - -import { spawn } from 'node:child_process'; -import fs from 'node:fs'; -import os from 'node:os'; -import path from 'node:path'; -import { fileURLToPath } from 'node:url'; -import Database from 'better-sqlite3'; -import puppeteer from 'puppeteer'; -import { banks, transactions, budgets } from './docs-seed/fake-data.mjs'; - -const HERE = path.dirname(fileURLToPath(import.meta.url)); -const REPO_ROOT = path.resolve(HERE, '..'); -const OUT_DIR = path.join(REPO_ROOT, 'website/src/assets/screenshots'); - -const VIEWPORT = { width: 1600, height: 1100, deviceScaleFactor: 2 }; -const PORT = 4399; // separate from prod 41234 and dev 3000 - -const SCREENS = [ - { name: 'home-light.png', path: '/', theme: 'light' }, - { name: 'dashboard-light.png', path: '/budget', theme: 'light' }, - { name: 'dashboard-dark.png', path: '/', theme: 'dark' }, - { name: 'transactions-light.png', path: '/transactions', theme: 'light' }, - { name: 'settings-banks-light.png', path: '/settings/bank', theme: 'light' }, - { name: 'settings-ai-light.png', path: '/settings/ai', theme: 'light' }, - { name: 'settings-categories-light.png', path: '/settings/categories', theme: 'light' }, - { - name: 'setup-bank-light.png', - path: '/setup', - theme: 'light', - }, -]; - -function seedDb(dbDir) { - fs.mkdirSync(dbDir, { recursive: true }); - const dbPath = path.join(dbDir, 'spent.db'); - const db = new Database(dbPath); - db.pragma('journal_mode = WAL'); - const migrationPath = path.join(REPO_ROOT, 'src/server/db/migrations/001_initial.sql'); - db.exec(fs.readFileSync(migrationPath, 'utf-8')); - - // Insert banks with dummy encrypted blobs. The dashboard doesn't decrypt - // these to render — it only shows the provider label. - const insertBank = db.prepare( - `INSERT INTO bank_credentials (provider, credentials_encrypted, iv, auth_tag) - VALUES (?, ?, ?, ?)`, - ); - for (const b of banks) { - insertBank.run(b.provider, Buffer.from('demo'), Buffer.from('demo'), Buffer.from('demo')); - } - - // Insert transactions. Schema in src/server/db/migrations/001_initial.sql. - // If column list here drifts from schema, update both — the failure will be - // immediate and noisy. - const insertTx = db.prepare( - `INSERT INTO transactions - (account_number, date, processed_date, original_amount, original_currency, - charged_amount, charged_currency, description, type, status, category_id, provider, hash) - VALUES (@account, @date, @date, @amount, 'ILS', @amount, 'ILS', @desc, 'normal', 'completed', @cat, @provider, @hash)`, - ); - for (let i = 0; i < transactions.length; i++) { - const t = transactions[i]; - insertTx.run({ - account: '****1234', - date: t.date, - amount: t.amount, - desc: t.merchant, - cat: t.cat, - provider: t.provider, - hash: `demo-${i.toString().padStart(4, '0')}`, - }); - } - - // Budgets table may or may not exist depending on schema; soft-fail. - try { - const insertBudget = db.prepare( - `INSERT INTO budgets (category_id, monthly_target) VALUES (?, ?)`, - ); - for (const b of budgets) insertBudget.run(b.category_id, b.monthly_target); - } catch { - console.log(' (skipping budgets table — not in schema)'); - } - - db.close(); - return dbPath; -} - -function startServer(dataDir) { - const env = { ...process.env, SPENT_DATA_DIR: dataDir, PORT: String(PORT) }; - const child = spawn('npm', ['run', 'dev', '--', '-p', String(PORT)], { - cwd: REPO_ROOT, env, stdio: ['ignore', 'pipe', 'pipe'], - }); - return child; -} - -async function waitForServer() { - const url = `http://127.0.0.1:${PORT}/api/health`; - const deadline = Date.now() + 60_000; - while (Date.now() < deadline) { - try { - const r = await fetch(url, { signal: AbortSignal.timeout(2000) }); - if (r.ok) return true; - } catch {} - await new Promise(r => setTimeout(r, 500)); - } - return false; -} - -async function setTheme(page, theme) { - await page.evaluate((t) => { - const html = document.documentElement; - html.classList.remove('light', 'dark'); - html.classList.add(t); - try { localStorage.setItem('theme', t); } catch {} - }, theme); -} - -(async () => { - fs.mkdirSync(OUT_DIR, { recursive: true }); - - const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'spent-docs-')); - console.log(`Tmp data dir: ${tmpDir}`); - - console.log('Seeding fake data...'); - seedDb(tmpDir); - console.log(' ✓ seeded'); - - console.log('Starting Next.js (this can take 10-20s)...'); - const server = startServer(tmpDir); - server.stdout.on('data', (d) => process.stdout.write(` [next] ${d}`)); - server.stderr.on('data', (d) => process.stderr.write(` [next!] ${d}`)); - - const ready = await waitForServer(); - if (!ready) { - server.kill('SIGTERM'); - fs.rmSync(tmpDir, { recursive: true, force: true }); - throw new Error('Next.js did not become ready in 60s'); - } - console.log(' ✓ Next.js is ready'); - - const browser = await puppeteer.launch({ - headless: 'new', - defaultViewport: VIEWPORT, - args: ['--no-sandbox', '--disable-setuid-sandbox'], - }); - - try { - const page = await browser.newPage(); - await page.setViewport(VIEWPORT); - await page.goto(`http://127.0.0.1:${PORT}/`, { waitUntil: 'networkidle2', timeout: 30000 }); - - for (const screen of SCREENS) { - const dest = path.join(OUT_DIR, screen.name); - console.log(`Capturing ${screen.name} ← ${screen.path} (${screen.theme})`); - await setTheme(page, screen.theme); - await page.goto(`http://127.0.0.1:${PORT}${screen.path}`, { waitUntil: 'networkidle2', timeout: 30000 }); - await setTheme(page, screen.theme); - await new Promise(r => setTimeout(r, 1200)); - await page.screenshot({ path: dest, fullPage: false }); - console.log(' ✓ saved'); - } - } finally { - await browser.close(); - server.kill('SIGTERM'); - await new Promise(r => setTimeout(r, 1000)); - fs.rmSync(tmpDir, { recursive: true, force: true }); - console.log('Cleaned up tmp dir'); - } -})().catch((err) => { - console.error('Failed:', err); - process.exit(1); -}); diff --git a/scripts/capture-screenshots.mjs b/scripts/capture-screenshots.mjs index f518c69..060388d 100644 --- a/scripts/capture-screenshots.mjs +++ b/scripts/capture-screenshots.mjs @@ -8,122 +8,122 @@ * node scripts/capture-screenshots.mjs */ -import puppeteer from 'puppeteer'; -import path from 'node:path'; -import fs from 'node:fs/promises'; +import fs from "node:fs/promises"; +import path from "node:path"; +import puppeteer from "puppeteer"; -const APP_URL = 'http://127.0.0.1:3000'; +const APP_URL = "http://127.0.0.1:3000"; const OUT_DIR = path.resolve( - new URL('.', import.meta.url).pathname, - '../website/src/assets/screenshots' + new URL(".", import.meta.url).pathname, + "../website/src/assets/screenshots", ); const VIEWPORT = { width: 1600, height: 1100, deviceScaleFactor: 2 }; const SCREENS = [ - { - name: 'home-light.png', - path: '/', - theme: 'light', - }, - { - name: 'dashboard-light.png', - // Use /budget as the secondary dashboard view for peek-inside side - path: '/budget', - theme: 'light', - }, - { - name: 'dashboard-dark.png', - // Home page in dark for the dark mode strip — richer than budget alone - path: '/', - theme: 'dark', - }, - { - name: 'transactions-light.png', - path: '/transactions', - theme: 'light', - }, - { - name: 'setup-bank-light.png', - // Real connected-banks settings view, opening the Add bank picker so the - // shot shows BOTH connected banks AND the full list of supported ones. - path: '/settings/bank', - theme: 'light', - afterLoad: async (page) => { - await page.evaluate(() => { - // Close any open dropdowns by clicking body - document.body.click(); - }); - await new Promise((r) => setTimeout(r, 300)); - await page.evaluate(() => { - const btns = Array.from(document.querySelectorAll('button, a')); - const addBtn = btns.find((b) => /add bank/i.test(b.textContent || '')); - if (addBtn) (addBtn).click(); - }); - await new Promise((r) => setTimeout(r, 600)); - }, - }, + { + name: "home-light.png", + path: "/", + theme: "light", + }, + { + name: "dashboard-light.png", + // Use /budget as the secondary dashboard view for peek-inside side + path: "/budget", + theme: "light", + }, + { + name: "dashboard-dark.png", + // Home page in dark for the dark mode strip — richer than budget alone + path: "/", + theme: "dark", + }, + { + name: "transactions-light.png", + path: "/transactions", + theme: "light", + }, + { + name: "setup-bank-light.png", + // Real connected-banks settings view, opening the Add bank picker so the + // shot shows BOTH connected banks AND the full list of supported ones. + path: "/settings/bank", + theme: "light", + afterLoad: async (page) => { + await page.evaluate(() => { + // Close any open dropdowns by clicking body + document.body.click(); + }); + await new Promise((r) => setTimeout(r, 300)); + await page.evaluate(() => { + const btns = Array.from(document.querySelectorAll("button, a")); + const addBtn = btns.find((b) => /add bank/i.test(b.textContent || "")); + if (addBtn) addBtn.click(); + }); + await new Promise((r) => setTimeout(r, 600)); + }, + }, ]; const setTheme = async (page, theme) => { - await page.evaluate((t) => { - const html = document.documentElement; - html.classList.remove('light', 'dark'); - html.classList.add(t); - try { - localStorage.setItem('theme', t); - } catch {} - }, theme); + await page.evaluate((t) => { + const html = document.documentElement; + html.classList.remove("light", "dark"); + html.classList.add(t); + try { + localStorage.setItem("theme", t); + } catch {} + }, theme); }; (async () => { - await fs.mkdir(OUT_DIR, { recursive: true }); - - const browser = await puppeteer.launch({ - headless: 'new', - defaultViewport: VIEWPORT, - args: ['--no-sandbox', '--disable-setuid-sandbox'], - }); - - try { - const page = await browser.newPage(); - await page.setViewport(VIEWPORT); - - // Prime localStorage on the right origin - await page.goto(APP_URL, { waitUntil: 'networkidle2', timeout: 15000 }); - - for (const screen of SCREENS) { - const dest = path.join(OUT_DIR, screen.name); - console.log(`Capturing ${screen.name} ← ${screen.path} (${screen.theme}) → ${dest}`); - - await setTheme(page, screen.theme); - await page.goto(`${APP_URL}${screen.path}`, { - waitUntil: 'networkidle2', - timeout: 20000, - }); - // Re-apply theme after navigation (in case it was reset) - await setTheme(page, screen.theme); - - if (screen.injectCss) { - await page.addStyleTag({ content: screen.injectCss }); - } - - // Give animations a moment to settle - await new Promise((r) => setTimeout(r, 1200)); - - if (typeof screen.afterLoad === 'function') { - await screen.afterLoad(page); - } - - await page.screenshot({ path: dest, fullPage: false }); - console.log(` ✓ saved`); - } - } finally { - await browser.close(); - } - - console.log('\nDone. Files in:', OUT_DIR); + await fs.mkdir(OUT_DIR, { recursive: true }); + + const browser = await puppeteer.launch({ + headless: "new", + defaultViewport: VIEWPORT, + args: ["--no-sandbox", "--disable-setuid-sandbox"], + }); + + try { + const page = await browser.newPage(); + await page.setViewport(VIEWPORT); + + // Prime localStorage on the right origin + await page.goto(APP_URL, { waitUntil: "networkidle2", timeout: 15000 }); + + for (const screen of SCREENS) { + const dest = path.join(OUT_DIR, screen.name); + console.log(`Capturing ${screen.name} ← ${screen.path} (${screen.theme}) → ${dest}`); + + await setTheme(page, screen.theme); + await page.goto(`${APP_URL}${screen.path}`, { + waitUntil: "networkidle2", + timeout: 20000, + }); + // Re-apply theme after navigation (in case it was reset) + await setTheme(page, screen.theme); + + if (screen.injectCss) { + await page.addStyleTag({ content: screen.injectCss }); + } + + // Give animations a moment to settle + await new Promise((r) => setTimeout(r, 1200)); + + if (typeof screen.afterLoad === "function") { + await screen.afterLoad(page); + } + + await page.screenshot({ path: dest, fullPage: false }); + console.log(` ✓ saved`); + } + } finally { + await browser.close(); + } + + console.log("\nDone. Files in:", OUT_DIR); })().catch((err) => { - console.error('Failed:', err); - process.exit(1); + console.error("Failed:", err); + process.exit(1); }); diff --git a/scripts/check-i18n.mjs b/scripts/check-i18n.mjs new file mode 100644 index 0000000..8fc6db3 --- /dev/null +++ b/scripts/check-i18n.mjs @@ -0,0 +1,65 @@ +#!/usr/bin/env bun +// Runs @lingual/i18n-check (recommended by next-intl docs) against +// src/i18n/messages. The --ignore list covers (a) namespaces that the static +// parser cannot trace because keys are accessed dynamically and (b) a baseline +// set of pre-existing orphan keys to lock current state without churning the +// CI-enable PR. Treat additions to baseline.* as tech debt — every key here +// is something to delete once setup wizard and settings get cleaner i18n. + +import { spawn } from "node:child_process"; + +// Legitimate dynamic-access namespaces. These keys exist for a reason; the +// linter just can't see the access pattern. +const dynamicNamespaces = ["banks.*", "categoriesSeeded.*", "settings.sidebar.*", "nav.*"]; + +// Pre-existing orphan keys, grandfathered to land CI without a huge cleanup +// diff. New code MUST NOT add to this list — fix the orphan or use the key. +const baseline = [ + "setup.*", + "common.*", + "transactions.allCategories", + "transactions.allAccounts", + "settings.bank.transactionsCount", + "settings.bank.justNow", + "settings.bank.minutesAgo", + "settings.bank.hoursAgo", + "settings.bank.daysAgo", + "settings.categories.title", + "settings.categories.description", + "settings.categories.tabExpense", + "settings.categories.tabIncome", + "settings.categories.searchPlaceholder", + "settings.categories.newGroupButton", + "settings.categories.newGroupDialogTitle", + "settings.categories.newGroupName", + "settings.categories.newGroupNamePlaceholder", + "settings.categories.newGroupKind", + "settings.categories.createButton", + "settings.categories.createdToast", + "settings.categories.createGroupFailed", + "settings.categories.noMatching", + "settings.categories.ungrouped", + "settings.categories.editGroup", + "settings.categories.spentLabel", + "settings.categories.tracking", + "settings.categories.noBudget", +]; + +const args = [ + "--bun", + "@lingual/i18n-check@latest", + "--format", + "next-intl", + "--source", + "en", + "--locales", + "src/i18n/messages", + "--unused", + "src", + "--ignore", + ...dynamicNamespaces, + ...baseline, +]; + +const child = spawn("bunx", args, { stdio: "inherit" }); +child.on("exit", (code) => process.exit(code ?? 1)); diff --git a/scripts/docs-seed/fake-data.mjs b/scripts/docs-seed/fake-data.mjs deleted file mode 100644 index 876de16..0000000 --- a/scripts/docs-seed/fake-data.mjs +++ /dev/null @@ -1,86 +0,0 @@ -// scripts/docs-seed/fake-data.mjs -// -// Curated fake dataset for docs screenshots. Used only by -// scripts/capture-docs-screenshots.mjs. No real user data. - -const today = new Date(); - -function daysAgo(n) { - const d = new Date(today); - d.setDate(d.getDate() - n); - return d.toISOString().slice(0, 10); -} - -// Reference: schema in src/server/db/migrations/001_initial.sql -export const banks = [ - { provider: 'isracard', label: 'Isracard' }, - { provider: 'hapoalim', label: 'Bank Hapoalim' }, - { provider: 'max', label: 'Max' }, -]; - -// 16 seeded categories from 001_initial.sql, by 1-based id. -export const categoryIds = { - Groceries: 1, - Restaurants: 2, - Transport: 3, - Shopping: 4, - Entertainment: 5, - Health: 6, - Education: 7, - BillsUtilities: 8, - Subscriptions: 9, - Travel: 10, - CashATM: 11, - Transfers: 12, - Insurance: 13, - Home: 14, - PersonalCare: 15, -}; - -export const transactions = [ - // This-month coffees - { date: daysAgo(0), merchant: 'Aroma · Rothschild', amount: -14.5, cat: categoryIds.Restaurants, provider: 'isracard' }, - { date: daysAgo(1), merchant: 'Cofix · Allenby', amount: -7.0, cat: categoryIds.Restaurants, provider: 'isracard' }, - { date: daysAgo(2), merchant: 'Aroma · Sarona', amount: -14.5, cat: categoryIds.Restaurants, provider: 'isracard' }, - // Groceries - { date: daysAgo(3), merchant: 'Shufersal Deal', amount: -284.2, cat: categoryIds.Groceries, provider: 'hapoalim' }, - { date: daysAgo(8), merchant: 'Tiv Taam', amount: -198.4, cat: categoryIds.Groceries, provider: 'isracard' }, - { date: daysAgo(15), merchant: 'Shufersal Deal', amount: -312.8, cat: categoryIds.Groceries, provider: 'hapoalim' }, - // Transport - { date: daysAgo(4), merchant: 'Rav-Kav', amount: -124.0, cat: categoryIds.Transport, provider: 'isracard' }, - { date: daysAgo(11), merchant: 'Gett', amount: -42.5, cat: categoryIds.Transport, provider: 'max' }, - { date: daysAgo(18), merchant: 'Rav-Kav', amount: -124.0, cat: categoryIds.Transport, provider: 'isracard' }, - // Bills - { date: daysAgo(6), merchant: 'Cellcom', amount: -89.0, cat: categoryIds.BillsUtilities, provider: 'hapoalim' }, - { date: daysAgo(9), merchant: 'Bezeq International', amount: -119.0, cat: categoryIds.BillsUtilities, provider: 'hapoalim' }, - { date: daysAgo(14), merchant: 'Hot · Pakage', amount: -209.0, cat: categoryIds.BillsUtilities, provider: 'hapoalim' }, - // Subscriptions - { date: daysAgo(7), merchant: 'Netflix', amount: -49.9, cat: categoryIds.Subscriptions, provider: 'max' }, - { date: daysAgo(7), merchant: 'Spotify', amount: -19.9, cat: categoryIds.Subscriptions, provider: 'max' }, - { date: daysAgo(7), merchant: 'iCloud+', amount: -8.9, cat: categoryIds.Subscriptions, provider: 'max' }, - // Food delivery - { date: daysAgo(2), merchant: 'Wolt · Pizza Domino', amount: -68.0, cat: categoryIds.Restaurants, provider: 'isracard' }, - { date: daysAgo(5), merchant: 'Tenten', amount: -120.0, cat: categoryIds.Restaurants, provider: 'isracard' }, - // Health - { date: daysAgo(10), merchant: 'Super-Pharm', amount: -94.7, cat: categoryIds.Health, provider: 'hapoalim' }, - { date: daysAgo(20), merchant: 'Clalit Pharmacy', amount: -34.0, cat: categoryIds.Health, provider: 'hapoalim' }, - // Last month - { date: daysAgo(32), merchant: 'Shufersal Deal', amount: -298.4, cat: categoryIds.Groceries, provider: 'hapoalim' }, - { date: daysAgo(34), merchant: 'Rav-Kav', amount: -124.0, cat: categoryIds.Transport, provider: 'isracard' }, - { date: daysAgo(37), merchant: 'Cellcom', amount: -89.0, cat: categoryIds.BillsUtilities, provider: 'hapoalim' }, - { date: daysAgo(40), merchant: 'Netflix', amount: -49.9, cat: categoryIds.Subscriptions, provider: 'max' }, - { date: daysAgo(45), merchant: 'Aroma · Dizengoff', amount: -14.5, cat: categoryIds.Restaurants, provider: 'isracard' }, - { date: daysAgo(48), merchant: 'Wolt · Sushi Bazaar', amount: -82.0, cat: categoryIds.Restaurants, provider: 'isracard' }, - // Income (positive amount) - { date: daysAgo(28), merchant: 'Acme Industries · Payroll', amount: 18500.0, cat: categoryIds.Transfers, provider: 'hapoalim' }, - { date: daysAgo(58), merchant: 'Acme Industries · Payroll', amount: 18500.0, cat: categoryIds.Transfers, provider: 'hapoalim' }, -]; - -// Optional: monthly budget targets per parent category. -export const budgets = [ - { category_id: categoryIds.Groceries, monthly_target: 1200 }, - { category_id: categoryIds.Restaurants, monthly_target: 800 }, - { category_id: categoryIds.Transport, monthly_target: 500 }, - { category_id: categoryIds.BillsUtilities, monthly_target: 600 }, - { category_id: categoryIds.Subscriptions, monthly_target: 150 }, -]; diff --git a/scripts/fetch-bank-logos.mjs b/scripts/fetch-bank-logos.mjs index 847b943..5f8f005 100644 --- a/scripts/fetch-bank-logos.mjs +++ b/scripts/fetch-bank-logos.mjs @@ -12,77 +12,73 @@ * node scripts/fetch-bank-logos.mjs */ -import fs from 'node:fs/promises'; -import path from 'node:path'; +import fs from "node:fs/promises"; +import path from "node:path"; -const OUT_DIR = path.resolve( - new URL('.', import.meta.url).pathname, - '../website/src/assets/banks' -); +const OUT_DIR = path.resolve(new URL(".", import.meta.url).pathname, "../website/src/assets/banks"); // Curated Wikimedia URLs (PNG thumbnails of SVG sources). // 250px is enough at the rendered tile size (~80–96px). const LOGOS = { - hapoalim: - 'https://upload.wikimedia.org/wikipedia/commons/thumb/6/66/Bank_happoalim_2018_logo.svg/250px-Bank_happoalim_2018_logo.svg.png', - leumi: - 'https://upload.wikimedia.org/wikipedia/en/thumb/f/f8/Bank_Leumi_logo.svg/250px-Bank_Leumi_logo.svg.png', - mizrahi: - 'https://upload.wikimedia.org/wikipedia/commons/thumb/6/62/%D7%9C%D7%95%D7%92%D7%95_%D7%A9%D7%9C_%D7%91%D7%A0%D7%A7_%D7%9E%D7%96%D7%A8%D7%97%D7%99-%D7%98%D7%A4%D7%97%D7%95%D7%AA.svg/250px-%D7%9C%D7%95%D7%92%D7%95_%D7%A9%D7%9C_%D7%91%D7%A0%D7%A7_%D7%9E%D7%96%D7%A8%D7%97%D7%99-%D7%98%D7%A4%D7%97%D7%95%D7%AA.svg.png', - discount: - 'https://upload.wikimedia.org/wikipedia/commons/thumb/6/6f/Discount_Bank%2C_Ltd_logo.svg/250px-Discount_Bank%2C_Ltd_logo.svg.png', - isracard: - 'https://upload.wikimedia.org/wikipedia/en/thumb/a/ab/Isracard_2023_Logo.svg/250px-Isracard_2023_Logo.svg.png', - fibi: - 'https://upload.wikimedia.org/wikipedia/en/thumb/9/96/First_International_Bank_of_Israel_logo.svg/250px-First_International_Bank_of_Israel_logo.svg.png', - union: - 'https://upload.wikimedia.org/wikipedia/en/thumb/d/d6/Bank_Igud_logo.svg/250px-Bank_Igud_logo.svg.png', + hapoalim: + "https://upload.wikimedia.org/wikipedia/commons/thumb/6/66/Bank_happoalim_2018_logo.svg/250px-Bank_happoalim_2018_logo.svg.png", + leumi: + "https://upload.wikimedia.org/wikipedia/en/thumb/f/f8/Bank_Leumi_logo.svg/250px-Bank_Leumi_logo.svg.png", + mizrahi: + "https://upload.wikimedia.org/wikipedia/commons/thumb/6/62/%D7%9C%D7%95%D7%92%D7%95_%D7%A9%D7%9C_%D7%91%D7%A0%D7%A7_%D7%9E%D7%96%D7%A8%D7%97%D7%99-%D7%98%D7%A4%D7%97%D7%95%D7%AA.svg/250px-%D7%9C%D7%95%D7%92%D7%95_%D7%A9%D7%9C_%D7%91%D7%A0%D7%A7_%D7%9E%D7%96%D7%A8%D7%97%D7%99-%D7%98%D7%A4%D7%97%D7%95%D7%AA.svg.png", + discount: + "https://upload.wikimedia.org/wikipedia/commons/thumb/6/6f/Discount_Bank%2C_Ltd_logo.svg/250px-Discount_Bank%2C_Ltd_logo.svg.png", + isracard: + "https://upload.wikimedia.org/wikipedia/en/thumb/a/ab/Isracard_2023_Logo.svg/250px-Isracard_2023_Logo.svg.png", + fibi: "https://upload.wikimedia.org/wikipedia/en/thumb/9/96/First_International_Bank_of_Israel_logo.svg/250px-First_International_Bank_of_Israel_logo.svg.png", + union: + "https://upload.wikimedia.org/wikipedia/en/thumb/d/d6/Bank_Igud_logo.svg/250px-Bank_Igud_logo.svg.png", }; -const userAgent = 'SpentLandingBuild/0.1 (https://github.com/Shaya16/Spent)'; +const userAgent = "SpentLandingBuild/0.1 (https://github.com/alon710/Spent)"; const fetchLogo = async (url) => { - try { - const res = await fetch(url, { - headers: { 'User-Agent': userAgent, Accept: 'image/png,image/*' }, - redirect: 'follow', - signal: AbortSignal.timeout(15000), - }); - if (!res.ok) return { ok: false, status: res.status }; - const ct = res.headers.get('content-type') || ''; - if (!ct.startsWith('image/')) return { ok: false, status: 'wrong-content-type' }; - const buf = Buffer.from(await res.arrayBuffer()); - if (buf.length < 500) return { ok: false, status: 'too-small' }; - return { ok: true, buf }; - } catch (e) { - return { ok: false, status: e.message }; - } + try { + const res = await fetch(url, { + headers: { "User-Agent": userAgent, Accept: "image/png,image/*" }, + redirect: "follow", + signal: AbortSignal.timeout(15000), + }); + if (!res.ok) return { ok: false, status: res.status }; + const ct = res.headers.get("content-type") || ""; + if (!ct.startsWith("image/")) return { ok: false, status: "wrong-content-type" }; + const buf = Buffer.from(await res.arrayBuffer()); + if (buf.length < 500) return { ok: false, status: "too-small" }; + return { ok: true, buf }; + } catch (e) { + return { ok: false, status: e.message }; + } }; (async () => { - await fs.mkdir(OUT_DIR, { recursive: true }); + await fs.mkdir(OUT_DIR, { recursive: true }); - const found = []; - const errors = []; - for (const [id, url] of Object.entries(LOGOS)) { - const result = await fetchLogo(url); - if (result.ok) { - const dest = path.join(OUT_DIR, `${id}.png`); - await fs.writeFile(dest, result.buf); - found.push(id); - console.log(`✓ ${id.padEnd(12)} ${result.buf.length.toString().padStart(7)} bytes`); - } else { - errors.push({ id, status: result.status }); - console.log(`✗ ${id.padEnd(12)} ${result.status}`); - } - } + const found = []; + const errors = []; + for (const [id, url] of Object.entries(LOGOS)) { + const result = await fetchLogo(url); + if (result.ok) { + const dest = path.join(OUT_DIR, `${id}.png`); + await fs.writeFile(dest, result.buf); + found.push(id); + console.log(`✓ ${id.padEnd(12)} ${result.buf.length.toString().padStart(7)} bytes`); + } else { + errors.push({ id, status: result.status }); + console.log(`✗ ${id.padEnd(12)} ${result.status}`); + } + } - console.log(`\nFetched ${found.length}/${Object.keys(LOGOS).length} logos.`); - if (errors.length) { - console.log(`Failed: ${errors.map((e) => `${e.id} (${e.status})`).join(', ')}`); - } - console.log(`\nNote: banks without curated sources fall back to the colored letter-tile.`); + console.log(`\nFetched ${found.length}/${Object.keys(LOGOS).length} logos.`); + if (errors.length) { + console.log(`Failed: ${errors.map((e) => `${e.id} (${e.status})`).join(", ")}`); + } + console.log(`\nNote: banks without curated sources fall back to the colored letter-tile.`); })().catch((err) => { - console.error('Failed:', err); - process.exit(1); + console.error("Failed:", err); + process.exit(1); }); diff --git a/scripts/lint-changed.mjs b/scripts/lint-changed.mjs new file mode 100644 index 0000000..1e2a688 --- /dev/null +++ b/scripts/lint-changed.mjs @@ -0,0 +1,46 @@ +#!/usr/bin/env bun +// Strict ESLint, scoped to the files this branch changes versus the base branch. +// +// The repository carries a known lint backlog in older UI components (react-compiler +// rules: static-components, set-state-in-effect, rules-of-hooks). Rather than block +// every PR on that backlog, we gate only the files a branch actually touches so the +// backlog is paid down incrementally. Run `bun run lint` for the full-tree view. + +import { spawnSync } from "node:child_process"; + +const BASE = process.env.LINT_BASE_REF ?? "origin/main"; + +function git(args) { + const res = spawnSync("git", args, { encoding: "utf8" }); + return res.status === 0 ? res.stdout.trim() : ""; +} + +// Best-effort: make sure the base ref is present (no-op locally if already fetched). +spawnSync("git", ["fetch", "--quiet", "origin", BASE.replace(/^origin\//, "")], { + stdio: "ignore", +}); + +const mergeBase = git(["merge-base", BASE, "HEAD"]); +if (!mergeBase) { + console.log( + `lint:changed: no merge-base with ${BASE}; skipping (run "bun run lint" for full tree)`, + ); + process.exit(0); +} + +const diff = git(["diff", "--name-only", "--diff-filter=ACMR", mergeBase, "HEAD"]); +const files = diff + .split("\n") + .map((f) => f.trim()) + .filter((f) => /\.(ts|tsx|mjs)$/.test(f)); + +if (files.length === 0) { + console.log("lint:changed: no TS/JS files changed versus base; nothing to lint"); + process.exit(0); +} + +console.log(`lint:changed: linting ${files.length} changed file(s)`); +const eslint = spawnSync("eslint", ["--max-warnings=0", ...files], { + stdio: "inherit", +}); +process.exit(eslint.status ?? 1); diff --git a/scripts/oklch-to-hex.mjs b/scripts/oklch-to-hex.mjs index 143d9c5..f40073c 100644 --- a/scripts/oklch-to-hex.mjs +++ b/scripts/oklch-to-hex.mjs @@ -33,22 +33,22 @@ function oklchToHex(L, C, H) { } const categories = [ - ["Groceries", 0.72, 0.09, 145], - ["Restaurants", 0.76, 0.10, 40], - ["Transport", 0.72, 0.09, 230], - ["Shopping", 0.80, 0.09, 80], - ["Entertainment", 0.76, 0.09, 10], - ["Health", 0.74, 0.08, 170], - ["Education", 0.72, 0.09, 275], - ["Bills & Utilities", 0.74, 0.04, 80], - ["Subscriptions", 0.73, 0.09, 295], - ["Travel", 0.74, 0.09, 220], - ["Cash & ATM", 0.82, 0.09, 90], - ["Transfers", 0.74, 0.035, 270], - ["Insurance", 0.76, 0.09, 20], - ["Home", 0.76, 0.09, 75], - ["Personal Care", 0.78, 0.09, 325], - ["Other", 0.74, 0.022, 85], + ["Groceries", 0.72, 0.09, 145], + ["Restaurants", 0.76, 0.1, 40], + ["Transport", 0.72, 0.09, 230], + ["Shopping", 0.8, 0.09, 80], + ["Entertainment", 0.76, 0.09, 10], + ["Health", 0.74, 0.08, 170], + ["Education", 0.72, 0.09, 275], + ["Bills & Utilities", 0.74, 0.04, 80], + ["Subscriptions", 0.73, 0.09, 295], + ["Travel", 0.74, 0.09, 220], + ["Cash & ATM", 0.82, 0.09, 90], + ["Transfers", 0.74, 0.035, 270], + ["Insurance", 0.76, 0.09, 20], + ["Home", 0.76, 0.09, 75], + ["Personal Care", 0.78, 0.09, 325], + ["Other", 0.74, 0.022, 85], ]; for (const [name, L, C, H] of categories) { @@ -57,18 +57,18 @@ for (const [name, L, C, H] of categories) { console.log("\nNew-category palette (for AI proposals):"); const palette = [ - ["light olive", 0.78, 0.09, 130], - ["sandy orange", 0.78, 0.10, 60], - ["light cyan-blue", 0.76, 0.09, 210], - ["bright pink", 0.74, 0.10, 340], - ["medium violet", 0.66, 0.11, 290], - ["jade", 0.76, 0.09, 170], - ["dusty indigo", 0.66, 0.09, 270], - ["medium slate", 0.74, 0.025, 260], - ["mauve", 0.75, 0.09, 310], - ["mint", 0.82, 0.08, 165], - ["sand gold", 0.82, 0.09, 95], - ["sage tan", 0.78, 0.04, 95], + ["light olive", 0.78, 0.09, 130], + ["sandy orange", 0.78, 0.1, 60], + ["light cyan-blue", 0.76, 0.09, 210], + ["bright pink", 0.74, 0.1, 340], + ["medium violet", 0.66, 0.11, 290], + ["jade", 0.76, 0.09, 170], + ["dusty indigo", 0.66, 0.09, 270], + ["medium slate", 0.74, 0.025, 260], + ["mauve", 0.75, 0.09, 310], + ["mint", 0.82, 0.08, 165], + ["sand gold", 0.82, 0.09, 95], + ["sage tan", 0.78, 0.04, 95], ]; for (const [name, L, C, H] of palette) { console.log(`${name.padEnd(20)} oklch(${L} ${C} ${H}) → ${oklchToHex(L, C, H)}`); diff --git a/scripts/service/hosts.mjs b/scripts/service/hosts.mjs deleted file mode 100644 index b8df9d0..0000000 --- a/scripts/service/hosts.mjs +++ /dev/null @@ -1,173 +0,0 @@ -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import { spawnSync } from "node:child_process"; -import { HOST, FRIENDLY_HOST } from "./paths.mjs"; - -const MARKER_START = "# >>> spent (managed) >>>"; -const MARKER_END = "# <<< spent <<<"; -const MANAGED_LINE = `${HOST}\t${FRIENDLY_HOST}`; - -export function hostsFilePath() { - if (process.platform === "win32") { - const sysroot = process.env.SystemRoot ?? "C:\\Windows"; - return path.join(sysroot, "System32", "drivers", "etc", "hosts"); - } - return "/etc/hosts"; -} - -function buildManagedBlock() { - return [ - MARKER_START, - `# Added by Spent (scripts/service). Resolves ${FRIENDLY_HOST} to loopback.`, - "# Do not edit between markers; `npm run service:uninstall` removes them.", - MANAGED_LINE, - MARKER_END, - ].join(os.EOL); -} - -function stripManagedBlock(content) { - const start = content.indexOf(MARKER_START); - if (start === -1) return content; - const end = content.indexOf(MARKER_END); - if (end === -1) return content; - const after = end + MARKER_END.length; - let trimmed = content.slice(0, start) + content.slice(after); - trimmed = trimmed.replace(/\n{3,}/g, "\n\n"); - if (!trimmed.endsWith(os.EOL)) trimmed += os.EOL; - return trimmed; -} - -function readCurrent() { - const p = hostsFilePath(); - if (!fs.existsSync(p)) return { path: p, content: "" }; - return { path: p, content: fs.readFileSync(p, "utf-8") }; -} - -export function hasManagedBlock() { - const { content } = readCurrent(); - return content.includes(MARKER_START) && content.includes(MARKER_END); -} - -function writeWithPrivilege(targetPath, newContent) { - if (process.platform === "win32") { - return writeWindows(targetPath, newContent); - } - return writeUnix(targetPath, newContent); -} - -function writeUnix(targetPath, newContent) { - const tmp = path.join(os.tmpdir(), `spent-hosts-${process.pid}.tmp`); - fs.writeFileSync(tmp, newContent, { mode: 0o644 }); - try { - console.log(`Updating ${targetPath} (requires sudo for this step only).`); - const r = spawnSync("sudo", ["cp", tmp, targetPath], { stdio: "inherit" }); - if (r.status !== 0) { - throw new Error( - `sudo cp failed (exit ${r.status}). Hosts file not modified.`, - ); - } - } finally { - try { - fs.unlinkSync(tmp); - } catch { - // best-effort cleanup - } - } -} - -function writeWindows(targetPath, newContent) { - try { - fs.writeFileSync(targetPath, newContent); - return; - } catch (err) { - throw new Error( - `Cannot write ${targetPath}: ${(err instanceof Error ? err.message : err)}.\n` + - "Re-run this command from an Administrator PowerShell:\n" + - ' Start-Process powershell -Verb RunAs -ArgumentList "-NoExit","-Command","cd \\"' + - process.cwd() + - '\\"; npm run service:install"', - ); - } -} - -// Look for a managed block that points to the *previous* hostname -// (spent.local), regardless of the current FRIENDLY_HOST value. Used to -// detect legacy installs that need migration. -function hasLegacyLocalBlock(content) { - if (!content.includes(MARKER_START)) return false; - const start = content.indexOf(MARKER_START); - const end = content.indexOf(MARKER_END); - if (end === -1) return false; - const block = content.slice(start, end + MARKER_END.length); - return /\bspent\.local\b/.test(block) && !/\bspent\.localhost\b/.test(block); -} - -function hasCurrentBlock(content) { - if (!content.includes(MARKER_START)) return false; - const start = content.indexOf(MARKER_START); - const end = content.indexOf(MARKER_END); - if (end === -1) return false; - const block = content.slice(start, end + MARKER_END.length); - return block.includes(MANAGED_LINE); -} - -export function addManagedBlock() { - const { path: p, content } = readCurrent(); - const isWindows = process.platform === "win32"; - const legacy = hasLegacyLocalBlock(content); - - if (!isWindows) { - // macOS and Linux resolve *.localhost natively (macOS via the system - // resolver, Linux via nsswitch `myhostname`). No hosts entry needed. - // If a legacy `spent.local` block is still present, strip it so the old - // hostname stops resolving (avoids stale bookmarks hitting a slow mDNS - // detour or the now-wrong loopback line). - if (!legacy && !content.includes(MARKER_START)) { - return false; - } - if (legacy) { - console.log( - "Removing legacy spent.local hosts entry (no longer needed on this OS; *.localhost resolves natively).", - ); - } - const newContent = stripManagedBlock(content); - writeWithPrivilege(p, newContent); - return true; - } - - // Windows: write/refresh the managed block. - if (hasCurrentBlock(content)) { - console.log(`Hosts file already has the spent block. No changes.`); - return false; - } - - const stripped = stripManagedBlock(content); - const block = buildManagedBlock(); - const base = stripped.endsWith("\n") || stripped === "" ? stripped : stripped + os.EOL; - const newContent = base + block + os.EOL; - - if (legacy) { - console.log(`Migrating hosts entry from spent.local to ${FRIENDLY_HOST}.`); - } else { - console.log(`Will append these lines to ${p}:`); - console.log("---"); - console.log(block); - console.log("---"); - } - - writeWithPrivilege(p, newContent); - return true; -} - -export function removeManagedBlock() { - const { path: p, content } = readCurrent(); - if (!content.includes(MARKER_START)) { - console.log(`No spent block found in ${p}. Nothing to remove.`); - return false; - } - - const newContent = stripManagedBlock(content); - writeWithPrivilege(p, newContent); - return true; -} diff --git a/scripts/service/install.mjs b/scripts/service/install.mjs deleted file mode 100644 index 99ad8a0..0000000 --- a/scripts/service/install.mjs +++ /dev/null @@ -1,54 +0,0 @@ -#!/usr/bin/env node -import { assertRepoRoot, assertNotRoot, LOOPBACK_URL, URL_BASE } from "./paths.mjs"; - -const COMMANDS = new Set([ - "install", - "uninstall", - "start", - "stop", - "status", - "logs", - "open", -]); - -function usage() { - console.error( - "usage: node scripts/service/install.mjs ", - ); -} - -async function main() { - const cmd = process.argv[2]; - if (!cmd || !COMMANDS.has(cmd)) { - usage(); - process.exit(2); - } - - assertNotRoot(); - const pkg = assertRepoRoot(); - - let impl; - switch (process.platform) { - case "darwin": - impl = await import("./macos.mjs"); - break; - case "linux": - impl = await import("./linux.mjs"); - break; - case "win32": - impl = await import("./windows.mjs"); - break; - default: - throw new Error( - `Unsupported platform: ${process.platform}. ` + - `Spent's service installer supports macOS, Linux, and Windows.`, - ); - } - - await impl.run(cmd, { pkg, loopbackUrl: LOOPBACK_URL, friendlyUrl: URL_BASE }); -} - -main().catch((err) => { - console.error(err instanceof Error ? err.message : err); - process.exit(1); -}); diff --git a/scripts/service/linux.mjs b/scripts/service/linux.mjs deleted file mode 100644 index 93c38a9..0000000 --- a/scripts/service/linux.mjs +++ /dev/null @@ -1,170 +0,0 @@ -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import { spawnSync } from "node:child_process"; -import { PORT, REPO_ROOT, renderTemplate } from "./paths.mjs"; -import { addManagedBlock, removeManagedBlock } from "./hosts.mjs"; - -const UNIT_NAME = "spent.service"; -const UNIT_DIR = path.join( - process.env.XDG_CONFIG_HOME ?? path.join(os.homedir(), ".config"), - "systemd", - "user", -); -const UNIT_PATH = path.join(UNIT_DIR, UNIT_NAME); -const LOG_DIR = path.join( - process.env.XDG_STATE_HOME ?? path.join(os.homedir(), ".local", "state"), - "spent", - "log", -); - -function whichNode() { - const r = spawnSync("which", ["node"], { encoding: "utf-8" }); - if (r.status !== 0) throw new Error("Cannot find `node` on PATH."); - return r.stdout.trim(); -} - -function ensureLogDir() { - fs.mkdirSync(LOG_DIR, { recursive: true }); - try { - fs.chmodSync(LOG_DIR, 0o700); - } catch { - // best-effort - } -} - -function writeUnit() { - const nodePath = whichNode(); - const pathEnv = `${path.dirname(nodePath)}:/usr/local/bin:/usr/bin:/bin`; - const content = renderTemplate("spent.service", { - nodePath, - repoRoot: REPO_ROOT, - port: PORT, - pathEnv, - logDir: LOG_DIR, - }); - fs.mkdirSync(UNIT_DIR, { recursive: true }); - fs.writeFileSync(UNIT_PATH, content, { mode: 0o644 }); -} - -function systemctl(args, opts = {}) { - return spawnSync("systemctl", ["--user", ...args], { - encoding: "utf-8", - stdio: opts.stdio ?? "inherit", - }); -} - -function checkPortBinding() { - const r = spawnSync( - "ss", - ["-Hltn", `sport = :${PORT}`], - { encoding: "utf-8" }, - ); - if (r.status !== 0) { - return { listening: false }; - } - const lines = r.stdout.split("\n").filter(Boolean); - const onLoopback = lines.some( - (l) => l.includes(`127.0.0.1:${PORT}`) || l.includes(`[::1]:${PORT}`), - ); - const onWildcard = lines.some( - (l) => l.includes(`0.0.0.0:${PORT}`) || l.includes(`*:${PORT}`), - ); - return { listening: lines.length > 0, onLoopback, onWildcard }; -} - -function ensureSystemdAvailable() { - const r = spawnSync("systemctl", ["--user", "is-system-running"], { - encoding: "utf-8", - }); - if (r.error || (r.status !== 0 && !r.stdout)) { - throw new Error( - "systemd user instance not available. " + - "On WSL/some minimal distros, enable lingering: `loginctl enable-linger $USER`. " + - "Or run the server manually with `npm run start`.", - ); - } -} - -function preflight() { - if (!fs.existsSync(path.join(REPO_ROOT, ".next"))) { - console.warn( - "WARNING: .next/ not found. Run `npm run build` before installing the service.", - ); - } -} - -export async function run(cmd, { friendlyUrl }) { - switch (cmd) { - case "install": { - preflight(); - ensureSystemdAvailable(); - ensureLogDir(); - writeUnit(); - systemctl(["daemon-reload"]); - systemctl(["enable", "--now", UNIT_NAME]); - try { - addManagedBlock(); - } catch (err) { - console.error(`Hosts file edit failed: ${err.message}`); - console.error("Service is still installed. You can fix hosts later."); - } - setTimeout(() => { - const state = checkPortBinding(); - if (state.onWildcard) { - console.error( - `DANGER: server is bound to wildcard address on :${PORT}. ` + - `Stop the service immediately with: npm run service:stop`, - ); - process.exit(1); - } - console.log(`Spent is running. Open ${friendlyUrl}.`); - }, 1500); - return; - } - case "uninstall": { - systemctl(["disable", "--now", UNIT_NAME]); - if (fs.existsSync(UNIT_PATH)) fs.unlinkSync(UNIT_PATH); - systemctl(["daemon-reload"]); - try { - removeManagedBlock(); - } catch (err) { - console.error(`Hosts file cleanup failed: ${err.message}`); - } - console.log("Spent service removed. The repo and data/ directory are untouched."); - return; - } - case "start": { - systemctl(["start", UNIT_NAME]); - return; - } - case "stop": { - systemctl(["stop", UNIT_NAME]); - return; - } - case "status": { - systemctl(["status", UNIT_NAME, "--no-pager"]); - const port = checkPortBinding(); - console.log( - `\nPort ${PORT}: ${ - port.onLoopback - ? "127.0.0.1 (ok)" - : port.onWildcard - ? "WILDCARD (NOT OK)" - : "not bound" - }`, - ); - return; - } - case "logs": { - spawnSync("journalctl", ["--user", "-u", UNIT_NAME, "-f"], { - stdio: "inherit", - }); - return; - } - case "open": { - spawnSync("xdg-open", [friendlyUrl], { stdio: "inherit" }); - return; - } - } -} diff --git a/scripts/service/macos.mjs b/scripts/service/macos.mjs deleted file mode 100644 index 505aba4..0000000 --- a/scripts/service/macos.mjs +++ /dev/null @@ -1,173 +0,0 @@ -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import { spawnSync } from "node:child_process"; -import { PORT, REPO_ROOT, renderTemplate } from "./paths.mjs"; -import { addManagedBlock, removeManagedBlock } from "./hosts.mjs"; - -const LABEL = "com.spent.app"; -const PLIST_PATH = path.join(os.homedir(), "Library", "LaunchAgents", `${LABEL}.plist`); -const LOG_DIR = path.join(os.homedir(), "Library", "Logs", "Spent"); - -function whichNode() { - const r = spawnSync("which", ["node"], { encoding: "utf-8" }); - if (r.status !== 0) throw new Error("Cannot find `node` on PATH."); - return r.stdout.trim(); -} - -function ensureLogDir() { - fs.mkdirSync(LOG_DIR, { recursive: true }); - try { - fs.chmodSync(LOG_DIR, 0o700); - } catch { - // best-effort - } -} - -function writePlist() { - const nodePath = whichNode(); - const pathEnv = `${path.dirname(nodePath)}:/usr/local/bin:/opt/homebrew/bin:/usr/bin:/bin`; - const content = renderTemplate("com.spent.app.plist", { - nodePath, - repoRoot: REPO_ROOT, - port: PORT, - pathEnv, - logDir: LOG_DIR, - }); - fs.mkdirSync(path.dirname(PLIST_PATH), { recursive: true }); - fs.writeFileSync(PLIST_PATH, content, { mode: 0o644 }); -} - -function launchctl(args, opts = {}) { - return spawnSync("launchctl", args, { - encoding: "utf-8", - stdio: opts.stdio ?? "pipe", - }); -} - -function bootstrap() { - const uid = process.getuid(); - return launchctl(["bootstrap", `gui/${uid}`, PLIST_PATH], { stdio: "inherit" }); -} - -function bootout() { - const uid = process.getuid(); - return launchctl(["bootout", `gui/${uid}/${LABEL}`]); -} - -function kickstart() { - const uid = process.getuid(); - return launchctl(["kickstart", "-k", `gui/${uid}/${LABEL}`]); -} - -function checkPortBinding() { - const r = spawnSync( - "lsof", - ["-nP", `-iTCP:${PORT}`, "-sTCP:LISTEN"], - { encoding: "utf-8" }, - ); - if (r.status !== 0) return { listening: false }; - const lines = r.stdout.split("\n").filter(Boolean); - const onLoopback = lines.some((l) => l.includes(`127.0.0.1:${PORT}`)); - const onWildcard = lines.some( - (l) => l.includes(`*:${PORT}`) || l.includes(`0.0.0.0:${PORT}`), - ); - return { listening: lines.length > 0, onLoopback, onWildcard }; -} - -function preflight() { - if (!fs.existsSync(path.join(REPO_ROOT, ".next"))) { - console.warn( - "WARNING: .next/ not found. Run `npm run build` before installing the service.", - ); - } -} - -export async function run(cmd, { friendlyUrl, loopbackUrl }) { - switch (cmd) { - case "install": { - preflight(); - ensureLogDir(); - writePlist(); - try { - addManagedBlock(); - } catch (err) { - console.error(`Hosts file edit failed: ${err.message}`); - console.error("Service file is still installed. You can fix hosts later."); - } - bootstrap(); - - setTimeout(() => { - const state = checkPortBinding(); - if (state.onWildcard) { - console.error( - `DANGER: server is bound to wildcard address on :${PORT}. ` + - `Inspect the plist at ${PLIST_PATH} and remove the service immediately.`, - ); - process.exit(1); - } - if (state.onLoopback) { - console.log(`Spent is running. Open ${friendlyUrl} or ${loopbackUrl}.`); - } else { - console.log( - `Service installed. Check status: npm run service:status`, - ); - } - }, 1500); - return; - } - case "uninstall": { - bootout(); - if (fs.existsSync(PLIST_PATH)) fs.unlinkSync(PLIST_PATH); - try { - removeManagedBlock(); - } catch (err) { - console.error(`Hosts file cleanup failed: ${err.message}`); - } - console.log("Spent service removed. The repo and data/ directory are untouched."); - return; - } - case "start": { - const r = bootstrap(); - if (r.status !== 0 && r.stderr?.includes("already loaded")) { - kickstart(); - } - console.log("Spent started."); - return; - } - case "stop": { - bootout(); - console.log("Spent stopped."); - return; - } - case "status": { - const r = launchctl(["print", `gui/${process.getuid()}/${LABEL}`]); - const loaded = r.status === 0; - const port = checkPortBinding(); - console.log(`LaunchAgent loaded: ${loaded ? "yes" : "no"}`); - console.log( - `Port ${PORT} bound: ${ - port.onLoopback - ? "yes (127.0.0.1, ok)" - : port.onWildcard - ? "yes (wildcard, NOT OK)" - : "no" - }`, - ); - console.log(`Plist: ${PLIST_PATH}`); - console.log(`Logs: ${LOG_DIR}/{out,err}.log`); - return; - } - case "logs": { - const errLog = path.join(LOG_DIR, "err.log"); - const outLog = path.join(LOG_DIR, "out.log"); - console.log(`tail -f ${errLog} ${outLog}`); - spawnSync("tail", ["-f", errLog, outLog], { stdio: "inherit" }); - return; - } - case "open": { - spawnSync("open", [friendlyUrl], { stdio: "inherit" }); - return; - } - } -} diff --git a/scripts/service/paths.mjs b/scripts/service/paths.mjs deleted file mode 100644 index a10ed71..0000000 --- a/scripts/service/paths.mjs +++ /dev/null @@ -1,49 +0,0 @@ -import path from "node:path"; -import { fileURLToPath } from "node:url"; -import fs from "node:fs"; - -const here = path.dirname(fileURLToPath(import.meta.url)); - -export const PORT = 41234; -export const HOST = "127.0.0.1"; -export const FRIENDLY_HOST = "spent.localhost"; -export const URL_BASE = `http://${FRIENDLY_HOST}:${PORT}`; -export const LOOPBACK_URL = `http://${HOST}:${PORT}`; - -export const REPO_ROOT = path.resolve(here, "..", ".."); -export const TEMPLATES_DIR = path.join(here, "templates"); - -export function assertRepoRoot() { - const pkgPath = path.join(REPO_ROOT, "package.json"); - if (!fs.existsSync(pkgPath)) { - throw new Error(`Refusing to run: no package.json at ${pkgPath}`); - } - const pkg = JSON.parse(fs.readFileSync(pkgPath, "utf-8")); - if (pkg.name !== "spent") { - throw new Error( - `Refusing to run: ${pkgPath} is for "${pkg.name}", not "spent". ` + - `This script must run from the spent repo.`, - ); - } - return pkg; -} - -export function assertNotRoot() { - if (process.platform === "win32") return; - if (typeof process.getuid === "function" && process.getuid() === 0) { - throw new Error( - "Refusing to run as root. Run this script as your normal user. " + - "The hosts file edit step will prompt for sudo on its own.", - ); - } -} - -export function renderTemplate(name, vars) { - const tpl = fs.readFileSync(path.join(TEMPLATES_DIR, name), "utf-8"); - return tpl.replace(/\{\{(\w+)\}\}/g, (_, key) => { - if (!(key in vars)) { - throw new Error(`Template ${name} references unknown var: ${key}`); - } - return String(vars[key]); - }); -} diff --git a/scripts/service/templates/com.spent.app.plist b/scripts/service/templates/com.spent.app.plist deleted file mode 100644 index aeabc17..0000000 --- a/scripts/service/templates/com.spent.app.plist +++ /dev/null @@ -1,50 +0,0 @@ - - - - - Label - com.spent.app - - ProgramArguments - - {{nodePath}} - {{repoRoot}}/node_modules/next/dist/bin/next - start - -H - 127.0.0.1 - -p - {{port}} - - - WorkingDirectory - {{repoRoot}} - - EnvironmentVariables - - NODE_ENV - production - PATH - {{pathEnv}} - - - RunAtLoad - - - KeepAlive - - SuccessfulExit - - Crashed - - - - ProcessType - Background - - StandardOutPath - {{logDir}}/out.log - - StandardErrorPath - {{logDir}}/err.log - - diff --git a/scripts/service/templates/spent-launcher.vbs b/scripts/service/templates/spent-launcher.vbs deleted file mode 100644 index 76ab494..0000000 --- a/scripts/service/templates/spent-launcher.vbs +++ /dev/null @@ -1,10 +0,0 @@ -' Spent launcher. Runs node + next start without a visible console window. -' wscript.exe is a GUI-subsystem host (no console attached), and WshShell.Run -' with intWindowStyle=0 + bWaitOnReturn=True keeps wscript alive as the parent -' so Task Scheduler can manage the lifecycle through its Job Object. -Option Explicit -Dim shell -Set shell = CreateObject("WScript.Shell") -shell.CurrentDirectory = "{{repoRoot}}" -shell.Run """{{nodePath}}"" ""{{repoRoot}}\node_modules\next\dist\bin\next"" start -H 127.0.0.1 -p {{port}}", 0, True -Set shell = Nothing diff --git a/scripts/service/templates/spent-task.xml b/scripts/service/templates/spent-task.xml deleted file mode 100644 index a436331..0000000 --- a/scripts/service/templates/spent-task.xml +++ /dev/null @@ -1,52 +0,0 @@ - - - - Spent — local personal finance tracker. Runs on 127.0.0.1:{{port}} only. - \Spent - - - - true - {{userId}} - - - - - {{userId}} - InteractiveToken - LeastPrivilege - - - - IgnoreNew - false - false - true - true - false - - false - false - - true - true - false - false - true - true - false - PT0S - 7 - - PT1M - 3 - - - - - wscript.exe - "{{launcherVbsPath}}" - {{repoRoot}} - - - diff --git a/scripts/service/templates/spent.service b/scripts/service/templates/spent.service deleted file mode 100644 index edc1272..0000000 --- a/scripts/service/templates/spent.service +++ /dev/null @@ -1,26 +0,0 @@ -[Unit] -Description=Spent — local personal finance tracker -Documentation=https://github.com/spent/spent -After=network.target - -[Service] -Type=simple -WorkingDirectory={{repoRoot}} -ExecStart={{nodePath}} {{repoRoot}}/node_modules/next/dist/bin/next start -H 127.0.0.1 -p {{port}} -Environment=NODE_ENV=production -Environment=PATH={{pathEnv}} -Restart=on-failure -RestartSec=5 -StandardOutput=append:{{logDir}}/out.log -StandardError=append:{{logDir}}/err.log - -# Hardening: the app only needs its own dir and loopback. -NoNewPrivileges=true -PrivateTmp=true -ProtectSystem=full -ProtectHome=read-only -ReadWritePaths={{repoRoot}}/data {{logDir}} -RestrictAddressFamilies=AF_INET AF_INET6 AF_UNIX - -[Install] -WantedBy=default.target diff --git a/scripts/service/windows.mjs b/scripts/service/windows.mjs deleted file mode 100644 index bfa03bd..0000000 --- a/scripts/service/windows.mjs +++ /dev/null @@ -1,191 +0,0 @@ -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import { spawnSync } from "node:child_process"; -import { PORT, REPO_ROOT, renderTemplate } from "./paths.mjs"; -import { addManagedBlock, removeManagedBlock } from "./hosts.mjs"; - -const TASK_NAME = "Spent"; -const LAUNCHER_DIR = path.join(os.homedir(), "AppData", "Local", "Spent"); -const LAUNCHER_VBS = path.join(LAUNCHER_DIR, "spent-launcher.vbs"); - -function whichNode() { - const r = spawnSync("where", ["node"], { encoding: "utf-8", shell: false }); - if (r.status !== 0) throw new Error("Cannot find `node` on PATH."); - return r.stdout.split(/\r?\n/).filter(Boolean)[0].trim(); -} - -// wscript.exe is a GUI-subsystem host, so launching node through this -// VBScript avoids the visible console window node.exe would otherwise get. -function writeLauncherVbs() { - fs.mkdirSync(LAUNCHER_DIR, { recursive: true }); - const content = renderTemplate("spent-launcher.vbs", { - port: PORT, - repoRoot: REPO_ROOT, - nodePath: whichNode(), - }); - fs.writeFileSync(LAUNCHER_VBS, content, { encoding: "utf-8" }); -} - -function removeLauncherVbs() { - try { - fs.unlinkSync(LAUNCHER_VBS); - } catch { - // best-effort - } - try { - fs.rmdirSync(LAUNCHER_DIR); - } catch { - // directory not empty or already gone; leave it - } -} - -function userId() { - return `${process.env.USERDOMAIN ?? os.hostname()}\\${process.env.USERNAME ?? os.userInfo().username}`; -} - -function isAdmin() { - const r = spawnSync("net", ["session"], { encoding: "utf-8" }); - return r.status === 0; -} - -function writeTaskXml() { - const xmlPath = path.join(os.tmpdir(), `spent-task-${process.pid}.xml`); - const content = renderTemplate("spent-task.xml", { - port: PORT, - repoRoot: REPO_ROOT, - nodePath: whichNode(), - userId: userId(), - launcherVbsPath: LAUNCHER_VBS, - }); - fs.writeFileSync(xmlPath, "" + content, { encoding: "utf16le" }); - return xmlPath; -} - -function schtasks(args) { - return spawnSync("schtasks", args, { encoding: "utf-8" }); -} - -function checkPortBinding() { - const r = spawnSync("netstat", ["-ano", "-p", "TCP"], { encoding: "utf-8" }); - if (r.status !== 0) return { listening: false }; - const lines = r.stdout.split(/\r?\n/); - const portStr = `:${PORT}`; - const matches = lines.filter((l) => l.includes(portStr) && l.includes("LISTENING")); - const onLoopback = matches.some((l) => l.includes(`127.0.0.1${portStr}`)); - const onWildcard = matches.some( - (l) => l.includes(`0.0.0.0${portStr}`) || /\s\[::\]:/.test(l) ? l.includes(portStr) : false, - ); - return { listening: matches.length > 0, onLoopback, onWildcard }; -} - -function preflight() { - if (!fs.existsSync(path.join(REPO_ROOT, ".next"))) { - console.warn( - "WARNING: .next/ not found. Run `npm run build` before installing the service.", - ); - } - if (!isAdmin()) { - console.warn( - "Note: hosts file edit needs Administrator. " + - "If it fails, re-run this command from an elevated terminal.", - ); - } -} - -export async function run(cmd, { friendlyUrl, loopbackUrl }) { - switch (cmd) { - case "install": { - preflight(); - writeLauncherVbs(); - const xmlPath = writeTaskXml(); - try { - const r = schtasks(["/Create", "/TN", TASK_NAME, "/XML", xmlPath, "/F"]); - if (r.status !== 0) { - console.error(r.stderr || r.stdout); - throw new Error(`schtasks /Create failed (exit ${r.status}).`); - } - const run = schtasks(["/Run", "/TN", TASK_NAME]); - if (run.status !== 0) console.error(run.stderr || run.stdout); - } finally { - try { - fs.unlinkSync(xmlPath); - } catch { - // best-effort - } - } - try { - addManagedBlock(); - // Windows caches negative DNS lookups, so even a fresh `spent.localhost` - // entry can still resolve as NXDOMAIN until the cache is cleared. - spawnSync("ipconfig", ["/flushdns"], { stdio: "ignore" }); - } catch (err) { - console.error(`Hosts file edit failed: ${err.message}`); - console.error("Task is still installed. You can bookmark " + loopbackUrl); - } - setTimeout(() => { - const state = checkPortBinding(); - if (state.onWildcard && !state.onLoopback) { - console.error( - `DANGER: server is bound to wildcard on :${PORT}. ` + - `Stop the task immediately: npm run service:stop`, - ); - process.exit(1); - } - console.log(`Spent is running. Open ${friendlyUrl} or ${loopbackUrl}.`); - }, 2000); - return; - } - case "uninstall": { - schtasks(["/End", "/TN", TASK_NAME]); - const r = schtasks(["/Delete", "/TN", TASK_NAME, "/F"]); - if (r.status !== 0 && !/cannot find/i.test(r.stderr ?? "")) { - console.error(r.stderr || r.stdout); - } - removeLauncherVbs(); - try { - removeManagedBlock(); - } catch (err) { - console.error(`Hosts file cleanup failed: ${err.message}`); - } - console.log("Spent task removed. The repo and data/ directory are untouched."); - return; - } - case "start": { - schtasks(["/Run", "/TN", TASK_NAME]); - return; - } - case "stop": { - schtasks(["/End", "/TN", TASK_NAME]); - return; - } - case "status": { - const r = schtasks(["/Query", "/TN", TASK_NAME, "/V", "/FO", "LIST"]); - console.log(r.stdout || r.stderr); - const port = checkPortBinding(); - console.log( - `Port ${PORT}: ${ - port.onLoopback - ? "127.0.0.1 (ok)" - : port.onWildcard - ? "WILDCARD (NOT OK)" - : "not bound" - }`, - ); - return; - } - case "logs": { - console.log( - "Windows Task Scheduler does not capture stdout/stderr by default.", - ); - console.log( - "Run the server manually with `npm run start` to see logs interactively.", - ); - return; - } - case "open": { - spawnSync("cmd", ["/c", "start", "", friendlyUrl], { stdio: "inherit" }); - return; - } - } -} diff --git a/scripts/setup.mjs b/scripts/setup.mjs deleted file mode 100644 index 8a45b32..0000000 --- a/scripts/setup.mjs +++ /dev/null @@ -1,459 +0,0 @@ -#!/usr/bin/env node -// One-shot setup: build Next, install the always-on service, build + install -// the platform menubar, register login auto-start, open the dashboard. -// -// Platform paths: -// macOS -> Spent.app installed to ~/Applications, registered as Login Item -// Windows -> Spent.exe installed to %LOCALAPPDATA%\Programs\Spent, .lnk in Startup -// Linux -> service only (no native tray) - -import { spawnSync } from "node:child_process"; -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import readline from "node:readline"; -import { fileURLToPath } from "node:url"; - -const HERE = path.dirname(fileURLToPath(import.meta.url)); -const REPO_ROOT = path.resolve(HERE, ".."); -const SERVICE_INSTALL = path.join(HERE, "service", "install.mjs"); -const FRIENDLY_URL = "http://spent.localhost:41234"; -const NPM = process.platform === "win32" ? "npm.cmd" : "npm"; - -function step(msg) { - console.log(`\n=> ${msg}`); -} - -function done(msg) { - console.log(` ${msg}`); -} - -function fail(msg) { - console.error(`\nsetup: ${msg}`); - process.exit(1); -} - -function run(cmd, args, opts = {}) { - // Node v20.12.2+ rejects direct spawning of .cmd/.bat on Windows with EINVAL - // (CVE-2024-27980 mitigation). Use shell:true for those; args here are - // hardcoded so the usual injection risk does not apply. - const needsShell = process.platform === "win32" && /\.(cmd|bat)$/i.test(cmd); - const r = spawnSync(cmd, args, { - stdio: "inherit", - shell: needsShell, - ...opts, - }); - if (r.error) { - throw new Error( - `\`${cmd} ${args.join(" ")}\` failed to launch: ${r.error.message}`, - ); - } - if (r.status !== 0) { - const sig = r.signal ? ` (signal: ${r.signal})` : ""; - throw new Error(`\`${cmd} ${args.join(" ")}\` exited with status ${r.status}${sig}`); - } -} - -function which(cmd) { - const tool = process.platform === "win32" ? "where" : "which"; - const r = spawnSync(tool, [cmd], { encoding: "utf-8" }); - if (r.status !== 0) return null; - return r.stdout.split(/\r?\n/).filter(Boolean)[0]?.trim() || null; -} - -function sleep(ms) { - return new Promise((res) => setTimeout(res, ms)); -} - -// Prompt for yes/no. Non-TTY (CI, piped stdin) falls back to the default -// so we don't hang. Empty input takes the default. Anything starting with -// y is yes. -async function askYesNo(question, defaultYes = true) { - const suffix = defaultYes ? " [Y/n] " : " [y/N] "; - if (!process.stdin.isTTY) { - console.log(question + suffix + (defaultYes ? "y (non-interactive)" : "n (non-interactive)")); - return defaultYes; - } - return new Promise((resolve) => { - const rl = readline.createInterface({ input: process.stdin, output: process.stdout }); - rl.question(question + suffix, (answer) => { - rl.close(); - const trimmed = (answer ?? "").trim().toLowerCase(); - if (!trimmed) return resolve(defaultYes); - resolve(trimmed === "y" || trimmed === "yes"); - }); - }); -} - -async function ensureMacPrereq() { - if (which("swift")) return true; - - console.log(""); - console.log("Xcode Command Line Tools are required to build the macOS menubar."); - console.log("Size: about 1 GB. Time: 5-15 minutes."); - console.log("Will run: xcode-select --install (opens a system dialog)"); - console.log(""); - - const yes = await askYesNo("Install now?"); - if (!yes) return false; - - step("Installing Xcode Command Line Tools"); - console.log(" A system dialog will appear. Click 'Install' and accept the license."); - // xcode-select --install returns immediately after dispatching the dialog, - // so we have to poll for swift becoming available. - spawnSync("xcode-select", ["--install"], { stdio: "inherit" }); - - console.log(""); - step("Waiting for install to finish (this can take 5-15 minutes)"); - const start = Date.now(); - const timeoutMs = 30 * 60 * 1000; - let lastMin = -1; - while (Date.now() - start < timeoutMs) { - if (which("swift")) { - done("Xcode Command Line Tools are ready"); - return true; - } - await sleep(10 * 1000); - const min = Math.floor((Date.now() - start) / 60000); - if (min !== lastMin && min > 0) { - console.log(` still waiting (${min} min elapsed)...`); - lastMin = min; - } - } - console.error(" timed out after 30 minutes. Re-run `npm run setup` after install completes."); - return false; -} - -function hasDotnet8Sdk(dotnetPath) { - const r = spawnSync(dotnetPath, ["--list-sdks"], { encoding: "utf-8" }); - return r.status === 0 && /^(8|9|1\d)\./m.test(r.stdout || ""); -} - -async function ensureWindowsPrereq() { - const dotnet = which("dotnet"); - if (dotnet && hasDotnet8Sdk(dotnet)) return true; - - console.log(""); - console.log(".NET 8 SDK is required to build the Windows menubar."); - console.log("Size: about 200 MB. Time: 2-5 minutes."); - - if (!which("winget")) { - console.log(""); - console.error("winget is not available on this machine. Install .NET 8 SDK manually:"); - console.error(" https://dotnet.microsoft.com/download/dotnet/8.0"); - console.error("Then re-run `npm run setup`."); - return false; - } - - console.log("Will run: winget install Microsoft.DotNet.SDK.8"); - console.log(""); - - const yes = await askYesNo("Install now?"); - if (!yes) return false; - - step("Installing .NET 8 SDK via winget"); - const r = spawnSync( - "winget", - [ - "install", - "Microsoft.DotNet.SDK.8", - "--accept-package-agreements", - "--accept-source-agreements", - "-e", - ], - { stdio: "inherit", shell: true }, - ); - if (r.status !== 0) { - console.error(" winget exited with non-zero status. Check the output above."); - return false; - } - - // winget refreshes PATH for new shells, not the current one. Re-query in - // case it landed in a directory our process already knows about; otherwise - // tell the user to reopen the terminal. - const dotnet2 = which("dotnet"); - if (dotnet2 && hasDotnet8Sdk(dotnet2)) { - done(".NET 8 SDK is ready"); - return true; - } - console.error(" .NET 8 SDK installed, but `dotnet` is not yet on PATH for this shell."); - console.error(" Close and reopen your terminal, then re-run `npm run setup`."); - return false; -} - -function preflight() { - if (!fs.existsSync(path.join(REPO_ROOT, "node_modules", "next"))) { - fail( - "Dependencies not installed. Run this first:\n" + - " npm install\n" + - "Then re-run `npm run setup`.", - ); - } - - if (process.platform === "win32") { - // Hosts file edit needs Administrator on Windows, but setup itself does - // not. If the shell is not elevated, warn (so the user knows why - // spent.localhost may not resolve on older Windows) and keep going. The - // web app at 127.0.0.1 works regardless. - const r = spawnSync("net", ["session"], { encoding: "utf-8" }); - if (r.status !== 0) { - console.log(""); - console.log("Note: this shell is not Administrator."); - console.log("Setup will continue, but the hosts entry for spent.localhost cannot be added."); - console.log(" - http://127.0.0.1:41234 will work normally."); - console.log(" - http://spent.localhost:41234 may not resolve on older Windows builds."); - console.log("To guarantee the friendly hostname, relaunch from an elevated"); - console.log("PowerShell (Win+X -> 'Terminal (Admin)') and run `npm run service:install`."); - console.log(""); - } - } -} - -function windowsHostsHasSpentLocalhost() { - const hostsPath = `${process.env.SystemRoot ?? "C:\\Windows"}\\System32\\drivers\\etc\\hosts`; - try { - const content = fs.readFileSync(hostsPath, "utf-8"); - return /^[^#\n]*\b127\.0\.0\.1\b[^\n]*\bspent\.localhost\b/m.test(content); - } catch { - return false; - } -} - -function dashboardUrl() { - // macOS and Linux resolve *.localhost natively (no hosts file needed). - // On Windows, fall back to loopback if the hosts entry wasn't written - // (e.g. user didn't run setup from an elevated shell). - if (process.platform !== "win32") return FRIENDLY_URL; - return windowsHostsHasSpentLocalhost() ? FRIENDLY_URL : "http://127.0.0.1:41234"; -} - -function buildNextApp() { - step("Building Next.js app"); - run(NPM, ["run", "build"], { cwd: REPO_ROOT }); -} - -function installService() { - step("Installing background service"); - run(process.execPath, [SERVICE_INSTALL, "install"], { cwd: REPO_ROOT }); -} - -// launchctl/schtasks/systemctl all return as soon as the job is registered, -// not when `next start` is actually listening. Poll /api/health so the -// browser-open at the end of setup doesn't beat the server to the punch. -async function waitForServer(maxMs = 60000) { - const url = "http://127.0.0.1:41234/api/health"; - const start = Date.now(); - let lastErr = null; - while (Date.now() - start < maxMs) { - try { - const r = await fetch(url, { signal: AbortSignal.timeout(2000) }); - if (r.ok) { - const data = await r.json().catch(() => ({})); - if (data && data.ok === true) return true; - } - } catch (err) { - lastErr = err; - } - await new Promise((res) => setTimeout(res, 500)); - } - if (lastErr) { - console.error(` last error while polling: ${lastErr.message ?? lastErr}`); - } - return false; -} - -async function macSetup() { - const ok = await ensureMacPrereq(); - if (!ok) { - step("Skipping menubar"); - console.log(` Web app is installed and running at ${dashboardUrl()}`); - console.log(" Install Xcode Command Line Tools and re-run `npm run setup` to add the menubar."); - return; - } - - step("Building Spent.app"); - run("bash", [path.join(REPO_ROOT, "menubar", "mac", "build.sh")]); - - step("Installing Spent.app to ~/Applications"); - const appsDir = path.join(os.homedir(), "Applications"); - fs.mkdirSync(appsDir, { recursive: true }); - const target = path.join(appsDir, "Spent.app"); - if (fs.existsSync(target)) { - fs.rmSync(target, { recursive: true, force: true }); - } - run("cp", ["-R", path.join(REPO_ROOT, "menubar", "mac", "build", "Spent.app"), target]); - done(target); - - step("Registering Login Item"); - addLoginItemMac(target); - - step("Launching menubar"); - spawnSync("open", [target], { stdio: "ignore" }); - - step("Opening dashboard"); - spawnSync("open", [dashboardUrl()], { stdio: "ignore" }); -} - -function addLoginItemMac(appPath) { - const check = spawnSync( - "osascript", - ["-e", 'tell application "System Events" to get the name of every login item'], - { encoding: "utf-8" }, - ); - if (check.status === 0 && /(^|, )Spent($|,)/.test(check.stdout.trim())) { - done("already registered"); - return; - } - - const escapedPath = appPath.replace(/"/g, '\\"'); - const r = spawnSync( - "osascript", - [ - "-e", - `tell application "System Events" to make login item at end with properties {path:"${escapedPath}", hidden:true}`, - ], - { encoding: "utf-8" }, - ); - if (r.status === 0) { - done("added"); - } else { - console.error(` could not add login item: ${r.stderr?.trim() || "unknown error"}`); - console.error(" add manually: System Settings -> General -> Login Items -> +"); - } -} - -async function windowsSetup() { - const ok = await ensureWindowsPrereq(); - if (!ok) { - step("Skipping menubar"); - console.log(` Web app is installed and running at ${dashboardUrl()}`); - console.log(" Install .NET 8 SDK and re-run `npm run setup` to add the menubar."); - return; - } - - step("Building Spent.exe"); - run("powershell", [ - "-ExecutionPolicy", "Bypass", - "-File", path.join(REPO_ROOT, "menubar", "windows", "build.ps1"), - ]); - - step("Installing Spent.exe to %LOCALAPPDATA%\\Programs\\Spent"); - const localAppData = process.env.LOCALAPPDATA; - if (!localAppData) fail("LOCALAPPDATA env var is not set."); - const installDir = path.join(localAppData, "Programs", "Spent"); - fs.mkdirSync(installDir, { recursive: true }); - const targetExe = path.join(installDir, "Spent.exe"); - const builtExe = path.join(REPO_ROOT, "menubar", "windows", "build", "Spent.exe"); - fs.copyFileSync(builtExe, targetExe); - done(targetExe); - - step("Adding Startup shortcut"); - addStartupShortcutWindows(targetExe); - - step("Launching menubar"); - spawnSync("powershell", ["-Command", `Start-Process "${targetExe}"`], { stdio: "ignore" }); - - step("Opening dashboard"); - spawnSync("cmd", ["/c", "start", "", dashboardUrl()], { stdio: "ignore" }); -} - -function addStartupShortcutWindows(exePath) { - const appData = process.env.APPDATA; - if (!appData) { - console.error(" APPDATA env var is not set; skipping Startup shortcut."); - return; - } - const startupDir = path.join( - appData, "Microsoft", "Windows", "Start Menu", "Programs", "Startup", - ); - fs.mkdirSync(startupDir, { recursive: true }); - const shortcutPath = path.join(startupDir, "Spent.lnk"); - if (fs.existsSync(shortcutPath)) { - done("already present"); - return; - } - - const ps = - `$ws = New-Object -ComObject WScript.Shell; ` + - `$sc = $ws.CreateShortcut('${shortcutPath.replace(/'/g, "''")}'); ` + - `$sc.TargetPath = '${exePath.replace(/'/g, "''")}'; ` + - `$sc.Save()`; - const r = spawnSync("powershell", ["-NoProfile", "-Command", ps], { encoding: "utf-8" }); - if (r.status === 0) { - done("added"); - } else { - console.error(` could not create startup shortcut: ${r.stderr?.trim() || "unknown error"}`); - } -} - -function linuxSetup() { - step("Linux: no native tray"); - console.log(" Spent on Linux is web-only. The service is installed and"); - console.log(" running. Control it with the npm scripts:"); - console.log(" npm run service:status / :start / :stop / :reload / :logs"); - - step("Opening dashboard"); - spawnSync("xdg-open", [dashboardUrl()], { stdio: "ignore" }); -} - -async function main() { - console.log("Spent setup"); - console.log(` platform: ${process.platform}`); - console.log(` repo: ${REPO_ROOT}`); - - preflight(); - buildNextApp(); - installService(); - - step("Waiting for server to come up"); - const ready = await waitForServer(); - if (ready) { - done("server is healthy"); - } else { - console.error(" server did not respond within 60s, continuing anyway."); - console.error(` check logs: npm run service:logs`); - } - - switch (process.platform) { - case "darwin": - await macSetup(); - break; - case "win32": - await windowsSetup(); - break; - case "linux": - linuxSetup(); - break; - default: - fail(`unsupported platform: ${process.platform}`); - } - - printCheatSheet(); -} - -function printCheatSheet() { - const url = dashboardUrl(); - console.log(""); - console.log("================================================================"); - console.log(` Done. Spent is at ${url}`); - if (url !== FRIENDLY_URL) { - console.log(` (${FRIENDLY_URL} would also work once the hosts entry is in place.)`); - } - console.log("================================================================"); - console.log(""); - console.log("Useful commands:"); - console.log(" npm run service:status see if the service is running"); - console.log(" npm run service:start start the background service"); - console.log(" npm run service:stop stop the background service"); - console.log(" npm run service:reload rebuild and restart (after code edits)"); - console.log(" npm run service:logs tail the server logs"); - console.log(" npm run service:open open the dashboard in your browser"); - console.log(" npm run uninstall remove the service and menubar"); - console.log(""); - console.log("Tip: bookmark the URL above so the daily flow is one click."); - console.log(""); -} - -main().catch((err) => { - fail(err instanceof Error ? err.message : String(err)); -}); diff --git a/scripts/uninstall.mjs b/scripts/uninstall.mjs deleted file mode 100644 index 2e73b59..0000000 --- a/scripts/uninstall.mjs +++ /dev/null @@ -1,157 +0,0 @@ -#!/usr/bin/env node -// Mirror of setup.mjs: tear down everything `npm run setup` installed. -// The data/ directory and the repo itself are deliberately NOT touched: -// uninstall is about removing the always-on service and the menubar, not -// the user's transactions and encryption key. - -import { spawnSync } from "node:child_process"; -import fs from "node:fs"; -import os from "node:os"; -import path from "node:path"; -import { fileURLToPath } from "node:url"; - -const HERE = path.dirname(fileURLToPath(import.meta.url)); -const REPO_ROOT = path.resolve(HERE, ".."); -const SERVICE_INSTALL = path.join(HERE, "service", "install.mjs"); - -function step(msg) { - console.log(`\n=> ${msg}`); -} - -function done(msg) { - console.log(` ${msg}`); -} - -// Uninstall is best-effort: missing pieces are not errors, they're already-gone. -function tryRun(cmd, args, opts = {}) { - return spawnSync(cmd, args, { stdio: "ignore", ...opts }); -} - -function uninstallService() { - step("Removing background service"); - const r = spawnSync(process.execPath, [SERVICE_INSTALL, "uninstall"], { - cwd: REPO_ROOT, - stdio: "inherit", - }); - if (r.status !== 0) { - console.error(` service uninstaller exited with status ${r.status}, continuing`); - } -} - -async function macUninstall() { - step("Quitting Spent menubar (if running)"); - tryRun("osascript", ["-e", 'tell application "Spent" to quit']); - // Give the GUI app a moment to flush and exit before we remove its bundle. - await new Promise((r) => setTimeout(r, 500)); - done("done"); - - step("Removing ~/Applications/Spent.app"); - const appPath = path.join(os.homedir(), "Applications", "Spent.app"); - if (fs.existsSync(appPath)) { - fs.rmSync(appPath, { recursive: true, force: true }); - done("removed"); - } else { - done("not present"); - } - - step("Removing Login Item"); - const r = spawnSync( - "osascript", - [ - "-e", - 'tell application "System Events" to delete (every login item whose name is "Spent")', - ], - { encoding: "utf-8" }, - ); - if (r.status === 0) { - done("removed"); - } else { - done("nothing to remove (or System Events refused)"); - } -} - -function windowsUninstall() { - step("Stopping Spent menubar (if running)"); - tryRun("taskkill", ["/IM", "Spent.exe", "/F"]); - done("done"); - - step("Removing %LOCALAPPDATA%\\Programs\\Spent"); - const localAppData = process.env.LOCALAPPDATA; - if (localAppData) { - const installDir = path.join(localAppData, "Programs", "Spent"); - if (fs.existsSync(installDir)) { - try { - fs.rmSync(installDir, { recursive: true, force: true }); - done("removed"); - } catch (err) { - console.error(` could not remove: ${err.message}`); - console.error(" close any open Spent windows and retry."); - } - } else { - done("not present"); - } - } else { - console.error(" LOCALAPPDATA env var is not set; skipping."); - } - - step("Removing Startup shortcut"); - const appData = process.env.APPDATA; - if (appData) { - const shortcutPath = path.join( - appData, - "Microsoft", - "Windows", - "Start Menu", - "Programs", - "Startup", - "Spent.lnk", - ); - if (fs.existsSync(shortcutPath)) { - fs.unlinkSync(shortcutPath); - done("removed"); - } else { - done("not present"); - } - } else { - console.error(" APPDATA env var is not set; skipping."); - } -} - -function linuxUninstall() { - step("Linux: no menubar to remove (web-only platform)"); -} - -async function main() { - console.log("Spent uninstall"); - console.log(` platform: ${process.platform}`); - console.log(` repo: ${REPO_ROOT}`); - - uninstallService(); - - switch (process.platform) { - case "darwin": - await macUninstall(); - break; - case "win32": - windowsUninstall(); - break; - case "linux": - linuxUninstall(); - break; - } - - const dataDir = path.join(REPO_ROOT, "data"); - console.log(""); - console.log("Spent is uninstalled. Service and menubar are gone."); - console.log(""); - console.log("Kept (intentionally):"); - console.log(` ${dataDir} (your transactions + encryption key)`); - console.log(` ${REPO_ROOT} (the repo itself)`); - console.log(""); - console.log("To wipe everything, delete those directories manually."); -} - -main().catch((err) => { - console.error(`\nuninstall: ${err instanceof Error ? err.message : String(err)}`); - process.exit(1); -}); diff --git a/scripts/verify-landing.mjs b/scripts/verify-landing.mjs index a257f63..4e11842 100644 --- a/scripts/verify-landing.mjs +++ b/scripts/verify-landing.mjs @@ -3,148 +3,141 @@ * Requires the Astro dev server at http://localhost:4321/Spent to be running. */ -import puppeteer from 'puppeteer'; -import path from 'node:path'; -import fs from 'node:fs/promises'; +import fs from "node:fs/promises"; +import path from "node:path"; +import puppeteer from "puppeteer"; -const LANDING_URL = 'http://localhost:4321/Spent/'; -const OUT_DIR = path.resolve( - new URL('.', import.meta.url).pathname, - '../tmp-landing-shots' -); +const LANDING_URL = "http://localhost:4321/Spent/"; +const OUT_DIR = path.resolve(new URL(".", import.meta.url).pathname, "../tmp-landing-shots"); const VIEWPORT = { width: 1440, height: 900, deviceScaleFactor: 1 }; // Sections in the order they appear on the page const SECTION_SELECTORS = [ - { name: '01-hero', selector: '.hero' }, - { name: '01b-disclaimer', selector: '.disclaimer-band' }, - { name: '02-oss', selector: '.oss' }, - { name: '03-how', selector: '.how' }, - { name: '04-promises', selector: '.promises' }, - { name: '05-peek', selector: '.peek' }, - { name: '06-setup', selector: '.setup' }, - { name: '07-dark', selector: '.dark' }, - { name: '07b-bilingual', selector: '.bilingual' }, - { name: '08-features', selector: '.features' }, - { name: '09-banks', selector: '.banks' }, - { name: '10-install-cta', selector: '.install-cta' }, - { name: '11-footer', selector: '.footer' }, + { name: "01-hero", selector: ".hero" }, + { name: "01b-disclaimer", selector: ".disclaimer-band" }, + { name: "02-oss", selector: ".oss" }, + { name: "03-how", selector: ".how" }, + { name: "04-promises", selector: ".promises" }, + { name: "05-peek", selector: ".peek" }, + { name: "06-setup", selector: ".setup" }, + { name: "07-dark", selector: ".dark" }, + { name: "07b-bilingual", selector: ".bilingual" }, + { name: "08-features", selector: ".features" }, + { name: "09-banks", selector: ".banks" }, + { name: "10-install-cta", selector: ".install-cta" }, + { name: "11-footer", selector: ".footer" }, ]; const waitForImages = async (page) => { - await page.evaluate(async () => { - const imgs = Array.from(document.querySelectorAll('img')); - // Force any lazy images to load - imgs.forEach((img) => { - img.loading = 'eager'; - }); - await Promise.all( - imgs.map((img) => { - if (img.complete && img.naturalHeight > 0) return Promise.resolve(); - return new Promise((resolve) => { - img.addEventListener('load', resolve, { once: true }); - img.addEventListener('error', resolve, { once: true }); - // safety timeout - setTimeout(resolve, 5000); - }); - }) - ); - }); + await page.evaluate(async () => { + const imgs = Array.from(document.querySelectorAll("img")); + // Force any lazy images to load + imgs.forEach((img) => { + img.loading = "eager"; + }); + await Promise.all( + imgs.map((img) => { + if (img.complete && img.naturalHeight > 0) return Promise.resolve(); + return new Promise((resolve) => { + img.addEventListener("load", resolve, { once: true }); + img.addEventListener("error", resolve, { once: true }); + // safety timeout + setTimeout(resolve, 5000); + }); + }), + ); + }); }; (async () => { - await fs.mkdir(OUT_DIR, { recursive: true }); - - const browser = await puppeteer.launch({ - headless: 'new', - defaultViewport: VIEWPORT, - args: ['--no-sandbox', '--disable-setuid-sandbox'], - }); - - const consoleErrors = []; - - try { - const page = await browser.newPage(); - await page.setViewport(VIEWPORT); - - page.on('console', (msg) => { - if (msg.type() === 'error') consoleErrors.push(msg.text()); - }); - page.on('pageerror', (err) => consoleErrors.push(`pageerror: ${err.message}`)); - - await page.goto(LANDING_URL, { waitUntil: 'networkidle2', timeout: 30000 }); - - // Reveal all fade-up sections + hide Astro dev toolbar - await page.evaluate(() => { - document.querySelectorAll('.spent-fade-up').forEach((el) => - el.classList.add('is-visible') - ); - const css = document.createElement('style'); - css.textContent = ` + await fs.mkdir(OUT_DIR, { recursive: true }); + + const browser = await puppeteer.launch({ + headless: "new", + defaultViewport: VIEWPORT, + args: ["--no-sandbox", "--disable-setuid-sandbox"], + }); + + const consoleErrors = []; + + try { + const page = await browser.newPage(); + await page.setViewport(VIEWPORT); + + page.on("console", (msg) => { + if (msg.type() === "error") consoleErrors.push(msg.text()); + }); + page.on("pageerror", (err) => consoleErrors.push(`pageerror: ${err.message}`)); + + await page.goto(LANDING_URL, { waitUntil: "networkidle2", timeout: 30000 }); + + // Reveal all fade-up sections + hide Astro dev toolbar + await page.evaluate(() => { + document.querySelectorAll(".spent-fade-up").forEach((el) => el.classList.add("is-visible")); + const css = document.createElement("style"); + css.textContent = ` astro-dev-toolbar, astro-dev-overlay, [astro-dev-toolbar] { display: none !important; } `; - document.head.appendChild(css); - }); - - // Scroll the whole page to trigger lazy loads - await page.evaluate(async () => { - const total = document.documentElement.scrollHeight; - for (let y = 0; y < total; y += 400) { - window.scrollTo(0, y); - await new Promise((r) => setTimeout(r, 80)); - } - window.scrollTo(0, 0); - }); - - await waitForImages(page); - await new Promise((r) => setTimeout(r, 600)); - - // Full-page screenshot - const fullPath = path.join(OUT_DIR, '00-fullpage.png'); - await page.screenshot({ path: fullPath, fullPage: true }); - console.log(`✓ 00-fullpage.png`); - - // Section-by-section - const errors = []; - for (const sec of SECTION_SELECTORS) { - const el = await page.$(sec.selector); - if (!el) { - errors.push(`✗ ${sec.name}: selector "${sec.selector}" not found`); - continue; - } - // Scroll the section into view so any lazy images load - await el.evaluate((node) => { - node.scrollIntoView({ block: 'start' }); - }); - await new Promise((r) => setTimeout(r, 250)); - await waitForImages(page); - const dest = path.join(OUT_DIR, `${sec.name}.png`); - try { - await el.screenshot({ path: dest }); - console.log(`✓ ${sec.name}.png (${sec.selector})`); - } catch (e) { - errors.push(`✗ ${sec.name}: ${e.message}`); - } - } - - if (errors.length) { - console.log('\n=== Capture errors ==='); - console.log(errors.join('\n')); - } - - console.log( - `\n=== Console errors during run: ${consoleErrors.length} ===` - ); - if (consoleErrors.length) { - console.log(consoleErrors.slice(0, 10).join('\n')); - } - - console.log(`\nShots in: ${OUT_DIR}`); - } finally { - await browser.close(); - } + document.head.appendChild(css); + }); + + // Scroll the whole page to trigger lazy loads + await page.evaluate(async () => { + const total = document.documentElement.scrollHeight; + for (let y = 0; y < total; y += 400) { + window.scrollTo(0, y); + await new Promise((r) => setTimeout(r, 80)); + } + window.scrollTo(0, 0); + }); + + await waitForImages(page); + await new Promise((r) => setTimeout(r, 600)); + + // Full-page screenshot + const fullPath = path.join(OUT_DIR, "00-fullpage.png"); + await page.screenshot({ path: fullPath, fullPage: true }); + console.log(`✓ 00-fullpage.png`); + + // Section-by-section + const errors = []; + for (const sec of SECTION_SELECTORS) { + const el = await page.$(sec.selector); + if (!el) { + errors.push(`✗ ${sec.name}: selector "${sec.selector}" not found`); + continue; + } + // Scroll the section into view so any lazy images load + await el.evaluate((node) => { + node.scrollIntoView({ block: "start" }); + }); + await new Promise((r) => setTimeout(r, 250)); + await waitForImages(page); + const dest = path.join(OUT_DIR, `${sec.name}.png`); + try { + await el.screenshot({ path: dest }); + console.log(`✓ ${sec.name}.png (${sec.selector})`); + } catch (e) { + errors.push(`✗ ${sec.name}: ${e.message}`); + } + } + + if (errors.length) { + console.log("\n=== Capture errors ==="); + console.log(errors.join("\n")); + } + + console.log(`\n=== Console errors during run: ${consoleErrors.length} ===`); + if (consoleErrors.length) { + console.log(consoleErrors.slice(0, 10).join("\n")); + } + + console.log(`\nShots in: ${OUT_DIR}`); + } finally { + await browser.close(); + } })().catch((err) => { - console.error('Failed:', err); - process.exit(1); + console.error("Failed:", err); + process.exit(1); }); diff --git a/src/app/[locale]/budget/page.tsx b/src/app/[locale]/budget/page.tsx new file mode 100644 index 0000000..b9e4b60 --- /dev/null +++ b/src/app/[locale]/budget/page.tsx @@ -0,0 +1,26 @@ +import type { Metadata } from "next"; +import { redirect } from "next/navigation"; +import { getTranslations } from "next-intl/server"; +import { Dashboard } from "@/components/dashboard/dashboard"; +import { AppShell } from "@/components/layout/app-shell"; +import { anyWorkspaceHasBankCredentials } from "@/server/db/queries/bank-credentials"; + +export const dynamic = "force-dynamic"; + +export async function generateMetadata(): Promise { + const t = await getTranslations("nav"); + return { title: t("budget") }; +} + +export default async function BudgetPage({ params }: { params: Promise<{ locale: string }> }) { + if (!anyWorkspaceHasBankCredentials()) { + const { locale } = await params; + redirect(`/${locale}/setup`); + } + + return ( + + + + ); +} diff --git a/src/app/[locale]/chat/page.tsx b/src/app/[locale]/chat/page.tsx new file mode 100644 index 0000000..87d5ddd --- /dev/null +++ b/src/app/[locale]/chat/page.tsx @@ -0,0 +1,41 @@ +import { asc } from "drizzle-orm"; +import type { Metadata } from "next"; +import { redirect } from "next/navigation"; +import { getTranslations } from "next-intl/server"; +import { ChatClient } from "@/components/chat/chat-client"; +import { ChatDisabled } from "@/components/chat/chat-disabled"; +import { AppShell } from "@/components/layout/app-shell"; +import { getOrm } from "@/server/db/orm"; +import { anyWorkspaceHasBankCredentials } from "@/server/db/queries/bank-credentials"; +import { getAppSettings } from "@/server/db/queries/settings"; +import { workspaces } from "@/server/db/schema"; + +export const dynamic = "force-dynamic"; + +export async function generateMetadata(): Promise { + const t = await getTranslations("nav"); + return { title: t("chat") }; +} + +function firstWorkspaceId(): number { + const row = getOrm() + .select({ id: workspaces.id }) + .from(workspaces) + .orderBy(asc(workspaces.id)) + .limit(1) + .get(); + if (!row) throw new Error("No workspace exists"); + return row.id; +} + +export default async function ChatPage({ params }: { params: Promise<{ locale: string }> }) { + if (!anyWorkspaceHasBankCredentials()) { + const { locale } = await params; + redirect(`/${locale}/setup`); + } + + const settings = getAppSettings(firstWorkspaceId()); + const enabled = settings.aiProvider !== "none"; + + return {enabled ? : }; +} diff --git a/src/app/layout.tsx b/src/app/[locale]/layout.tsx similarity index 52% rename from src/app/layout.tsx rename to src/app/[locale]/layout.tsx index 0cd4ead..b536d9d 100644 --- a/src/app/layout.tsx +++ b/src/app/[locale]/layout.tsx @@ -1,13 +1,14 @@ -import type { Metadata } from "next"; -import { Inter, Fraunces, Geist_Mono } from "next/font/google"; -import { getLocale, getMessages } from "next-intl/server"; -import { I18nProvider } from "@/i18n/client-provider"; +import type { Metadata, Viewport } from "next"; +import { Fraunces, Geist_Mono, Inter } from "next/font/google"; +import { notFound } from "next/navigation"; +import { getMessages, setRequestLocale } from "next-intl/server"; +import { QueryProvider } from "@/components/query-provider"; import { ThemeProvider } from "@/components/theme-provider"; -import { TooltipProvider } from "@/components/ui/tooltip"; import { Toaster } from "@/components/ui/sonner"; -import { QueryProvider } from "@/components/query-provider"; -import { dirFor, isLocale, defaultLocale } from "@/i18n/routing"; -import "./globals.css"; +import { TooltipProvider } from "@/components/ui/tooltip"; +import { I18nProvider } from "@/i18n/client-provider"; +import { dirFor, isLocale, locales } from "@/i18n/routing"; +import "../globals.css"; const inter = Inter({ variable: "--font-sans", @@ -26,17 +27,47 @@ const geistMono = Geist_Mono({ }); export const metadata: Metadata = { - title: "Spent", - description: "Personal finance tracker with AI-powered categorization", + metadataBase: new URL("http://localhost:3000"), + title: { + default: "Spent", + template: "%s · Spent", + }, + applicationName: "Spent", + description: + "A private, local-first personal finance tracker for Israeli banks. AI-powered categorization, with credentials that never leave your machine. Open source and self-hosted.", + openGraph: { + title: "Spent", + description: + "A private, local-first personal finance tracker for Israeli banks. Open source and self-hosted.", + siteName: "Spent", + type: "website", + }, +}; + +export const viewport: Viewport = { + colorScheme: "light dark", + themeColor: [ + { media: "(prefers-color-scheme: light)", color: "#fcf7ed" }, + { media: "(prefers-color-scheme: dark)", color: "#1c1a17" }, + ], }; +export function generateStaticParams() { + return locales.map((locale) => ({ locale })); +} + export default async function RootLayout({ children, + params, }: Readonly<{ children: React.ReactNode; + params: Promise<{ locale: string }>; }>) { - const rawLocale = await getLocale(); - const locale = isLocale(rawLocale) ? rawLocale : defaultLocale; + const { locale } = await params; + if (!isLocale(locale)) { + notFound(); + } + setRequestLocale(locale); const messages = await getMessages(); const dir = dirFor(locale); return ( diff --git a/src/app/[locale]/page.tsx b/src/app/[locale]/page.tsx new file mode 100644 index 0000000..ca8344b --- /dev/null +++ b/src/app/[locale]/page.tsx @@ -0,0 +1,19 @@ +import { redirect } from "next/navigation"; +import { HomePage } from "@/components/home/home-page"; +import { AppShell } from "@/components/layout/app-shell"; +import { anyWorkspaceHasBankCredentials } from "@/server/db/queries/bank-credentials"; + +export const dynamic = "force-dynamic"; + +export default async function Home({ params }: { params: Promise<{ locale: string }> }) { + if (!anyWorkspaceHasBankCredentials()) { + const { locale } = await params; + redirect(`/${locale}/setup`); + } + + return ( + + + + ); +} diff --git a/src/app/settings/ai/page.tsx b/src/app/[locale]/settings/ai/page.tsx similarity index 100% rename from src/app/settings/ai/page.tsx rename to src/app/[locale]/settings/ai/page.tsx diff --git a/src/app/settings/appearance/page.tsx b/src/app/[locale]/settings/appearance/page.tsx similarity index 57% rename from src/app/settings/appearance/page.tsx rename to src/app/[locale]/settings/appearance/page.tsx index 27542e0..c195413 100644 --- a/src/app/settings/appearance/page.tsx +++ b/src/app/[locale]/settings/appearance/page.tsx @@ -1,47 +1,25 @@ "use client"; -import { useRouter } from "next/navigation"; -import { useTheme } from "next-themes"; -import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import { useLocale, useTranslations } from "next-intl"; -import { toast } from "sonner"; -import { useIsHydrated } from "@/hooks/use-is-hydrated"; +import { useTheme } from "next-themes"; import { SectionShell, SettingCard } from "@/components/settings/section-shell"; -import { getSettings, updateSettings } from "@/lib/api"; -import type { AppSettings } from "@/lib/types"; +import { useIsHydrated } from "@/hooks/use-is-hydrated"; +import { usePathname, useRouter } from "@/i18n/navigation"; type Lang = "en" | "he"; export default function AppearanceSettingsPage() { const t = useTranslations("settings.appearance"); - const tCommon = useTranslations("common"); const { theme, setTheme } = useTheme(); const hydrated = useIsHydrated(); const active = hydrated ? (theme ?? "system") : null; const router = useRouter(); - const queryClient = useQueryClient(); + const pathname = usePathname(); + // The active language lives in the URL (/en, /he); next-intl persists the + // choice to the NEXT_LOCALE cookie on navigation. No database involved. const locale = useLocale() as Lang; - const { data: settings } = useQuery({ - queryKey: ["settings"], - queryFn: getSettings, - }); - const storedLang: Lang = settings?.language ?? locale; - - const mutation = useMutation({ - mutationFn: (next: Lang) => - updateSettings({ language: next } as Partial), - onSuccess: () => { - queryClient.invalidateQueries({ queryKey: ["settings"] }); - toast.success(tCommon("saved")); - router.refresh(); - }, - onError: (err) => { - toast.error(err instanceof Error ? err.message : tCommon("tryAgain")); - }, - }); - const themeOptions = [ { value: "light" as const, label: t("themeLight"), description: t("themeLightDesc") }, { value: "dark" as const, label: t("themeDark"), description: t("themeDarkDesc") }, @@ -64,15 +42,11 @@ export default function AppearanceSettingsPage() { key={o.value} onClick={() => setTheme(o.value)} className={`rounded-xl border p-4 text-start transition-colors ${ - isActive - ? "border-primary bg-primary/5" - : "border-border hover:border-primary/50" + isActive ? "border-primary bg-primary/5" : "border-border hover:border-primary/50" }`} >
{o.label}
-
- {o.description} -
+
{o.description}
); })} @@ -82,31 +56,21 @@ export default function AppearanceSettingsPage() {
{langOptions.map((o) => { - const isActive = storedLang === o.value; - const isPending = mutation.isPending && mutation.variables === o.value; + const isActive = locale === o.value; return ( ); })} diff --git a/src/app/settings/bank/page.tsx b/src/app/[locale]/settings/bank/page.tsx similarity index 88% rename from src/app/settings/bank/page.tsx rename to src/app/[locale]/settings/bank/page.tsx index 4839e8a..c0bb5c9 100644 --- a/src/app/settings/bank/page.tsx +++ b/src/app/[locale]/settings/bank/page.tsx @@ -1,16 +1,13 @@ "use client"; -import { useMemo, useState } from "react"; import { useQuery } from "@tanstack/react-query"; +import { ChevronRight, CircleAlert, CircleCheck, Loader2, Plus, RefreshCw } from "lucide-react"; import { useTranslations } from "next-intl"; -import { - ChevronRight, - Plus, - RefreshCw, - Loader2, - CircleCheck, - CircleAlert, -} from "lucide-react"; +import { useMemo, useState } from "react"; +import { BankDetailSheet } from "@/components/settings/bank-detail-sheet"; +import { SectionShell } from "@/components/settings/section-shell"; +import { useBankSync } from "@/components/settings/use-bank-sync"; +import { ProviderBadge } from "@/components/setup/provider-badge"; import { Button } from "@/components/ui/button"; import { DropdownMenu, @@ -18,15 +15,11 @@ import { DropdownMenuItem, DropdownMenuTrigger, } from "@/components/ui/dropdown-menu"; -import { ProviderBadge } from "@/components/setup/provider-badge"; -import { SectionShell } from "@/components/settings/section-shell"; -import { BankDetailSheet } from "@/components/settings/bank-detail-sheet"; -import { useBankSync } from "@/components/settings/use-bank-sync"; import { listIntegrations } from "@/lib/api"; -import { BANK_PROVIDERS } from "@/lib/types"; -import { translateProviderName, useFormatterLabels } from "@/lib/i18n-data"; import { formatLastSync } from "@/lib/formatters"; +import { translateProviderName, useFormatterLabels } from "@/lib/i18n-data"; import type { Integration } from "@/lib/types"; +import { BANK_PROVIDERS } from "@/lib/types"; interface SheetState { open: boolean; @@ -53,9 +46,7 @@ export default function BankSettingsPage() { }); const lastSync = useMemo(() => { - const stamps = integrations - .map((i) => i.lastSyncAt) - .filter((s): s is string => Boolean(s)); + const stamps = integrations.map((i) => i.lastSyncAt).filter((s): s is string => Boolean(s)); if (stamps.length === 0) return null; return stamps.sort().slice(-1)[0]; }, [integrations]); @@ -68,7 +59,7 @@ export default function BankSettingsPage() { const sheetIntegration: Integration | null = sheet.mode === "edit" && sheet.credentialId != null - ? integrations.find((i) => i.id === sheet.credentialId) ?? null + ? (integrations.find((i) => i.id === sheet.credentialId) ?? null) : null; const countLabel = @@ -126,10 +117,7 @@ export default function BankSettingsPage() { } > - + {translateProviderName(b.id, b.name, tBanks)} @@ -150,9 +138,7 @@ export default function BankSettingsPage() {
    {integrations.map((integration) => { - const info = BANK_PROVIDERS.find( - (b) => b.id === integration.provider - ); + const info = BANK_PROVIDERS.find((b) => b.id === integration.provider); if (!info) return null; const sync = stateFor(integration.id); const localName = translateProviderName(info.id, info.name, tBanks); @@ -194,10 +180,7 @@ export default function BankSettingsPage() {
    {integration.label} - +
    {localName} @@ -241,13 +224,7 @@ export default function BankSettingsPage() { ); } -function StatusPill({ - lastSyncAt, - syncing, -}: { - lastSyncAt: string | null; - syncing: boolean; -}) { +function StatusPill({ lastSyncAt, syncing }: { lastSyncAt: string | null; syncing: boolean }) { const t = useTranslations("settings.bank"); if (syncing) { return ( @@ -258,13 +235,13 @@ function StatusPill({ } if (!lastSyncAt) { return ( - + {t("statusNeverSynced")} ); } return ( - + {t("statusConnected")} ); @@ -289,12 +266,7 @@ function SyncShortButton({ ); } return ( - diff --git a/src/app/settings/categories/page.tsx b/src/app/[locale]/settings/categories/page.tsx similarity index 78% rename from src/app/settings/categories/page.tsx rename to src/app/[locale]/settings/categories/page.tsx index c0ef93f..7c1dd8a 100644 --- a/src/app/settings/categories/page.tsx +++ b/src/app/[locale]/settings/categories/page.tsx @@ -1,10 +1,12 @@ "use client"; -import { useMemo, useState } from "react"; import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import { ChevronRight, Plus, Search } from "lucide-react"; +import { useLocale, useTranslations } from "next-intl"; +import { useMemo, useState } from "react"; import { toast } from "sonner"; -import { Input } from "@/components/ui/input"; +import { CategoryDetailSheet } from "@/components/settings/category-detail-sheet"; +import { SectionShell } from "@/components/settings/section-shell"; import { Button } from "@/components/ui/button"; import { Dialog, @@ -14,6 +16,7 @@ import { DialogTitle, DialogTrigger, } from "@/components/ui/dialog"; +import { Input } from "@/components/ui/input"; import { Label } from "@/components/ui/label"; import { Select, @@ -22,17 +25,14 @@ import { SelectTrigger, SelectValue, } from "@/components/ui/select"; -import { SectionShell } from "@/components/settings/section-shell"; -import { CategoryDetailSheet } from "@/components/settings/category-detail-sheet"; -import { getMonthRange } from "@/lib/formatters"; -import { - createCategory, - getCategories, - getSummary, -} from "@/lib/api"; +import type { Locale } from "@/i18n/routing"; +import { createCategory, getCategories, getSummary } from "@/lib/api"; +import { formatCurrency, getMonthRange } from "@/lib/formatters"; import type { Category, CategoryKind, CategoryWithData } from "@/lib/types"; export default function CategoriesSettingsPage() { + const t = useTranslations("settings.categories"); + const tc = useTranslations("common"); const { from, to } = getMonthRange(); const { data: categories } = useQuery({ queryKey: ["categories"], @@ -55,13 +55,11 @@ export default function CategoriesSettingsPage() { const filtered = useMemo(() => { if (!categories) return []; - return categories - .filter((c) => c.kind === activeKind) - .filter((c) => - search.trim().length === 0 - ? true - : c.name.toLowerCase().includes(search.toLowerCase()) - ); + const needle = search.trim().toLowerCase(); + return categories.filter( + (c) => + c.kind === activeKind && (needle.length === 0 || c.name.toLowerCase().includes(needle)), + ); }, [categories, activeKind, search]); const { parents, childrenByParent, orphans } = useMemo(() => { @@ -80,9 +78,7 @@ export default function CategoriesSettingsPage() { childrenByParent.set(c.parentId, list); } }); - childrenByParent.forEach((list) => - list.sort((a, b) => a.name.localeCompare(b.name)) - ); + childrenByParent.forEach((list) => list.sort((a, b) => a.name.localeCompare(b.name))); const orphans = filtered .filter((c) => c.parentId == null && !parentIds.has(c.id)) .sort((a, b) => a.name.localeCompare(b.name)); @@ -91,23 +87,14 @@ export default function CategoriesSettingsPage() { return ( <> - +
    - setActiveKind("expense")} - > - Expense + setActiveKind("expense")}> + {t("tabExpense")} - setActiveKind("income")} - > - Income + setActiveKind("income")}> + {t("tabIncome")}
    @@ -115,7 +102,7 @@ export default function CategoriesSettingsPage() { setSearch(e.target.value)} - placeholder="Search categories…" + placeholder={t("searchPlaceholder")} className="ps-8" />
    @@ -124,11 +111,11 @@ export default function CategoriesSettingsPage() { {!categories ? (
    - Loading… + {tc("loading")}
    ) : filtered.length === 0 ? (
    - No matching categories. + {t("noMatching")}
    ) : (
    @@ -153,8 +140,8 @@ export default function CategoriesSettingsPage() { ))} {orphans.length > 0 ? ( @@ -194,9 +181,7 @@ function KindTab({ ) : null}
    @@ -254,6 +237,8 @@ function CategoryRow({ data: CategoryWithData | null; onSelect: () => void; }) { + const t = useTranslations("settings.categories"); + const locale = useLocale() as Locale; const description = category.description?.trim(); return (
  • @@ -271,15 +256,13 @@ function CategoryRow({ {category.name}
  • {description ? ( -
    - {description} -
    +
    {description}
    ) : null}
    {data ? (
    - ₪{Math.round(data.spent).toLocaleString("en-IL")} spent + {t("spentLabel", { amount: formatCurrency(Math.round(data.spent), "ILS", locale) })}
    ) : null}
    @@ -290,35 +273,33 @@ function CategoryRow({ ); } -function BudgetChip({ - category, - data, -}: { - category: Category; - data: CategoryWithData | null; -}) { +function BudgetChip({ category, data }: { category: Category; data: CategoryWithData | null }) { + const t = useTranslations("settings.categories"); + const locale = useLocale() as Locale; if (category.budgetMode === "tracking") { return ( - tracking + {t("tracking")} ); } if (!data || data.budget <= 0) { return ( - no budget + {t("noBudget")} ); } return ( - ₪{Math.round(data.budget).toLocaleString("en-IL")} + {formatCurrency(Math.round(data.budget), "ILS", locale)} ); } function NewGroupDialog({ kind }: { kind: CategoryKind }) { + const t = useTranslations("settings.categories"); + const tc = useTranslations("common"); const queryClient = useQueryClient(); const [open, setOpen] = useState(false); const [name, setName] = useState(""); @@ -334,12 +315,12 @@ function NewGroupDialog({ kind }: { kind: CategoryKind }) { onSuccess: () => { queryClient.invalidateQueries({ queryKey: ["categories"] }); queryClient.invalidateQueries({ queryKey: ["summary"] }); - toast.success(`Created "${name.trim()}"`); + toast.success(t("createdToast", { name: name.trim() })); setName(""); setOpen(false); }, onError: (err: Error) => { - toast.error(err.message || "Couldn't create group"); + toast.error(err.message || t("createGroupFailed")); }, }); @@ -355,22 +336,22 @@ function NewGroupDialog({ kind }: { kind: CategoryKind }) { render={ } /> - New parent group + {t("newGroupDialogTitle")}
    - + setName(e.target.value)} - placeholder="e.g. Food, Transport, Lifestyle" + placeholder={t("newGroupNamePlaceholder")} autoFocus onKeyDown={(e) => { if (e.key === "Enter" && name.trim().length > 0) { @@ -380,30 +361,27 @@ function NewGroupDialog({ kind }: { kind: CategoryKind }) { />
    - - v && setK(v as CategoryKind)}> - Expense - Income + {t("tabExpense")} + {t("tabIncome")}
    diff --git a/src/app/settings/data/page.tsx b/src/app/[locale]/settings/data/page.tsx similarity index 100% rename from src/app/settings/data/page.tsx rename to src/app/[locale]/settings/data/page.tsx diff --git a/src/app/settings/general/page.tsx b/src/app/[locale]/settings/general/page.tsx similarity index 74% rename from src/app/settings/general/page.tsx rename to src/app/[locale]/settings/general/page.tsx index 7831fc3..7537592 100644 --- a/src/app/settings/general/page.tsx +++ b/src/app/[locale]/settings/general/page.tsx @@ -1,9 +1,14 @@ "use client"; -import { useState } from "react"; import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import { useLocale, useTranslations } from "next-intl"; +import { useState } from "react"; import { toast } from "sonner"; +import { SectionShell, SettingCard } from "@/components/settings/section-shell"; +import { WorkspaceNameCard } from "@/components/settings/workspace-controls"; +import { Button } from "@/components/ui/button"; +import { Input, InputGroup } from "@/components/ui/input"; +import { Label } from "@/components/ui/label"; import { Select, SelectContent, @@ -11,15 +16,10 @@ import { SelectTrigger, SelectValue, } from "@/components/ui/select"; -import { Button } from "@/components/ui/button"; -import { Input, InputGroup } from "@/components/ui/input"; -import { Label } from "@/components/ui/label"; import { Switch } from "@/components/ui/switch"; -import { SectionShell, SettingCard } from "@/components/settings/section-shell"; -import { WorkspaceNameCard } from "@/components/settings/workspace-controls"; +import type { Locale } from "@/i18n/routing"; import { getSettings, getSummary, updateSettings } from "@/lib/api"; import { formatCurrency } from "@/lib/formatters"; -import type { Locale } from "@/i18n/routing"; function todayLocalISO(): string { const d = new Date(); @@ -58,7 +58,7 @@ export default function GeneralSettingsPage() { typicalMonthly={summary?.typicalMonthly ?? null} /> @@ -67,6 +67,10 @@ export default function GeneralSettingsPage() { initialEnabled={settings.autoSyncEnabled} initialTime={settings.autoSyncTime} /> + ) : ( @@ -88,9 +92,7 @@ function MonthlyTargetCard({ const tCommon = useTranslations("common"); const locale = useLocale() as Locale; const queryClient = useQueryClient(); - const [value, setValue] = useState( - initialTarget != null ? String(initialTarget) : "" - ); + const [value, setValue] = useState(initialTarget != null ? String(initialTarget) : ""); const mutation = useMutation({ mutationFn: updateSettings, @@ -103,15 +105,11 @@ function MonthlyTargetCard({ const parsed = value.trim() === "" ? null : Number(value); const valid = parsed == null || (Number.isFinite(parsed) && parsed >= 0); - const dirty = - (parsed ?? null) !== (initialTarget ?? null) && valid; + const dirty = (parsed ?? null) !== (initialTarget ?? null) && valid; return (
    - +
    @@ -134,16 +132,12 @@ function MonthlyTargetCard({ })}

    ) : ( -

    - {t("noPriorHistory")} -

    +

    {t("noPriorHistory")}

    )}
    +
    +
    + ); +} + +function AtmCard({ initialEnabled }: { initialEnabled: boolean }) { + const t = useTranslations("settings.general"); + const tCommon = useTranslations("common"); + const queryClient = useQueryClient(); + const [enabled, setEnabled] = useState(initialEnabled); + + const mutation = useMutation({ + mutationFn: updateSettings, + onSuccess: () => { + queryClient.invalidateQueries({ queryKey: ["settings"] }); + toast.success(tCommon("saved")); + }, + onError: (err) => { + toast.error(err instanceof Error ? err.message : t("failedToSave")); + }, + }); + + const dirty = enabled !== initialEnabled; + + return ( + +
    +
    + +

    {t("atmToggleHint")}

    +
    + +
    +
    + + + } + /> + +
    +
    +
    + {messages.length === 0 && ( + { + clearError(); + setInput(""); + sendMessage({ text: s }); + }} + busy={isBusy} + /> + )} + + {messages.map((message) => ( + + ))} + + {status === "submitted" && ( +
    + + {t("thinking")} +
    + )} + + {error && ( +
    + {t("error")} +
    + )} +
    +
    + +
    +
    { + e.preventDefault(); + submit(); + }} + className="mx-auto max-w-3xl" + > +
    +