diff --git a/AGENTS.md b/AGENTS.md
index c0ac4196..8a8ee2d0 100644
--- a/AGENTS.md
+++ b/AGENTS.md
@@ -27,6 +27,7 @@ After making changes in a specific package, run its check script:
| `apps/server` | `bun run check:server` | `bun run format:server` |
| `apps/sandbox` | `bun run check:sandbox` | `bun run format:sandbox` |
| `apps/server-simple` | `bun run check:server-simple` | `bun run format:server-simple` |
+| `packages/convex` | `bun run check:convex` | `bun run format:convex` |
| `packages/shared` | `bun run check:shared` | `bun run format:shared` |
## Code Style
diff --git a/README.md b/README.md
index 286e0c03..3e859a86 100644
--- a/README.md
+++ b/README.md
@@ -42,6 +42,8 @@ This is a Bun monorepo using Turborepo. **Only use `bun`** - never npm/yarn.
```bash
bun install
bun run dev:web # Start web app with Convex
+bun run dev:desktop # Start desktop app, web app, and Convex
+bun run dev:convex # Start Convex only
bun run dev:cli # Start CLI in watch mode
bun run server # Start server in watch mode
bun run cli # Run CLI (no watch)
@@ -70,6 +72,7 @@ All scripts use Turborepo for caching. Run from the repository root.
| `bun run check:all` | Type check all packages |
| `bun run check:cli` | Type check CLI |
| `bun run check:web` | Type check web app |
+| `bun run check:convex` | Type check Convex package |
| `bun run check:server` | Type check server |
| `bun run check:sandbox` | Type check sandbox |
| `bun run check:shared` | Type check shared package |
@@ -81,6 +84,7 @@ All scripts use Turborepo for caching. Run from the repository root.
| `bun run format:all` | Format all packages |
| `bun run format:cli` | Format CLI |
| `bun run format:web` | Format web app |
+| `bun run format:convex` | Format Convex package |
| `bun run format:server` | Format server |
| `bun run format:sandbox` | Format sandbox |
| `bun run format:shared` | Format shared package |
@@ -102,14 +106,16 @@ All scripts use Turborepo for caching. Run from the repository root.
### Packages
-| Package | Path | Description |
-| ----------------------- | ---------------------- | ---------------------------- |
-| `btca` | `apps/cli` | CLI tool |
-| `btca-server` | `apps/server` | API server |
-| `@btca/web` | `apps/web` | Web app (SvelteKit + Convex) |
-| `btca-sandbox` | `apps/sandbox` | Sandbox environment |
-| `@btca/shared` | `packages/shared` | Shared utilities |
-| `@btca/analytics-proxy` | `apps/analytics-proxy` | PostHog analytics proxy |
+| Package | Path | Description |
+| ----------------------- | ---------------------- | ------------------------------- |
+| `btca` | `apps/cli` | CLI tool |
+| `btca-server` | `apps/server` | API server |
+| `@btca/web` | `apps/web` | Web app (SvelteKit) |
+| `@btca/desktop` | `apps/desktop` | Desktop app (Tauri + SvelteKit) |
+| `@btca/convex` | `packages/convex` | Shared Convex backend |
+| `btca-sandbox` | `apps/sandbox` | Sandbox environment |
+| `@btca/shared` | `packages/shared` | Shared utilities |
+| `@btca/analytics-proxy` | `apps/analytics-proxy` | PostHog analytics proxy |
## model recs...
diff --git a/apps/desktop/.gitignore b/apps/desktop/.gitignore
new file mode 100644
index 00000000..745b4d4e
--- /dev/null
+++ b/apps/desktop/.gitignore
@@ -0,0 +1,13 @@
+node_modules
+
+# Output
+/.svelte-kit
+/build
+
+# Rust
+/src-tauri/target
+
+# Env
+.env
+.env.*
+!.env.example
diff --git a/apps/desktop/README.md b/apps/desktop/README.md
new file mode 100644
index 00000000..ff6be50e
--- /dev/null
+++ b/apps/desktop/README.md
@@ -0,0 +1,41 @@
+# @btca/desktop
+
+Bundled Tauri + SvelteKit desktop frontend for btca.
+
+## How it works
+
+- The desktop app now ships its own SvelteKit frontend inside Tauri
+- App UI code is copied from `apps/web` and kept close to the web app's authenticated `/app` experience
+- Server-backed flows still go over HTTP to the web app server via `PUBLIC_BACKEND_BASE_URL`
+- In development, `PUBLIC_BACKEND_BASE_URL` defaults to `http://localhost:5173`
+- Shared public env for Clerk and Convex is loaded from the `apps/web` env directory so the desktop app can stay aligned with the current stack
+
+## Dev
+
+1. Install workspace dependencies from the repo root:
+
+```sh
+bun install
+```
+
+2. Start the full desktop stack from the repo root:
+
+```sh
+bun run dev:desktop
+```
+
+3. `dev:desktop` runs Convex, the web app, and the Tauri desktop app together.
+
+4. The desktop frontend dev server runs on `http://localhost:1420`, and server-backed flows target `PUBLIC_BACKEND_BASE_URL`.
+
+## Env
+
+- `PUBLIC_BACKEND_BASE_URL`
+ Defaults to `http://localhost:5173` in dev
+ Falls back to `https://btca.dev` outside dev
+ Set this for non-default backend targets
+
+## Notes
+
+- The desktop app no longer embeds the hosted site
+- It still depends on the web server for MCP URLs, billing redirects, and any other same-origin HTTP flows that are not bundled into the desktop frontend
diff --git a/apps/desktop/package.json b/apps/desktop/package.json
new file mode 100644
index 00000000..0753b30a
--- /dev/null
+++ b/apps/desktop/package.json
@@ -0,0 +1,49 @@
+{
+ "name": "@btca/desktop",
+ "private": true,
+ "version": "0.0.1",
+ "type": "module",
+ "description": "Tauri desktop shell for the btca web app",
+ "scripts": {
+ "prepare": "svelte-kit sync || echo ''",
+ "dev:ui": "vite dev --host 127.0.0.1 --port 1420 --strictPort",
+ "build:ui": "vite build",
+ "dev": "tauri dev",
+ "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json && cargo check --manifest-path src-tauri/Cargo.toml",
+ "format": "prettier --plugin prettier-plugin-svelte --plugin prettier-plugin-tailwindcss --write package.json README.md svelte.config.js tsconfig.json vite.config.ts src src-tauri/tauri.conf.json src-tauri/capabilities/default.json && cargo fmt --manifest-path src-tauri/Cargo.toml --all"
+ },
+ "devDependencies": {
+ "@sveltejs/adapter-static": "^3.0.8",
+ "@sveltejs/kit": "^2.50.2",
+ "@sveltejs/vite-plugin-svelte": "^6.2.4",
+ "@tailwindcss/typography": "^0.5.19",
+ "@tailwindcss/vite": "^4.1.18",
+ "@tauri-apps/cli": "^2.10.1",
+ "prettier": "^3.8.1",
+ "prettier-plugin-svelte": "^3.4.1",
+ "prettier-plugin-tailwindcss": "^0.7.2",
+ "svelte": "^5.50.2",
+ "svelte-check": "^4.3.6",
+ "tailwindcss": "^4.1.18",
+ "typescript": "^5.9.3",
+ "vite": "^7.3.1"
+ },
+ "dependencies": {
+ "@btca/convex": "workspace:*",
+ "@btca/shared": "workspace:*",
+ "@clerk/clerk-js": "^5.122.1",
+ "@clerk/types": "^4.101.14",
+ "@lucide/svelte": "^0.562.0",
+ "@shikijs/langs": "^3.22.0",
+ "@shikijs/themes": "^3.22.0",
+ "better-result": "^2.7.0",
+ "convex": "^1.31.7",
+ "convex-svelte": "^0.0.12",
+ "isomorphic-dompurify": "^2.36.0",
+ "marked": "^17.0.2",
+ "nanoid": "^5.1.6",
+ "posthog-js": "^1.345.5",
+ "shiki": "^3.22.0",
+ "zod": "^4.3.6"
+ }
+}
diff --git a/apps/desktop/src-tauri/.gitignore b/apps/desktop/src-tauri/.gitignore
new file mode 100644
index 00000000..502406b4
--- /dev/null
+++ b/apps/desktop/src-tauri/.gitignore
@@ -0,0 +1,4 @@
+# Generated by Cargo
+# will have compiled files and executables
+/target/
+/gen/schemas
diff --git a/apps/desktop/src-tauri/Cargo.lock b/apps/desktop/src-tauri/Cargo.lock
new file mode 100644
index 00000000..ce395f32
--- /dev/null
+++ b/apps/desktop/src-tauri/Cargo.lock
@@ -0,0 +1,5022 @@
+# This file is automatically @generated by Cargo.
+# It is not intended for manual editing.
+version = 3
+
+[[package]]
+name = "adler2"
+version = "2.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "320119579fcad9c21884f5c4861d16174d0e06250625266f50fe6898340abefa"
+
+[[package]]
+name = "ahash"
+version = "0.7.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "891477e0c6a8957309ee5c45a6368af3ae14bb510732d2684ffa19af310920f9"
+dependencies = [
+ "getrandom 0.2.17",
+ "once_cell",
+ "version_check",
+]
+
+[[package]]
+name = "aho-corasick"
+version = "1.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ddd31a130427c27518df266943a5308ed92d4b226cc639f5a8f1002816174301"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "alloc-no-stdlib"
+version = "2.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cc7bb162ec39d46ab1ca8c77bf72e890535becd1751bb45f64c597edb4c8c6b3"
+
+[[package]]
+name = "alloc-stdlib"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94fb8275041c72129eb51b7d0322c29b8387a0386127718b096429201a5d6ece"
+dependencies = [
+ "alloc-no-stdlib",
+]
+
+[[package]]
+name = "android_log-sys"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "84521a3cf562bc62942e294181d9eef17eb38ceb8c68677bc49f144e4c3d4f8d"
+
+[[package]]
+name = "android_logger"
+version = "0.15.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dbb4e440d04be07da1f1bf44fb4495ebd58669372fe0cffa6e48595ac5bd88a3"
+dependencies = [
+ "android_log-sys",
+ "env_filter",
+ "log",
+]
+
+[[package]]
+name = "android_system_properties"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "anyhow"
+version = "1.0.102"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f202df86484c868dbad7eaa557ef785d5c66295e41b460ef922eca0723b842c"
+
+[[package]]
+name = "arrayvec"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
+
+[[package]]
+name = "atk"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "241b621213072e993be4f6f3a9e4b45f65b7e6faad43001be957184b7bb1824b"
+dependencies = [
+ "atk-sys",
+ "glib",
+ "libc",
+]
+
+[[package]]
+name = "atk-sys"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c5e48b684b0ca77d2bbadeef17424c2ea3c897d44d566a1617e7e8f30614d086"
+dependencies = [
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "atomic-waker"
+version = "1.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0"
+
+[[package]]
+name = "autocfg"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8"
+
+[[package]]
+name = "base64"
+version = "0.21.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567"
+
+[[package]]
+name = "base64"
+version = "0.22.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6"
+
+[[package]]
+name = "bitflags"
+version = "1.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
+
+[[package]]
+name = "bitflags"
+version = "2.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "843867be96c8daad0d758b57df9392b6d8d271134fce549de6ce169ff98a92af"
+dependencies = [
+ "serde_core",
+]
+
+[[package]]
+name = "bitvec"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bc2832c24239b0141d5674bb9174f9d68a8b5b3f2753311927c172ca46f7e9c"
+dependencies = [
+ "funty",
+ "radium",
+ "tap",
+ "wyz",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "block2"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cdeb9d870516001442e364c5220d3574d2da8dc765554b4a617230d33fa58ef5"
+dependencies = [
+ "objc2",
+]
+
+[[package]]
+name = "borsh"
+version = "1.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d1da5ab77c1437701eeff7c88d968729e7766172279eab0676857b3d63af7a6f"
+dependencies = [
+ "borsh-derive",
+ "cfg_aliases",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "1.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0686c856aa6aac0c4498f936d7d6a02df690f614c03e4d906d1018062b5c5e2c"
+dependencies = [
+ "once_cell",
+ "proc-macro-crate 3.5.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "brotli"
+version = "8.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4bd8b9603c7aa97359dbd97ecf258968c95f3adddd6db2f7e7a5bef101c84560"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+ "brotli-decompressor",
+]
+
+[[package]]
+name = "brotli-decompressor"
+version = "5.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "874bb8112abecc98cbd6d81ea4fa7e94fb9449648c93cc89aa40c81c24d7de03"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+]
+
+[[package]]
+name = "btca-desktop"
+version = "0.0.1"
+dependencies = [
+ "log",
+ "tauri",
+ "tauri-build",
+ "tauri-plugin-log",
+]
+
+[[package]]
+name = "bumpalo"
+version = "3.20.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5d20789868f4b01b2f2caec9f5c4e0213b41e3e5702a50157d699ae31ced2fcb"
+
+[[package]]
+name = "byte-unit"
+version = "5.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c6d47a4e2961fb8721bcfc54feae6455f2f64e7054f9bc67e875f0e77f4c58d"
+dependencies = [
+ "rust_decimal",
+ "schemars 1.2.1",
+ "serde",
+ "utf8-width",
+]
+
+[[package]]
+name = "bytecheck"
+version = "0.6.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "23cdc57ce23ac53c931e88a43d06d070a6fd142f2617be5855eb75efc9beb1c2"
+dependencies = [
+ "bytecheck_derive",
+ "ptr_meta",
+ "simdutf8",
+]
+
+[[package]]
+name = "bytecheck_derive"
+version = "0.6.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3db406d29fbcd95542e92559bed4d8ad92636d1ca8b3b72ede10b4bcc010e659"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "bytemuck"
+version = "1.25.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c8efb64bd706a16a1bdde310ae86b351e4d21550d98d056f22f8a7f7a2183fec"
+
+[[package]]
+name = "byteorder"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
+
+[[package]]
+name = "bytes"
+version = "1.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e748733b7cbc798e1434b6ac524f0c1ff2ab456fe201501e6497c8417a4fc33"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "cairo-rs"
+version = "0.18.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2"
+dependencies = [
+ "bitflags 2.11.0",
+ "cairo-sys-rs",
+ "glib",
+ "libc",
+ "once_cell",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "cairo-sys-rs"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "685c9fa8e590b8b3d678873528d83411db17242a73fccaed827770ea0fedda51"
+dependencies = [
+ "glib-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "camino"
+version = "1.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e629a66d692cb9ff1a1c664e41771b3dcaf961985a9774c0eb0bd1b51cf60a48"
+dependencies = [
+ "serde_core",
+]
+
+[[package]]
+name = "cargo-platform"
+version = "0.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e35af189006b9c0f00a064685c727031e3ed2d8020f7ba284d78cc2671bd36ea"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "cargo_metadata"
+version = "0.19.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dd5eb614ed4c27c5d706420e4320fbe3216ab31fa1c33cd8246ac36dae4479ba"
+dependencies = [
+ "camino",
+ "cargo-platform",
+ "semver",
+ "serde",
+ "serde_json",
+ "thiserror 2.0.18",
+]
+
+[[package]]
+name = "cargo_toml"
+version = "0.22.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "374b7c592d9c00c1f4972ea58390ac6b18cbb6ab79011f3bdc90a0b82ca06b77"
+dependencies = [
+ "serde",
+ "toml 0.9.12+spec-1.1.0",
+]
+
+[[package]]
+name = "cc"
+version = "1.2.56"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aebf35691d1bfb0ac386a69bac2fde4dd276fb618cf8bf4f5318fe285e821bb2"
+dependencies = [
+ "find-msvc-tools",
+ "shlex",
+]
+
+[[package]]
+name = "cesu8"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6d43a04d8753f35258c91f8ec639f792891f748a1edbd759cf1dcea3382ad83c"
+
+[[package]]
+name = "cfb"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d38f2da7a0a2c4ccf0065be06397cc26a81f4e528be095826eee9d4adbb8c60f"
+dependencies = [
+ "byteorder",
+ "fnv",
+ "uuid",
+]
+
+[[package]]
+name = "cfg-expr"
+version = "0.15.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d067ad48b8650848b989a59a86c6c36a995d02d2bf778d45c3c5d57bc2718f02"
+dependencies = [
+ "smallvec",
+ "target-lexicon",
+]
+
+[[package]]
+name = "cfg-if"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801"
+
+[[package]]
+name = "cfg_aliases"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
+
+[[package]]
+name = "chrono"
+version = "0.4.44"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c673075a2e0e5f4a1dde27ce9dee1ea4558c7ffe648f576438a20ca1d2acc4b0"
+dependencies = [
+ "iana-time-zone",
+ "num-traits",
+ "serde",
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "combine"
+version = "4.6.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd"
+dependencies = [
+ "bytes",
+ "memchr",
+]
+
+[[package]]
+name = "convert_case"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e"
+
+[[package]]
+name = "cookie"
+version = "0.18.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4ddef33a339a91ea89fb53151bd0a4689cfce27055c291dfa69945475d22c747"
+dependencies = [
+ "time",
+ "version_check",
+]
+
+[[package]]
+name = "core-foundation"
+version = "0.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2a6cd9ae233e7f62ba4e9353e81a88df7fc8a5987b8d445b4d90c879bd156f6"
+dependencies = [
+ "core-foundation-sys",
+ "libc",
+]
+
+[[package]]
+name = "core-foundation-sys"
+version = "0.8.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b"
+
+[[package]]
+name = "core-graphics"
+version = "0.25.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "064badf302c3194842cf2c5d61f56cc88e54a759313879cdf03abdd27d0c3b97"
+dependencies = [
+ "bitflags 2.11.0",
+ "core-foundation",
+ "core-graphics-types",
+ "foreign-types",
+ "libc",
+]
+
+[[package]]
+name = "core-graphics-types"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3d44a101f213f6c4cdc1853d4b78aef6db6bdfa3468798cc1d9912f4735013eb"
+dependencies = [
+ "bitflags 2.11.0",
+ "core-foundation",
+ "libc",
+]
+
+[[package]]
+name = "cpufeatures"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "crc32fast"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9481c1c90cbf2ac953f07c8d4a58aa3945c425b7185c9154d67a65e4230da511"
+dependencies = [
+ "cfg-if",
+]
+
+[[package]]
+name = "crossbeam-channel"
+version = "0.5.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "82b8f8f868b36967f9606790d1903570de9ceaf870a7bf9fbbd3016d636a2cb2"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-utils"
+version = "0.8.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28"
+
+[[package]]
+name = "crypto-common"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "78c8292055d1c1df0cce5d180393dc8cce0abec0a7102adb6c7b1eef6016d60a"
+dependencies = [
+ "generic-array",
+ "typenum",
+]
+
+[[package]]
+name = "cssparser"
+version = "0.29.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f93d03419cb5950ccfd3daf3ff1c7a36ace64609a1a8746d493df1ca0afde0fa"
+dependencies = [
+ "cssparser-macros",
+ "dtoa-short",
+ "itoa",
+ "matches",
+ "phf 0.10.1",
+ "proc-macro2",
+ "quote",
+ "smallvec",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "cssparser-macros"
+version = "0.6.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "13b588ba4ac1a99f7f2964d24b3d896ddc6bf847ee3855dbd4366f058cfcd331"
+dependencies = [
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "ctor"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32a2785755761f3ddc1492979ce1e48d2c00d09311c39e4466429188f3dd6501"
+dependencies = [
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "darling"
+version = "0.21.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9cdf337090841a411e2a7f3deb9187445851f91b309c0c0a29e05f74a00a48c0"
+dependencies = [
+ "darling_core",
+ "darling_macro",
+]
+
+[[package]]
+name = "darling_core"
+version = "0.21.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1247195ecd7e3c85f83c8d2a366e4210d588e802133e1e355180a9870b517ea4"
+dependencies = [
+ "fnv",
+ "ident_case",
+ "proc-macro2",
+ "quote",
+ "strsim",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "darling_macro"
+version = "0.21.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d38308df82d1080de0afee5d069fa14b0326a88c14f15c5ccda35b4a6c414c81"
+dependencies = [
+ "darling_core",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "deranged"
+version = "0.5.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7cd812cc2bc1d69d4764bd80df88b4317eaef9e773c75226407d9bc0876b211c"
+dependencies = [
+ "powerfmt",
+ "serde_core",
+]
+
+[[package]]
+name = "derive_more"
+version = "0.99.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6edb4b64a43d977b8e99788fe3a04d483834fba1215a7e02caa415b626497f7f"
+dependencies = [
+ "convert_case",
+ "proc-macro2",
+ "quote",
+ "rustc_version",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "digest"
+version = "0.10.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
+dependencies = [
+ "block-buffer",
+ "crypto-common",
+]
+
+[[package]]
+name = "dirs"
+version = "6.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c3e8aa94d75141228480295a7d0e7feb620b1a5ad9f12bc40be62411e38cce4e"
+dependencies = [
+ "dirs-sys",
+]
+
+[[package]]
+name = "dirs-sys"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e01a3366d27ee9890022452ee61b2b63a67e6f13f58900b651ff5665f0bb1fab"
+dependencies = [
+ "libc",
+ "option-ext",
+ "redox_users",
+ "windows-sys 0.61.2",
+]
+
+[[package]]
+name = "dispatch2"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e0e367e4e7da84520dedcac1901e4da967309406d1e51017ae1abfb97adbd38"
+dependencies = [
+ "bitflags 2.11.0",
+ "block2",
+ "libc",
+ "objc2",
+]
+
+[[package]]
+name = "displaydoc"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "dlopen2"
+version = "0.8.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5e2c5bd4158e66d1e215c49b837e11d62f3267b30c92f1d171c4d3105e3dc4d4"
+dependencies = [
+ "dlopen2_derive",
+ "libc",
+ "once_cell",
+ "winapi",
+]
+
+[[package]]
+name = "dlopen2_derive"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0fbbb781877580993a8707ec48672673ec7b81eeba04cfd2310bd28c08e47c8f"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "dpi"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d8b14ccef22fc6f5a8f4d7d768562a182c04ce9a3b3157b91390b52ddfdf1a76"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "dtoa"
+version = "1.0.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4c3cf4824e2d5f025c7b531afcb2325364084a16806f6d47fbc1f5fbd9960590"
+
+[[package]]
+name = "dtoa-short"
+version = "0.3.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cd1511a7b6a56299bd043a9c167a6d2bfb37bf84a6dfceaba651168adfb43c87"
+dependencies = [
+ "dtoa",
+]
+
+[[package]]
+name = "dunce"
+version = "1.0.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813"
+
+[[package]]
+name = "dyn-clone"
+version = "1.0.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0881ea181b1df73ff77ffaaf9c7544ecc11e82fba9b5f27b262a3c73a332555"
+
+[[package]]
+name = "embed-resource"
+version = "3.0.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "55a075fc573c64510038d7ee9abc7990635863992f83ebc52c8b433b8411a02e"
+dependencies = [
+ "cc",
+ "memchr",
+ "rustc_version",
+ "toml 0.9.12+spec-1.1.0",
+ "vswhom",
+ "winreg",
+]
+
+[[package]]
+name = "embed_plist"
+version = "1.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4ef6b89e5b37196644d8796de5268852ff179b44e96276cf4290264843743bb7"
+
+[[package]]
+name = "env_filter"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bf3c259d255ca70051b30e2e95b5446cdb8949ac4cd22c0d7fd634d89f568e2"
+dependencies = [
+ "log",
+ "regex",
+]
+
+[[package]]
+name = "equivalent"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f"
+
+[[package]]
+name = "erased-serde"
+version = "0.4.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d2add8a07dd6a8d93ff627029c51de145e12686fbc36ecb298ac22e74cf02dec"
+dependencies = [
+ "serde",
+ "serde_core",
+ "typeid",
+]
+
+[[package]]
+name = "fdeflate"
+version = "0.3.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e6853b52649d4ac5c0bd02320cddc5ba956bdb407c4b75a2c6b75bf51500f8c"
+dependencies = [
+ "simd-adler32",
+]
+
+[[package]]
+name = "fern"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4316185f709b23713e41e3195f90edef7fb00c3ed4adc79769cf09cc762a3b29"
+dependencies = [
+ "log",
+]
+
+[[package]]
+name = "field-offset"
+version = "0.3.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38e2275cc4e4fc009b0669731a1e5ab7ebf11f469eaede2bab9309a5b4d6057f"
+dependencies = [
+ "memoffset",
+ "rustc_version",
+]
+
+[[package]]
+name = "find-msvc-tools"
+version = "0.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582"
+
+[[package]]
+name = "flate2"
+version = "1.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "843fba2746e448b37e26a819579957415c8cef339bf08564fe8b7ddbd959573c"
+dependencies = [
+ "crc32fast",
+ "miniz_oxide",
+]
+
+[[package]]
+name = "fnv"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
+
+[[package]]
+name = "foldhash"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2"
+
+[[package]]
+name = "foreign-types"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d737d9aa519fb7b749cbc3b962edcf310a8dd1f4b67c91c4f83975dbdd17d965"
+dependencies = [
+ "foreign-types-macros",
+ "foreign-types-shared",
+]
+
+[[package]]
+name = "foreign-types-macros"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1a5c6c585bc94aaf2c7b51dd4c2ba22680844aba4c687be581871a6f518c5742"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "foreign-types-shared"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aa9a19cbb55df58761df49b23516a86d432839add4af60fc256da840f66ed35b"
+
+[[package]]
+name = "form_urlencoded"
+version = "1.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cb4cb245038516f5f85277875cdaa4f7d2c9a0fa0468de06ed190163b1581fcf"
+dependencies = [
+ "percent-encoding",
+]
+
+[[package]]
+name = "funty"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c"
+
+[[package]]
+name = "futf"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "df420e2e84819663797d1ec6544b13c5be84629e7bb00dc960d6917db2987843"
+dependencies = [
+ "mac",
+ "new_debug_unreachable",
+]
+
+[[package]]
+name = "futures-channel"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07bbe89c50d7a535e539b8c17bc0b49bdb77747034daa8087407d655f3f7cc1d"
+dependencies = [
+ "futures-core",
+]
+
+[[package]]
+name = "futures-core"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7e3450815272ef58cec6d564423f6e755e25379b217b0bc688e295ba24df6b1d"
+
+[[package]]
+name = "futures-executor"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "baf29c38818342a3b26b5b923639e7b1f4a61fc5e76102d4b1981c6dc7a7579d"
+dependencies = [
+ "futures-core",
+ "futures-task",
+ "futures-util",
+]
+
+[[package]]
+name = "futures-io"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cecba35d7ad927e23624b22ad55235f2239cfa44fd10428eecbeba6d6a717718"
+
+[[package]]
+name = "futures-macro"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e835b70203e41293343137df5c0664546da5745f82ec9b84d40be8336958447b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "futures-sink"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c39754e157331b013978ec91992bde1ac089843443c49cbc7f46150b0fad0893"
+
+[[package]]
+name = "futures-task"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "037711b3d59c33004d3856fbdc83b99d4ff37a24768fa1be9ce3538a1cde4393"
+
+[[package]]
+name = "futures-util"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "389ca41296e6190b48053de0321d02a77f32f8a5d2461dd38762c0593805c6d6"
+dependencies = [
+ "futures-core",
+ "futures-io",
+ "futures-macro",
+ "futures-sink",
+ "futures-task",
+ "memchr",
+ "pin-project-lite",
+ "slab",
+]
+
+[[package]]
+name = "fxhash"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c"
+dependencies = [
+ "byteorder",
+]
+
+[[package]]
+name = "gdk"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9f245958c627ac99d8e529166f9823fb3b838d1d41fd2b297af3075093c2691"
+dependencies = [
+ "cairo-rs",
+ "gdk-pixbuf",
+ "gdk-sys",
+ "gio",
+ "glib",
+ "libc",
+ "pango",
+]
+
+[[package]]
+name = "gdk-pixbuf"
+version = "0.18.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "50e1f5f1b0bfb830d6ccc8066d18db35c487b1b2b1e8589b5dfe9f07e8defaec"
+dependencies = [
+ "gdk-pixbuf-sys",
+ "gio",
+ "glib",
+ "libc",
+ "once_cell",
+]
+
+[[package]]
+name = "gdk-pixbuf-sys"
+version = "0.18.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f9839ea644ed9c97a34d129ad56d38a25e6756f99f3a88e15cd39c20629caf7"
+dependencies = [
+ "gio-sys",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "gdk-sys"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c2d13f38594ac1e66619e188c6d5a1adb98d11b2fcf7894fc416ad76aa2f3f7"
+dependencies = [
+ "cairo-sys-rs",
+ "gdk-pixbuf-sys",
+ "gio-sys",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "pango-sys",
+ "pkg-config",
+ "system-deps",
+]
+
+[[package]]
+name = "gdkwayland-sys"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "140071d506d223f7572b9f09b5e155afbd77428cd5cc7af8f2694c41d98dfe69"
+dependencies = [
+ "gdk-sys",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "pkg-config",
+ "system-deps",
+]
+
+[[package]]
+name = "gdkx11"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3caa00e14351bebbc8183b3c36690327eb77c49abc2268dd4bd36b856db3fbfe"
+dependencies = [
+ "gdk",
+ "gdkx11-sys",
+ "gio",
+ "glib",
+ "libc",
+ "x11",
+]
+
+[[package]]
+name = "gdkx11-sys"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e2e7445fe01ac26f11601db260dd8608fe172514eb63b3b5e261ea6b0f4428d"
+dependencies = [
+ "gdk-sys",
+ "glib-sys",
+ "libc",
+ "system-deps",
+ "x11",
+]
+
+[[package]]
+name = "generic-array"
+version = "0.14.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
+dependencies = [
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.1.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "wasi 0.9.0+wasi-snapshot-preview1",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ff2abc00be7fca6ebc474524697ae276ad847ad0a6b3faa4bcb027e9a4614ad0"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "wasi 0.11.1+wasi-snapshot-preview1",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.3.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "899def5c37c4fd7b2664648c28120ecec138e4d395b459e5ca34f9cce2dd77fd"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "r-efi 5.3.0",
+ "wasip2",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0de51e6874e94e7bf76d726fc5d13ba782deca734ff60d5bb2fb2607c7406555"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "r-efi 6.0.0",
+ "wasip2",
+ "wasip3",
+]
+
+[[package]]
+name = "gio"
+version = "0.18.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d4fc8f532f87b79cbc51a79748f16a6828fb784be93145a322fa14d06d354c73"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-util",
+ "gio-sys",
+ "glib",
+ "libc",
+ "once_cell",
+ "pin-project-lite",
+ "smallvec",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "gio-sys"
+version = "0.18.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "37566df850baf5e4cb0dfb78af2e4b9898d817ed9263d1090a2df958c64737d2"
+dependencies = [
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+ "winapi",
+]
+
+[[package]]
+name = "glib"
+version = "0.18.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "233daaf6e83ae6a12a52055f568f9d7cf4671dabb78ff9560ab6da230ce00ee5"
+dependencies = [
+ "bitflags 2.11.0",
+ "futures-channel",
+ "futures-core",
+ "futures-executor",
+ "futures-task",
+ "futures-util",
+ "gio-sys",
+ "glib-macros",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "memchr",
+ "once_cell",
+ "smallvec",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "glib-macros"
+version = "0.18.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc"
+dependencies = [
+ "heck 0.4.1",
+ "proc-macro-crate 2.0.2",
+ "proc-macro-error",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "glib-sys"
+version = "0.18.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898"
+dependencies = [
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "glob"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0cc23270f6e1808e30a928bdc84dea0b9b4136a8bc82338574f23baf47bbd280"
+
+[[package]]
+name = "gobject-sys"
+version = "0.18.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44"
+dependencies = [
+ "glib-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "gtk"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fd56fb197bfc42bd5d2751f4f017d44ff59fbb58140c6b49f9b3b2bdab08506a"
+dependencies = [
+ "atk",
+ "cairo-rs",
+ "field-offset",
+ "futures-channel",
+ "gdk",
+ "gdk-pixbuf",
+ "gio",
+ "glib",
+ "gtk-sys",
+ "gtk3-macros",
+ "libc",
+ "pango",
+ "pkg-config",
+]
+
+[[package]]
+name = "gtk-sys"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f29a1c21c59553eb7dd40e918be54dccd60c52b049b75119d5d96ce6b624414"
+dependencies = [
+ "atk-sys",
+ "cairo-sys-rs",
+ "gdk-pixbuf-sys",
+ "gdk-sys",
+ "gio-sys",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "pango-sys",
+ "system-deps",
+]
+
+[[package]]
+name = "gtk3-macros"
+version = "0.18.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "52ff3c5b21f14f0736fed6dcfc0bfb4225ebf5725f3c0209edeec181e4d73e9d"
+dependencies = [
+ "proc-macro-crate 1.3.1",
+ "proc-macro-error",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888"
+dependencies = [
+ "ahash",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.15.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9229cfe53dfd69f0609a49f65461bd93001ea1ef889cd5529dd176593f5338a1"
+dependencies = [
+ "foldhash",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.16.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "841d1cc9bed7f9236f321df977030373f4a4163ae1a7dbfe1a51a2c1a51d9100"
+
+[[package]]
+name = "heck"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
+
+[[package]]
+name = "heck"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea"
+
+[[package]]
+name = "hex"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70"
+
+[[package]]
+name = "html5ever"
+version = "0.29.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3b7410cae13cbc75623c98ac4cbfd1f0bedddf3227afc24f370cf0f50a44a11c"
+dependencies = [
+ "log",
+ "mac",
+ "markup5ever",
+ "match_token",
+]
+
+[[package]]
+name = "http"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3ba2a386d7f85a81f119ad7498ebe444d2e22c2af0b86b069416ace48b3311a"
+dependencies = [
+ "bytes",
+ "itoa",
+]
+
+[[package]]
+name = "http-body"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184"
+dependencies = [
+ "bytes",
+ "http",
+]
+
+[[package]]
+name = "http-body-util"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b021d93e26becf5dc7e1b75b1bed1fd93124b374ceb73f43d4d4eafec896a64a"
+dependencies = [
+ "bytes",
+ "futures-core",
+ "http",
+ "http-body",
+ "pin-project-lite",
+]
+
+[[package]]
+name = "httparse"
+version = "1.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6dbf3de79e51f3d586ab4cb9d5c3e2c14aa28ed23d180cf89b4df0454a69cc87"
+
+[[package]]
+name = "hyper"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2ab2d4f250c3d7b1c9fcdff1cece94ea4e2dfbec68614f7b87cb205f24ca9d11"
+dependencies = [
+ "atomic-waker",
+ "bytes",
+ "futures-channel",
+ "futures-core",
+ "http",
+ "http-body",
+ "httparse",
+ "itoa",
+ "pin-project-lite",
+ "pin-utils",
+ "smallvec",
+ "tokio",
+ "want",
+]
+
+[[package]]
+name = "hyper-util"
+version = "0.1.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "96547c2556ec9d12fb1578c4eaf448b04993e7fb79cbaad930a656880a6bdfa0"
+dependencies = [
+ "base64 0.22.1",
+ "bytes",
+ "futures-channel",
+ "futures-util",
+ "http",
+ "http-body",
+ "hyper",
+ "ipnet",
+ "libc",
+ "percent-encoding",
+ "pin-project-lite",
+ "socket2",
+ "tokio",
+ "tower-service",
+ "tracing",
+]
+
+[[package]]
+name = "iana-time-zone"
+version = "0.1.65"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e31bc9ad994ba00e440a8aa5c9ef0ec67d5cb5e5cb0cc7f8b744a35b389cc470"
+dependencies = [
+ "android_system_properties",
+ "core-foundation-sys",
+ "iana-time-zone-haiku",
+ "js-sys",
+ "log",
+ "wasm-bindgen",
+ "windows-core 0.62.2",
+]
+
+[[package]]
+name = "iana-time-zone-haiku"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "ico"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3e795dff5605e0f04bff85ca41b51a96b83e80b281e96231bcaaf1ac35103371"
+dependencies = [
+ "byteorder",
+ "png",
+]
+
+[[package]]
+name = "icu_collections"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4c6b649701667bbe825c3b7e6388cb521c23d88644678e83c0c4d0a621a34b43"
+dependencies = [
+ "displaydoc",
+ "potential_utf",
+ "yoke",
+ "zerofrom",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_locale_core"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "edba7861004dd3714265b4db54a3c390e880ab658fec5f7db895fae2046b5bb6"
+dependencies = [
+ "displaydoc",
+ "litemap",
+ "tinystr",
+ "writeable",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_normalizer"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f6c8828b67bf8908d82127b2054ea1b4427ff0230ee9141c54251934ab1b599"
+dependencies = [
+ "icu_collections",
+ "icu_normalizer_data",
+ "icu_properties",
+ "icu_provider",
+ "smallvec",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_normalizer_data"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7aedcccd01fc5fe81e6b489c15b247b8b0690feb23304303a9e560f37efc560a"
+
+[[package]]
+name = "icu_properties"
+version = "2.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "020bfc02fe870ec3a66d93e677ccca0562506e5872c650f893269e08615d74ec"
+dependencies = [
+ "icu_collections",
+ "icu_locale_core",
+ "icu_properties_data",
+ "icu_provider",
+ "zerotrie",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_properties_data"
+version = "2.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "616c294cf8d725c6afcd8f55abc17c56464ef6211f9ed59cccffe534129c77af"
+
+[[package]]
+name = "icu_provider"
+version = "2.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85962cf0ce02e1e0a629cc34e7ca3e373ce20dda4c4d7294bbd0bf1fdb59e614"
+dependencies = [
+ "displaydoc",
+ "icu_locale_core",
+ "writeable",
+ "yoke",
+ "zerofrom",
+ "zerotrie",
+ "zerovec",
+]
+
+[[package]]
+name = "id-arena"
+version = "2.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3d3067d79b975e8844ca9eb072e16b31c3c1c36928edf9c6789548c524d0d954"
+
+[[package]]
+name = "ident_case"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39"
+
+[[package]]
+name = "idna"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3b0875f23caa03898994f6ddc501886a45c7d3d62d04d2d90788d47be1b1e4de"
+dependencies = [
+ "idna_adapter",
+ "smallvec",
+ "utf8_iter",
+]
+
+[[package]]
+name = "idna_adapter"
+version = "1.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3acae9609540aa318d1bc588455225fb2085b9ed0c4f6bd0d9d5bcd86f1a0344"
+dependencies = [
+ "icu_normalizer",
+ "icu_properties",
+]
+
+[[package]]
+name = "indexmap"
+version = "1.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99"
+dependencies = [
+ "autocfg",
+ "hashbrown 0.12.3",
+ "serde",
+]
+
+[[package]]
+name = "indexmap"
+version = "2.13.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7714e70437a7dc3ac8eb7e6f8df75fd8eb422675fc7678aff7364301092b1017"
+dependencies = [
+ "equivalent",
+ "hashbrown 0.16.1",
+ "serde",
+ "serde_core",
+]
+
+[[package]]
+name = "infer"
+version = "0.19.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a588916bfdfd92e71cacef98a63d9b1f0d74d6599980d11894290e7ddefffcf7"
+dependencies = [
+ "cfb",
+]
+
+[[package]]
+name = "ipnet"
+version = "2.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d98f6fed1fde3f8c21bc40a1abb88dd75e67924f9cffc3ef95607bad8017f8e2"
+
+[[package]]
+name = "iri-string"
+version = "0.7.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c91338f0783edbd6195decb37bae672fd3b165faffb89bf7b9e6942f8b1a731a"
+dependencies = [
+ "memchr",
+ "serde",
+]
+
+[[package]]
+name = "itoa"
+version = "1.0.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "92ecc6618181def0457392ccd0ee51198e065e016d1d527a7ac1b6dc7c1f09d2"
+
+[[package]]
+name = "javascriptcore-rs"
+version = "1.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca5671e9ffce8ffba57afc24070e906da7fc4b1ba66f2cabebf61bf2ea257fcc"
+dependencies = [
+ "bitflags 1.3.2",
+ "glib",
+ "javascriptcore-rs-sys",
+]
+
+[[package]]
+name = "javascriptcore-rs-sys"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "af1be78d14ffa4b75b66df31840478fef72b51f8c2465d4ca7c194da9f7a5124"
+dependencies = [
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "jni"
+version = "0.21.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1a87aa2bb7d2af34197c04845522473242e1aa17c12f4935d5856491a7fb8c97"
+dependencies = [
+ "cesu8",
+ "cfg-if",
+ "combine",
+ "jni-sys",
+ "log",
+ "thiserror 1.0.69",
+ "walkdir",
+ "windows-sys 0.45.0",
+]
+
+[[package]]
+name = "jni-sys"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8eaf4bc02d17cbdd7ff4c7438cafcdf7fb9a4613313ad11b4f8fefe7d3fa0130"
+
+[[package]]
+name = "js-sys"
+version = "0.3.91"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b49715b7073f385ba4bc528e5747d02e66cb39c6146efb66b781f131f0fb399c"
+dependencies = [
+ "once_cell",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "json-patch"
+version = "3.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "863726d7afb6bc2590eeff7135d923545e5e964f004c2ccf8716c25e70a86f08"
+dependencies = [
+ "jsonptr",
+ "serde",
+ "serde_json",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "jsonptr"
+version = "0.6.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5dea2b27dd239b2556ed7a25ba842fe47fd602e7fc7433c2a8d6106d4d9edd70"
+dependencies = [
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "keyboard-types"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b750dcadc39a09dbadd74e118f6dd6598df77fa01df0cfcdc52c28dece74528a"
+dependencies = [
+ "bitflags 2.11.0",
+ "serde",
+ "unicode-segmentation",
+]
+
+[[package]]
+name = "kuchikiki"
+version = "0.8.8-speedreader"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "02cb977175687f33fa4afa0c95c112b987ea1443e5a51c8f8ff27dc618270cc2"
+dependencies = [
+ "cssparser",
+ "html5ever",
+ "indexmap 2.13.0",
+ "selectors",
+]
+
+[[package]]
+name = "leb128fmt"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09edd9e8b54e49e587e4f6295a7d29c3ea94d469cb40ab8ca70b288248a81db2"
+
+[[package]]
+name = "libappindicator"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03589b9607c868cc7ae54c0b2a22c8dc03dd41692d48f2d7df73615c6a95dc0a"
+dependencies = [
+ "glib",
+ "gtk",
+ "gtk-sys",
+ "libappindicator-sys",
+ "log",
+]
+
+[[package]]
+name = "libappindicator-sys"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e9ec52138abedcc58dc17a7c6c0c00a2bdb4f3427c7f63fa97fd0d859155caf"
+dependencies = [
+ "gtk-sys",
+ "libloading",
+ "once_cell",
+]
+
+[[package]]
+name = "libc"
+version = "0.2.182"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6800badb6cb2082ffd7b6a67e6125bb39f18782f793520caee8cb8846be06112"
+
+[[package]]
+name = "libloading"
+version = "0.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b67380fd3b2fbe7527a606e18729d21c6f3951633d0500574c4dc22d2d638b9f"
+dependencies = [
+ "cfg-if",
+ "winapi",
+]
+
+[[package]]
+name = "libredox"
+version = "0.1.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1744e39d1d6a9948f4f388969627434e31128196de472883b39f148769bfe30a"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "litemap"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6373607a59f0be73a39b6fe456b8192fcc3585f602af20751600e974dd455e77"
+
+[[package]]
+name = "lock_api"
+version = "0.4.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "224399e74b87b5f3557511d98dff8b14089b3dadafcab6bb93eab67d3aace965"
+dependencies = [
+ "scopeguard",
+]
+
+[[package]]
+name = "log"
+version = "0.4.29"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5e5032e24019045c762d3c0f28f5b6b8bbf38563a65908389bf7978758920897"
+dependencies = [
+ "value-bag",
+]
+
+[[package]]
+name = "mac"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c41e0c4fef86961ac6d6f8a82609f55f31b05e4fce149ac5710e439df7619ba4"
+
+[[package]]
+name = "markup5ever"
+version = "0.14.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c7a7213d12e1864c0f002f52c2923d4556935a43dec5e71355c2760e0f6e7a18"
+dependencies = [
+ "log",
+ "phf 0.11.3",
+ "phf_codegen 0.11.3",
+ "string_cache",
+ "string_cache_codegen",
+ "tendril",
+]
+
+[[package]]
+name = "match_token"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "88a9689d8d44bf9964484516275f5cd4c9b59457a6940c1d5d0ecbb94510a36b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "matches"
+version = "0.1.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5"
+
+[[package]]
+name = "memchr"
+version = "2.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f8ca58f447f06ed17d5fc4043ce1b10dd205e060fb3ce5b979b8ed8e59ff3f79"
+
+[[package]]
+name = "memoffset"
+version = "0.9.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "mime"
+version = "0.3.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a"
+
+[[package]]
+name = "miniz_oxide"
+version = "0.8.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fa76a2c86f704bdb222d66965fb3d63269ce38518b83cb0575fca855ebb6316"
+dependencies = [
+ "adler2",
+ "simd-adler32",
+]
+
+[[package]]
+name = "mio"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a69bcab0ad47271a0234d9422b131806bf3968021e5dc9328caf2d4cd58557fc"
+dependencies = [
+ "libc",
+ "wasi 0.11.1+wasi-snapshot-preview1",
+ "windows-sys 0.61.2",
+]
+
+[[package]]
+name = "muda"
+version = "0.17.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "01c1738382f66ed56b3b9c8119e794a2e23148ac8ea214eda86622d4cb9d415a"
+dependencies = [
+ "crossbeam-channel",
+ "dpi",
+ "gtk",
+ "keyboard-types",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-core-foundation",
+ "objc2-foundation",
+ "once_cell",
+ "png",
+ "serde",
+ "thiserror 2.0.18",
+ "windows-sys 0.60.2",
+]
+
+[[package]]
+name = "ndk"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c3f42e7bbe13d351b6bead8286a43aac9534b82bd3cc43e47037f012ebfd62d4"
+dependencies = [
+ "bitflags 2.11.0",
+ "jni-sys",
+ "log",
+ "ndk-sys",
+ "num_enum",
+ "raw-window-handle",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "ndk-context"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "27b02d87554356db9e9a873add8782d4ea6e3e58ea071a9adb9a2e8ddb884a8b"
+
+[[package]]
+name = "ndk-sys"
+version = "0.6.0+11769913"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ee6cda3051665f1fb8d9e08fc35c96d5a244fb1be711a03b71118828afc9a873"
+dependencies = [
+ "jni-sys",
+]
+
+[[package]]
+name = "new_debug_unreachable"
+version = "1.0.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "650eef8c711430f1a879fdd01d4745a7deea475becfb90269c06775983bbf086"
+
+[[package]]
+name = "nodrop"
+version = "0.1.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72ef4a56884ca558e5ddb05a1d1e7e1bfd9a68d9ed024c21704cc98872dae1bb"
+
+[[package]]
+name = "num-conv"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cf97ec579c3c42f953ef76dbf8d55ac91fb219dde70e49aa4a6b7d74e9919050"
+
+[[package]]
+name = "num-traits"
+version = "0.2.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "num_enum"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b1207a7e20ad57b847bbddc6776b968420d38292bbfe2089accff5e19e82454c"
+dependencies = [
+ "num_enum_derive",
+ "rustversion",
+]
+
+[[package]]
+name = "num_enum_derive"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ff32365de1b6743cb203b710788263c44a03de03802daf96092f2da4fe6ba4d7"
+dependencies = [
+ "proc-macro-crate 3.5.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "num_threads"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c7398b9c8b70908f6371f47ed36737907c87c52af34c268fed0bf0ceb92ead9"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "objc2"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3a12a8ed07aefc768292f076dc3ac8c48f3781c8f2d5851dd3d98950e8c5a89f"
+dependencies = [
+ "objc2-encode",
+ "objc2-exception-helper",
+]
+
+[[package]]
+name = "objc2-app-kit"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d49e936b501e5c5bf01fda3a9452ff86dc3ea98ad5f283e1455153142d97518c"
+dependencies = [
+ "bitflags 2.11.0",
+ "block2",
+ "objc2",
+ "objc2-core-foundation",
+ "objc2-foundation",
+]
+
+[[package]]
+name = "objc2-core-foundation"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2a180dd8642fa45cdb7dd721cd4c11b1cadd4929ce112ebd8b9f5803cc79d536"
+dependencies = [
+ "bitflags 2.11.0",
+ "dispatch2",
+ "objc2",
+]
+
+[[package]]
+name = "objc2-core-graphics"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e022c9d066895efa1345f8e33e584b9f958da2fd4cd116792e15e07e4720a807"
+dependencies = [
+ "bitflags 2.11.0",
+ "dispatch2",
+ "objc2",
+ "objc2-core-foundation",
+ "objc2-io-surface",
+]
+
+[[package]]
+name = "objc2-encode"
+version = "4.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ef25abbcd74fb2609453eb695bd2f860d389e457f67dc17cafc8b8cbc89d0c33"
+
+[[package]]
+name = "objc2-exception-helper"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c7a1c5fbb72d7735b076bb47b578523aedc40f3c439bea6dfd595c089d79d98a"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "objc2-foundation"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3e0adef53c21f888deb4fa59fc59f7eb17404926ee8a6f59f5df0fd7f9f3272"
+dependencies = [
+ "bitflags 2.11.0",
+ "block2",
+ "objc2",
+ "objc2-core-foundation",
+]
+
+[[package]]
+name = "objc2-io-surface"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "180788110936d59bab6bd83b6060ffdfffb3b922ba1396b312ae795e1de9d81d"
+dependencies = [
+ "bitflags 2.11.0",
+ "objc2",
+ "objc2-core-foundation",
+]
+
+[[package]]
+name = "objc2-quartz-core"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "96c1358452b371bf9f104e21ec536d37a650eb10f7ee379fff67d2e08d537f1f"
+dependencies = [
+ "bitflags 2.11.0",
+ "objc2",
+ "objc2-core-foundation",
+ "objc2-foundation",
+]
+
+[[package]]
+name = "objc2-ui-kit"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d87d638e33c06f577498cbcc50491496a3ed4246998a7fbba7ccb98b1e7eab22"
+dependencies = [
+ "bitflags 2.11.0",
+ "objc2",
+ "objc2-core-foundation",
+ "objc2-foundation",
+]
+
+[[package]]
+name = "objc2-web-kit"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2e5aaab980c433cf470df9d7af96a7b46a9d892d521a2cbbb2f8a4c16751e7f"
+dependencies = [
+ "bitflags 2.11.0",
+ "block2",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-core-foundation",
+ "objc2-foundation",
+]
+
+[[package]]
+name = "once_cell"
+version = "1.21.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "42f5e15c9953c5e4ccceeb2e7382a716482c34515315f7b03532b8b4e8393d2d"
+
+[[package]]
+name = "option-ext"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d"
+
+[[package]]
+name = "pango"
+version = "0.18.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4"
+dependencies = [
+ "gio",
+ "glib",
+ "libc",
+ "once_cell",
+ "pango-sys",
+]
+
+[[package]]
+name = "pango-sys"
+version = "0.18.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5"
+dependencies = [
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "parking_lot"
+version = "0.12.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "93857453250e3077bd71ff98b6a65ea6621a19bb0f559a85248955ac12c45a1a"
+dependencies = [
+ "lock_api",
+ "parking_lot_core",
+]
+
+[[package]]
+name = "parking_lot_core"
+version = "0.9.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2621685985a2ebf1c516881c026032ac7deafcda1a2c9b7850dc81e3dfcb64c1"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "redox_syscall",
+ "smallvec",
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "percent-encoding"
+version = "2.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220"
+
+[[package]]
+name = "phf"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3dfb61232e34fcb633f43d12c58f83c1df82962dcdfa565a4e866ffc17dafe12"
+dependencies = [
+ "phf_shared 0.8.0",
+]
+
+[[package]]
+name = "phf"
+version = "0.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fabbf1ead8a5bcbc20f5f8b939ee3f5b0f6f281b6ad3468b84656b658b455259"
+dependencies = [
+ "phf_macros 0.10.0",
+ "phf_shared 0.10.0",
+ "proc-macro-hack",
+]
+
+[[package]]
+name = "phf"
+version = "0.11.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fd6780a80ae0c52cc120a26a1a42c1ae51b247a253e4e06113d23d2c2edd078"
+dependencies = [
+ "phf_macros 0.11.3",
+ "phf_shared 0.11.3",
+]
+
+[[package]]
+name = "phf_codegen"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cbffee61585b0411840d3ece935cce9cb6321f01c45477d30066498cd5e1a815"
+dependencies = [
+ "phf_generator 0.8.0",
+ "phf_shared 0.8.0",
+]
+
+[[package]]
+name = "phf_codegen"
+version = "0.11.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aef8048c789fa5e851558d709946d6d79a8ff88c0440c587967f8e94bfb1216a"
+dependencies = [
+ "phf_generator 0.11.3",
+ "phf_shared 0.11.3",
+]
+
+[[package]]
+name = "phf_generator"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17367f0cc86f2d25802b2c26ee58a7b23faeccf78a396094c13dced0d0182526"
+dependencies = [
+ "phf_shared 0.8.0",
+ "rand 0.7.3",
+]
+
+[[package]]
+name = "phf_generator"
+version = "0.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5d5285893bb5eb82e6aaf5d59ee909a06a16737a8970984dd7746ba9283498d6"
+dependencies = [
+ "phf_shared 0.10.0",
+ "rand 0.8.5",
+]
+
+[[package]]
+name = "phf_generator"
+version = "0.11.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c80231409c20246a13fddb31776fb942c38553c51e871f8cbd687a4cfb5843d"
+dependencies = [
+ "phf_shared 0.11.3",
+ "rand 0.8.5",
+]
+
+[[package]]
+name = "phf_macros"
+version = "0.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "58fdf3184dd560f160dd73922bea2d5cd6e8f064bf4b13110abd81b03697b4e0"
+dependencies = [
+ "phf_generator 0.10.0",
+ "phf_shared 0.10.0",
+ "proc-macro-hack",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "phf_macros"
+version = "0.11.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f84ac04429c13a7ff43785d75ad27569f2951ce0ffd30a3321230db2fc727216"
+dependencies = [
+ "phf_generator 0.11.3",
+ "phf_shared 0.11.3",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "phf_shared"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c00cf8b9eafe68dde5e9eaa2cef8ee84a9336a47d566ec55ca16589633b65af7"
+dependencies = [
+ "siphasher 0.3.11",
+]
+
+[[package]]
+name = "phf_shared"
+version = "0.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6796ad771acdc0123d2a88dc428b5e38ef24456743ddb1744ed628f9815c096"
+dependencies = [
+ "siphasher 0.3.11",
+]
+
+[[package]]
+name = "phf_shared"
+version = "0.11.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67eabc2ef2a60eb7faa00097bd1ffdb5bd28e62bf39990626a582201b7a754e5"
+dependencies = [
+ "siphasher 1.0.2",
+]
+
+[[package]]
+name = "pin-project-lite"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a89322df9ebe1c1578d689c92318e070967d1042b512afbe49518723f4e6d5cd"
+
+[[package]]
+name = "pin-utils"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
+
+[[package]]
+name = "pkg-config"
+version = "0.3.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7edddbd0b52d732b21ad9a5fab5c704c14cd949e5e9a1ec5929a24fded1b904c"
+
+[[package]]
+name = "plist"
+version = "1.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "740ebea15c5d1428f910cd1a5f52cebf8d25006245ed8ade92702f4943d91e07"
+dependencies = [
+ "base64 0.22.1",
+ "indexmap 2.13.0",
+ "quick-xml",
+ "serde",
+ "time",
+]
+
+[[package]]
+name = "png"
+version = "0.17.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "82151a2fc869e011c153adc57cf2789ccb8d9906ce52c0b39a6b5697749d7526"
+dependencies = [
+ "bitflags 1.3.2",
+ "crc32fast",
+ "fdeflate",
+ "flate2",
+ "miniz_oxide",
+]
+
+[[package]]
+name = "potential_utf"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b73949432f5e2a09657003c25bca5e19a0e9c84f8058ca374f49e0ebe605af77"
+dependencies = [
+ "zerovec",
+]
+
+[[package]]
+name = "powerfmt"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391"
+
+[[package]]
+name = "ppv-lite86"
+version = "0.2.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9"
+dependencies = [
+ "zerocopy",
+]
+
+[[package]]
+name = "precomputed-hash"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "925383efa346730478fb4838dbe9137d2a47675ad789c546d150a6e1dd4ab31c"
+
+[[package]]
+name = "prettyplease"
+version = "0.2.37"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "479ca8adacdd7ce8f1fb39ce9ecccbfe93a3f1344b3d0d97f20bc0196208f62b"
+dependencies = [
+ "proc-macro2",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "1.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f4c021e1093a56626774e81216a4ce732a735e5bad4868a03f3ed65ca0c3919"
+dependencies = [
+ "once_cell",
+ "toml_edit 0.19.15",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "2.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b00f26d3400549137f92511a46ac1cd8ce37cb5598a96d382381458b992a5d24"
+dependencies = [
+ "toml_datetime 0.6.3",
+ "toml_edit 0.20.2",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "3.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e67ba7e9b2b56446f1d419b1d807906278ffa1a658a8a5d8a39dcb1f5a78614f"
+dependencies = [
+ "toml_edit 0.25.4+spec-1.1.0",
+]
+
+[[package]]
+name = "proc-macro-error"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c"
+dependencies = [
+ "proc-macro-error-attr",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+ "version_check",
+]
+
+[[package]]
+name = "proc-macro-error-attr"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "version_check",
+]
+
+[[package]]
+name = "proc-macro-hack"
+version = "0.5.20+deprecated"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc375e1527247fe1a97d8b7156678dfe7c1af2fc075c9a4db3690ecd2a148068"
+
+[[package]]
+name = "proc-macro2"
+version = "1.0.106"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fd00f0bb2e90d81d1044c2b32617f68fcb9fa3bb7640c23e9c748e53fb30934"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "ptr_meta"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0738ccf7ea06b608c10564b31debd4f5bc5e197fc8bfe088f68ae5ce81e7a4f1"
+dependencies = [
+ "ptr_meta_derive",
+]
+
+[[package]]
+name = "ptr_meta_derive"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "16b845dbfca988fa33db069c0e230574d15a3088f147a87b64c7589eb662c9ac"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "quick-xml"
+version = "0.38.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b66c2058c55a409d601666cffe35f04333cf1013010882cec174a7467cd4e21c"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "quote"
+version = "1.0.45"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "41f2619966050689382d2b44f664f4bc593e129785a36d6ee376ddf37259b924"
+dependencies = [
+ "proc-macro2",
+]
+
+[[package]]
+name = "r-efi"
+version = "5.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "69cdb34c158ceb288df11e18b4bd39de994f6657d83847bdffdbd7f346754b0f"
+
+[[package]]
+name = "r-efi"
+version = "6.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f8dcc9c7d52a811697d2151c701e0d08956f92b0e24136cf4cf27b57a6a0d9bf"
+
+[[package]]
+name = "radium"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc33ff2d4973d518d823d61aa239014831e521c75da58e3df4840d3f47749d09"
+
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "getrandom 0.1.16",
+ "libc",
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+ "rand_pcg",
+]
+
+[[package]]
+name = "rand"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
+dependencies = [
+ "libc",
+ "rand_chacha 0.3.1",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
+dependencies = [
+ "getrandom 0.1.16",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
+dependencies = [
+ "getrandom 0.2.17",
+]
+
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_pcg"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "16abd0c1b639e9eb4d7c50c0b8100b0d0f849be2349829c740fe8e6eb4816429"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "raw-window-handle"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "20675572f6f24e9e76ef639bc5552774ed45f1c30e2951e1e99c59888861c539"
+
+[[package]]
+name = "redox_syscall"
+version = "0.5.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ed2bf2547551a7053d6fdfafda3f938979645c44812fbfcda098faae3f1a362d"
+dependencies = [
+ "bitflags 2.11.0",
+]
+
+[[package]]
+name = "redox_users"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a4e608c6638b9c18977b00b475ac1f28d14e84b27d8d42f70e0bf1e3dec127ac"
+dependencies = [
+ "getrandom 0.2.17",
+ "libredox",
+ "thiserror 2.0.18",
+]
+
+[[package]]
+name = "ref-cast"
+version = "1.0.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f354300ae66f76f1c85c5f84693f0ce81d747e2c3f21a45fef496d89c960bf7d"
+dependencies = [
+ "ref-cast-impl",
+]
+
+[[package]]
+name = "ref-cast-impl"
+version = "1.0.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b7186006dcb21920990093f30e3dea63b7d6e977bf1256be20c3563a5db070da"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "regex"
+version = "1.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e10754a14b9137dd7b1e3e5b0493cc9171fdd105e0ab477f51b72e7f3ac0e276"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-automata",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-automata"
+version = "0.4.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e1dd4122fc1595e8162618945476892eefca7b88c52820e74af6262213cae8f"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-syntax"
+version = "0.8.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc897dd8d9e8bd1ed8cdad82b5966c3e0ecae09fb1907d58efaa013543185d0a"
+
+[[package]]
+name = "rend"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "71fe3824f5629716b1589be05dacd749f6aa084c87e00e016714a8cdfccc997c"
+dependencies = [
+ "bytecheck",
+]
+
+[[package]]
+name = "reqwest"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ab3f43e3283ab1488b624b44b0e988d0acea0b3214e694730a055cb6b2efa801"
+dependencies = [
+ "base64 0.22.1",
+ "bytes",
+ "futures-core",
+ "futures-util",
+ "http",
+ "http-body",
+ "http-body-util",
+ "hyper",
+ "hyper-util",
+ "js-sys",
+ "log",
+ "percent-encoding",
+ "pin-project-lite",
+ "serde",
+ "serde_json",
+ "sync_wrapper",
+ "tokio",
+ "tokio-util",
+ "tower",
+ "tower-http",
+ "tower-service",
+ "url",
+ "wasm-bindgen",
+ "wasm-bindgen-futures",
+ "wasm-streams",
+ "web-sys",
+]
+
+[[package]]
+name = "rkyv"
+version = "0.7.46"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2297bf9c81a3f0dc96bc9521370b88f054168c29826a75e89c55ff196e7ed6a1"
+dependencies = [
+ "bitvec",
+ "bytecheck",
+ "bytes",
+ "hashbrown 0.12.3",
+ "ptr_meta",
+ "rend",
+ "rkyv_derive",
+ "seahash",
+ "tinyvec",
+ "uuid",
+]
+
+[[package]]
+name = "rkyv_derive"
+version = "0.7.46"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "84d7b42d4b8d06048d3ac8db0eb31bcb942cbeb709f0b5f2b2ebde398d3038f5"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "rust_decimal"
+version = "1.40.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "61f703d19852dbf87cbc513643fa81428361eb6940f1ac14fd58155d295a3eb0"
+dependencies = [
+ "arrayvec",
+ "borsh",
+ "bytes",
+ "num-traits",
+ "rand 0.8.5",
+ "rkyv",
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "rustc_version"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92"
+dependencies = [
+ "semver",
+]
+
+[[package]]
+name = "rustversion"
+version = "1.0.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b39cdef0fa800fc44525c84ccb54a029961a8215f9619753635a9c0d2538d46d"
+
+[[package]]
+name = "same-file"
+version = "1.0.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502"
+dependencies = [
+ "winapi-util",
+]
+
+[[package]]
+name = "schemars"
+version = "0.8.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3fbf2ae1b8bc8e02df939598064d22402220cd5bbcca1c76f7d6a310974d5615"
+dependencies = [
+ "dyn-clone",
+ "indexmap 1.9.3",
+ "schemars_derive",
+ "serde",
+ "serde_json",
+ "url",
+ "uuid",
+]
+
+[[package]]
+name = "schemars"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4cd191f9397d57d581cddd31014772520aa448f65ef991055d7f61582c65165f"
+dependencies = [
+ "dyn-clone",
+ "ref-cast",
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "schemars"
+version = "1.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a2b42f36aa1cd011945615b92222f6bf73c599a102a300334cd7f8dbeec726cc"
+dependencies = [
+ "dyn-clone",
+ "ref-cast",
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "schemars_derive"
+version = "0.8.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32e265784ad618884abaea0600a9adf15393368d840e0222d101a072f3f7534d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "serde_derive_internals",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "scopeguard"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
+
+[[package]]
+name = "seahash"
+version = "4.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c107b6f4780854c8b126e228ea8869f4d7b71260f962fefb57b996b8959ba6b"
+
+[[package]]
+name = "selectors"
+version = "0.24.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0c37578180969d00692904465fb7f6b3d50b9a2b952b87c23d0e2e5cb5013416"
+dependencies = [
+ "bitflags 1.3.2",
+ "cssparser",
+ "derive_more",
+ "fxhash",
+ "log",
+ "phf 0.8.0",
+ "phf_codegen 0.8.0",
+ "precomputed-hash",
+ "servo_arc",
+ "smallvec",
+]
+
+[[package]]
+name = "semver"
+version = "1.0.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d767eb0aabc880b29956c35734170f26ed551a859dbd361d140cdbeca61ab1e2"
+dependencies = [
+ "serde",
+ "serde_core",
+]
+
+[[package]]
+name = "serde"
+version = "1.0.228"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a8e94ea7f378bd32cbbd37198a4a91436180c5bb472411e48b5ec2e2124ae9e"
+dependencies = [
+ "serde_core",
+ "serde_derive",
+]
+
+[[package]]
+name = "serde-untagged"
+version = "0.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f9faf48a4a2d2693be24c6289dbe26552776eb7737074e6722891fadbe6c5058"
+dependencies = [
+ "erased-serde",
+ "serde",
+ "serde_core",
+ "typeid",
+]
+
+[[package]]
+name = "serde_core"
+version = "1.0.228"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "41d385c7d4ca58e59fc732af25c3983b67ac852c1a25000afe1175de458b67ad"
+dependencies = [
+ "serde_derive",
+]
+
+[[package]]
+name = "serde_derive"
+version = "1.0.228"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d540f220d3187173da220f885ab66608367b6574e925011a9353e4badda91d79"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "serde_derive_internals"
+version = "0.29.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "serde_json"
+version = "1.0.149"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83fc039473c5595ace860d8c4fafa220ff474b3fc6bfdb4293327f1a37e94d86"
+dependencies = [
+ "itoa",
+ "memchr",
+ "serde",
+ "serde_core",
+ "zmij",
+]
+
+[[package]]
+name = "serde_repr"
+version = "0.1.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "175ee3e80ae9982737ca543e96133087cbd9a485eecc3bc4de9c1a37b47ea59c"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "serde_spanned"
+version = "0.6.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf41e0cfaf7226dca15e8197172c295a782857fcb97fad1808a166870dee75a3"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "serde_spanned"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f8bbf91e5a4d6315eee45e704372590b30e260ee83af6639d64557f51b067776"
+dependencies = [
+ "serde_core",
+]
+
+[[package]]
+name = "serde_with"
+version = "3.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "381b283ce7bc6b476d903296fb59d0d36633652b633b27f64db4fb46dcbfc3b9"
+dependencies = [
+ "base64 0.22.1",
+ "chrono",
+ "hex",
+ "indexmap 1.9.3",
+ "indexmap 2.13.0",
+ "schemars 0.9.0",
+ "schemars 1.2.1",
+ "serde_core",
+ "serde_json",
+ "serde_with_macros",
+ "time",
+]
+
+[[package]]
+name = "serde_with_macros"
+version = "3.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a6d4e30573c8cb306ed6ab1dca8423eec9a463ea0e155f45399455e0368b27e0"
+dependencies = [
+ "darling",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "serialize-to-javascript"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04f3666a07a197cdb77cdf306c32be9b7f598d7060d50cfd4d5aa04bfd92f6c5"
+dependencies = [
+ "serde",
+ "serde_json",
+ "serialize-to-javascript-impl",
+]
+
+[[package]]
+name = "serialize-to-javascript-impl"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "772ee033c0916d670af7860b6e1ef7d658a4629a6d0b4c8c3e67f09b3765b75d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "servo_arc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d52aa42f8fdf0fed91e5ce7f23d8138441002fa31dca008acf47e6fd4721f741"
+dependencies = [
+ "nodrop",
+ "stable_deref_trait",
+]
+
+[[package]]
+name = "sha2"
+version = "0.10.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "digest",
+]
+
+[[package]]
+name = "shlex"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
+
+[[package]]
+name = "simd-adler32"
+version = "0.3.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e320a6c5ad31d271ad523dcf3ad13e2767ad8b1cb8f047f75a8aeaf8da139da2"
+
+[[package]]
+name = "simdutf8"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3a9fe34e3e7a50316060351f37187a3f546bce95496156754b601a5fa71b76e"
+
+[[package]]
+name = "siphasher"
+version = "0.3.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d"
+
+[[package]]
+name = "siphasher"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2aa850e253778c88a04c3d7323b043aeda9d3e30d5971937c1855769763678e"
+
+[[package]]
+name = "slab"
+version = "0.4.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0c790de23124f9ab44544d7ac05d60440adc586479ce501c1d6d7da3cd8c9cf5"
+
+[[package]]
+name = "smallvec"
+version = "1.15.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67b1b7a3b5fe4f1376887184045fcf45c69e92af734b7aaddc05fb777b6fbd03"
+
+[[package]]
+name = "socket2"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "86f4aa3ad99f2088c990dfa82d367e19cb29268ed67c574d10d0a4bfe71f07e0"
+dependencies = [
+ "libc",
+ "windows-sys 0.60.2",
+]
+
+[[package]]
+name = "softbuffer"
+version = "0.4.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aac18da81ebbf05109ab275b157c22a653bb3c12cf884450179942f81bcbf6c3"
+dependencies = [
+ "bytemuck",
+ "js-sys",
+ "ndk",
+ "objc2",
+ "objc2-core-foundation",
+ "objc2-core-graphics",
+ "objc2-foundation",
+ "objc2-quartz-core",
+ "raw-window-handle",
+ "redox_syscall",
+ "tracing",
+ "wasm-bindgen",
+ "web-sys",
+ "windows-sys 0.61.2",
+]
+
+[[package]]
+name = "soup3"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "471f924a40f31251afc77450e781cb26d55c0b650842efafc9c6cbd2f7cc4f9f"
+dependencies = [
+ "futures-channel",
+ "gio",
+ "glib",
+ "libc",
+ "soup3-sys",
+]
+
+[[package]]
+name = "soup3-sys"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ebe8950a680a12f24f15ebe1bf70db7af98ad242d9db43596ad3108aab86c27"
+dependencies = [
+ "gio-sys",
+ "glib-sys",
+ "gobject-sys",
+ "libc",
+ "system-deps",
+]
+
+[[package]]
+name = "stable_deref_trait"
+version = "1.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6ce2be8dc25455e1f91df71bfa12ad37d7af1092ae736f3a6cd0e37bc7810596"
+
+[[package]]
+name = "string_cache"
+version = "0.8.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf776ba3fa74f83bf4b63c3dcbbf82173db2632ed8452cb2d891d33f459de70f"
+dependencies = [
+ "new_debug_unreachable",
+ "parking_lot",
+ "phf_shared 0.11.3",
+ "precomputed-hash",
+ "serde",
+]
+
+[[package]]
+name = "string_cache_codegen"
+version = "0.5.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c711928715f1fe0fe509c53b43e993a9a557babc2d0a3567d0a3006f1ac931a0"
+dependencies = [
+ "phf_generator 0.11.3",
+ "phf_shared 0.11.3",
+ "proc-macro2",
+ "quote",
+]
+
+[[package]]
+name = "strsim"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f"
+
+[[package]]
+name = "swift-rs"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4057c98e2e852d51fdcfca832aac7b571f6b351ad159f9eda5db1655f8d0c4d7"
+dependencies = [
+ "base64 0.21.7",
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "syn"
+version = "1.0.109"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "syn"
+version = "2.0.117"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e665b8803e7b1d2a727f4023456bbbbe74da67099c585258af0ad9c5013b9b99"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "sync_wrapper"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0bf256ce5efdfa370213c1dabab5935a12e49f2c58d15e9eac2870d3b4f27263"
+dependencies = [
+ "futures-core",
+]
+
+[[package]]
+name = "synstructure"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "728a70f3dbaf5bab7f0c4b1ac8d7ae5ea60a4b5549c8a5914361c99147a709d2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "system-deps"
+version = "6.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349"
+dependencies = [
+ "cfg-expr",
+ "heck 0.5.0",
+ "pkg-config",
+ "toml 0.8.2",
+ "version-compare",
+]
+
+[[package]]
+name = "tao"
+version = "0.34.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e06d52c379e63da659a483a958110bbde891695a0ecb53e48cc7786d5eda7bb"
+dependencies = [
+ "bitflags 2.11.0",
+ "block2",
+ "core-foundation",
+ "core-graphics",
+ "crossbeam-channel",
+ "dispatch2",
+ "dlopen2",
+ "dpi",
+ "gdkwayland-sys",
+ "gdkx11-sys",
+ "gtk",
+ "jni",
+ "libc",
+ "log",
+ "ndk",
+ "ndk-context",
+ "ndk-sys",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-foundation",
+ "once_cell",
+ "parking_lot",
+ "raw-window-handle",
+ "tao-macros",
+ "unicode-segmentation",
+ "url",
+ "windows",
+ "windows-core 0.61.2",
+ "windows-version",
+ "x11-dl",
+]
+
+[[package]]
+name = "tao-macros"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4e16beb8b2ac17db28eab8bca40e62dbfbb34c0fcdc6d9826b11b7b5d047dfd"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "tap"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369"
+
+[[package]]
+name = "target-lexicon"
+version = "0.12.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1"
+
+[[package]]
+name = "tauri"
+version = "2.10.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "da77cc00fb9028caf5b5d4650f75e31f1ef3693459dfca7f7e506d1ecef0ba2d"
+dependencies = [
+ "anyhow",
+ "bytes",
+ "cookie",
+ "dirs",
+ "dunce",
+ "embed_plist",
+ "getrandom 0.3.4",
+ "glob",
+ "gtk",
+ "heck 0.5.0",
+ "http",
+ "jni",
+ "libc",
+ "log",
+ "mime",
+ "muda",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-foundation",
+ "objc2-ui-kit",
+ "objc2-web-kit",
+ "percent-encoding",
+ "plist",
+ "raw-window-handle",
+ "reqwest",
+ "serde",
+ "serde_json",
+ "serde_repr",
+ "serialize-to-javascript",
+ "swift-rs",
+ "tauri-build",
+ "tauri-macros",
+ "tauri-runtime",
+ "tauri-runtime-wry",
+ "tauri-utils",
+ "thiserror 2.0.18",
+ "tokio",
+ "tray-icon",
+ "url",
+ "webkit2gtk",
+ "webview2-com",
+ "window-vibrancy",
+ "windows",
+]
+
+[[package]]
+name = "tauri-build"
+version = "2.5.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4bbc990d1dbf57a8e1c7fa2327f2a614d8b757805603c1b9ba5c81bade09fd4d"
+dependencies = [
+ "anyhow",
+ "cargo_toml",
+ "dirs",
+ "glob",
+ "heck 0.5.0",
+ "json-patch",
+ "schemars 0.8.22",
+ "semver",
+ "serde",
+ "serde_json",
+ "tauri-utils",
+ "tauri-winres",
+ "toml 0.9.12+spec-1.1.0",
+ "walkdir",
+]
+
+[[package]]
+name = "tauri-codegen"
+version = "2.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d4a24476afd977c5d5d169f72425868613d82747916dd29e0a357c84c4bd6d29"
+dependencies = [
+ "base64 0.22.1",
+ "brotli",
+ "ico",
+ "json-patch",
+ "plist",
+ "png",
+ "proc-macro2",
+ "quote",
+ "semver",
+ "serde",
+ "serde_json",
+ "sha2",
+ "syn 2.0.117",
+ "tauri-utils",
+ "thiserror 2.0.18",
+ "time",
+ "url",
+ "uuid",
+ "walkdir",
+]
+
+[[package]]
+name = "tauri-macros"
+version = "2.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d39b349a98dadaffebb73f0a40dcd1f23c999211e5a2e744403db384d0c33de7"
+dependencies = [
+ "heck 0.5.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+ "tauri-codegen",
+ "tauri-utils",
+]
+
+[[package]]
+name = "tauri-plugin"
+version = "2.5.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ddde7d51c907b940fb573006cdda9a642d6a7c8153657e88f8a5c3c9290cd4aa"
+dependencies = [
+ "anyhow",
+ "glob",
+ "plist",
+ "schemars 0.8.22",
+ "serde",
+ "serde_json",
+ "tauri-utils",
+ "toml 0.9.12+spec-1.1.0",
+ "walkdir",
+]
+
+[[package]]
+name = "tauri-plugin-log"
+version = "2.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7545bd67f070a4500432c826e2e0682146a1d6712aee22a2786490156b574d93"
+dependencies = [
+ "android_logger",
+ "byte-unit",
+ "fern",
+ "log",
+ "objc2",
+ "objc2-foundation",
+ "serde",
+ "serde_json",
+ "serde_repr",
+ "swift-rs",
+ "tauri",
+ "tauri-plugin",
+ "thiserror 2.0.18",
+ "time",
+]
+
+[[package]]
+name = "tauri-runtime"
+version = "2.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2826d79a3297ed08cd6ea7f412644ef58e32969504bc4fbd8d7dbeabc4445ea2"
+dependencies = [
+ "cookie",
+ "dpi",
+ "gtk",
+ "http",
+ "jni",
+ "objc2",
+ "objc2-ui-kit",
+ "objc2-web-kit",
+ "raw-window-handle",
+ "serde",
+ "serde_json",
+ "tauri-utils",
+ "thiserror 2.0.18",
+ "url",
+ "webkit2gtk",
+ "webview2-com",
+ "windows",
+]
+
+[[package]]
+name = "tauri-runtime-wry"
+version = "2.10.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e11ea2e6f801d275fdd890d6c9603736012742a1c33b96d0db788c9cdebf7f9e"
+dependencies = [
+ "gtk",
+ "http",
+ "jni",
+ "log",
+ "objc2",
+ "objc2-app-kit",
+ "once_cell",
+ "percent-encoding",
+ "raw-window-handle",
+ "softbuffer",
+ "tao",
+ "tauri-runtime",
+ "tauri-utils",
+ "url",
+ "webkit2gtk",
+ "webview2-com",
+ "windows",
+ "wry",
+]
+
+[[package]]
+name = "tauri-utils"
+version = "2.8.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "219a1f983a2af3653f75b5747f76733b0da7ff03069c7a41901a5eb3ace4557d"
+dependencies = [
+ "anyhow",
+ "brotli",
+ "cargo_metadata",
+ "ctor",
+ "dunce",
+ "glob",
+ "html5ever",
+ "http",
+ "infer",
+ "json-patch",
+ "kuchikiki",
+ "log",
+ "memchr",
+ "phf 0.11.3",
+ "proc-macro2",
+ "quote",
+ "regex",
+ "schemars 0.8.22",
+ "semver",
+ "serde",
+ "serde-untagged",
+ "serde_json",
+ "serde_with",
+ "swift-rs",
+ "thiserror 2.0.18",
+ "toml 0.9.12+spec-1.1.0",
+ "url",
+ "urlpattern",
+ "uuid",
+ "walkdir",
+]
+
+[[package]]
+name = "tauri-winres"
+version = "0.3.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1087b111fe2b005e42dbdc1990fc18593234238d47453b0c99b7de1c9ab2c1e0"
+dependencies = [
+ "dunce",
+ "embed-resource",
+ "toml 0.9.12+spec-1.1.0",
+]
+
+[[package]]
+name = "tendril"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d24a120c5fc464a3458240ee02c299ebcb9d67b5249c8848b09d639dca8d7bb0"
+dependencies = [
+ "futf",
+ "mac",
+ "utf-8",
+]
+
+[[package]]
+name = "thiserror"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52"
+dependencies = [
+ "thiserror-impl 1.0.69",
+]
+
+[[package]]
+name = "thiserror"
+version = "2.0.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4288b5bcbc7920c07a1149a35cf9590a2aa808e0bc1eafaade0b80947865fbc4"
+dependencies = [
+ "thiserror-impl 2.0.18",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "2.0.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebc4ee7f67670e9b64d05fa4253e753e016c6c95ff35b89b7941d6b856dec1d5"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "time"
+version = "0.3.47"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "743bd48c283afc0388f9b8827b976905fb217ad9e647fae3a379a9283c4def2c"
+dependencies = [
+ "deranged",
+ "itoa",
+ "libc",
+ "num-conv",
+ "num_threads",
+ "powerfmt",
+ "serde_core",
+ "time-core",
+ "time-macros",
+]
+
+[[package]]
+name = "time-core"
+version = "0.1.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7694e1cfe791f8d31026952abf09c69ca6f6fa4e1a1229e18988f06a04a12dca"
+
+[[package]]
+name = "time-macros"
+version = "0.2.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2e70e4c5a0e0a8a4823ad65dfe1a6930e4f4d756dcd9dd7939022b5e8c501215"
+dependencies = [
+ "num-conv",
+ "time-core",
+]
+
+[[package]]
+name = "tinystr"
+version = "0.8.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "42d3e9c45c09de15d06dd8acf5f4e0e399e85927b7f00711024eb7ae10fa4869"
+dependencies = [
+ "displaydoc",
+ "zerovec",
+]
+
+[[package]]
+name = "tinyvec"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bfa5fdc3bce6191a1dbc8c02d5c8bffcf557bafa17c124c5264a458f1b0613fa"
+dependencies = [
+ "tinyvec_macros",
+]
+
+[[package]]
+name = "tinyvec_macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
+
+[[package]]
+name = "tokio"
+version = "1.50.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "27ad5e34374e03cfffefc301becb44e9dc3c17584f414349ebe29ed26661822d"
+dependencies = [
+ "bytes",
+ "libc",
+ "mio",
+ "pin-project-lite",
+ "socket2",
+ "windows-sys 0.61.2",
+]
+
+[[package]]
+name = "tokio-util"
+version = "0.7.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ae9cec805b01e8fc3fd2fe289f89149a9b66dd16786abd8b19cfa7b48cb0098"
+dependencies = [
+ "bytes",
+ "futures-core",
+ "futures-sink",
+ "pin-project-lite",
+ "tokio",
+]
+
+[[package]]
+name = "toml"
+version = "0.8.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d"
+dependencies = [
+ "serde",
+ "serde_spanned 0.6.9",
+ "toml_datetime 0.6.3",
+ "toml_edit 0.20.2",
+]
+
+[[package]]
+name = "toml"
+version = "0.9.12+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cf92845e79fc2e2def6a5d828f0801e29a2f8acc037becc5ab08595c7d5e9863"
+dependencies = [
+ "indexmap 2.13.0",
+ "serde_core",
+ "serde_spanned 1.0.4",
+ "toml_datetime 0.7.5+spec-1.1.0",
+ "toml_parser",
+ "toml_writer",
+ "winnow 0.7.15",
+]
+
+[[package]]
+name = "toml_datetime"
+version = "0.6.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "toml_datetime"
+version = "0.7.5+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "92e1cfed4a3038bc5a127e35a2d360f145e1f4b971b551a2ba5fd7aedf7e1347"
+dependencies = [
+ "serde_core",
+]
+
+[[package]]
+name = "toml_datetime"
+version = "1.0.0+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32c2555c699578a4f59f0cc68e5116c8d7cabbd45e1409b989d4be085b53f13e"
+dependencies = [
+ "serde_core",
+]
+
+[[package]]
+name = "toml_edit"
+version = "0.19.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1b5bb770da30e5cbfde35a2d7b9b8a2c4b8ef89548a7a6aeab5c9a576e3e7421"
+dependencies = [
+ "indexmap 2.13.0",
+ "toml_datetime 0.6.3",
+ "winnow 0.5.40",
+]
+
+[[package]]
+name = "toml_edit"
+version = "0.20.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338"
+dependencies = [
+ "indexmap 2.13.0",
+ "serde",
+ "serde_spanned 0.6.9",
+ "toml_datetime 0.6.3",
+ "winnow 0.5.40",
+]
+
+[[package]]
+name = "toml_edit"
+version = "0.25.4+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7193cbd0ce53dc966037f54351dbbcf0d5a642c7f0038c382ef9e677ce8c13f2"
+dependencies = [
+ "indexmap 2.13.0",
+ "toml_datetime 1.0.0+spec-1.1.0",
+ "toml_parser",
+ "winnow 0.7.15",
+]
+
+[[package]]
+name = "toml_parser"
+version = "1.0.9+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "702d4415e08923e7e1ef96cd5727c0dfed80b4d2fa25db9647fe5eb6f7c5a4c4"
+dependencies = [
+ "winnow 0.7.15",
+]
+
+[[package]]
+name = "toml_writer"
+version = "1.0.6+spec-1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ab16f14aed21ee8bfd8ec22513f7287cd4a91aa92e44edfe2c17ddd004e92607"
+
+[[package]]
+name = "tower"
+version = "0.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebe5ef63511595f1344e2d5cfa636d973292adc0eec1f0ad45fae9f0851ab1d4"
+dependencies = [
+ "futures-core",
+ "futures-util",
+ "pin-project-lite",
+ "sync_wrapper",
+ "tokio",
+ "tower-layer",
+ "tower-service",
+]
+
+[[package]]
+name = "tower-http"
+version = "0.6.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d4e6559d53cc268e5031cd8429d05415bc4cb4aefc4aa5d6cc35fbf5b924a1f8"
+dependencies = [
+ "bitflags 2.11.0",
+ "bytes",
+ "futures-util",
+ "http",
+ "http-body",
+ "iri-string",
+ "pin-project-lite",
+ "tower",
+ "tower-layer",
+ "tower-service",
+]
+
+[[package]]
+name = "tower-layer"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "121c2a6cda46980bb0fcd1647ffaf6cd3fc79a013de288782836f6df9c48780e"
+
+[[package]]
+name = "tower-service"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3"
+
+[[package]]
+name = "tracing"
+version = "0.1.44"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "63e71662fa4b2a2c3a26f570f037eb95bb1f85397f3cd8076caed2f026a6d100"
+dependencies = [
+ "pin-project-lite",
+ "tracing-core",
+]
+
+[[package]]
+name = "tracing-core"
+version = "0.1.36"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "db97caf9d906fbde555dd62fa95ddba9eecfd14cb388e4f491a66d74cd5fb79a"
+dependencies = [
+ "once_cell",
+]
+
+[[package]]
+name = "tray-icon"
+version = "0.21.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a5e85aa143ceb072062fc4d6356c1b520a51d636e7bc8e77ec94be3608e5e80c"
+dependencies = [
+ "crossbeam-channel",
+ "dirs",
+ "libappindicator",
+ "muda",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-core-foundation",
+ "objc2-core-graphics",
+ "objc2-foundation",
+ "once_cell",
+ "png",
+ "serde",
+ "thiserror 2.0.18",
+ "windows-sys 0.60.2",
+]
+
+[[package]]
+name = "try-lock"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b"
+
+[[package]]
+name = "typeid"
+version = "1.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bc7d623258602320d5c55d1bc22793b57daff0ec7efc270ea7d55ce1d5f5471c"
+
+[[package]]
+name = "typenum"
+version = "1.19.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "562d481066bde0658276a35467c4af00bdc6ee726305698a55b86e61d7ad82bb"
+
+[[package]]
+name = "unic-char-property"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a8c57a407d9b6fa02b4795eb81c5b6652060a15a7903ea981f3d723e6c0be221"
+dependencies = [
+ "unic-char-range",
+]
+
+[[package]]
+name = "unic-char-range"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0398022d5f700414f6b899e10b8348231abf9173fa93144cbc1a43b9793c1fbc"
+
+[[package]]
+name = "unic-common"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "80d7ff825a6a654ee85a63e80f92f054f904f21e7d12da4e22f9834a4aaa35bc"
+
+[[package]]
+name = "unic-ucd-ident"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e230a37c0381caa9219d67cf063aa3a375ffed5bf541a452db16e744bdab6987"
+dependencies = [
+ "unic-char-property",
+ "unic-char-range",
+ "unic-ucd-version",
+]
+
+[[package]]
+name = "unic-ucd-version"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "96bd2f2237fe450fcd0a1d2f5f4e91711124f7857ba2e964247776ebeeb7b0c4"
+dependencies = [
+ "unic-common",
+]
+
+[[package]]
+name = "unicode-ident"
+version = "1.0.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75"
+
+[[package]]
+name = "unicode-segmentation"
+version = "1.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493"
+
+[[package]]
+name = "unicode-xid"
+version = "0.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebc1c04c71510c7f702b52b7c350734c9ff1295c464a03335b00bb84fc54f853"
+
+[[package]]
+name = "url"
+version = "2.5.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ff67a8a4397373c3ef660812acab3268222035010ab8680ec4215f38ba3d0eed"
+dependencies = [
+ "form_urlencoded",
+ "idna",
+ "percent-encoding",
+ "serde",
+ "serde_derive",
+]
+
+[[package]]
+name = "urlpattern"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "70acd30e3aa1450bc2eece896ce2ad0d178e9c079493819301573dae3c37ba6d"
+dependencies = [
+ "regex",
+ "serde",
+ "unic-ucd-ident",
+ "url",
+]
+
+[[package]]
+name = "utf-8"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9"
+
+[[package]]
+name = "utf8-width"
+version = "0.1.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1292c0d970b54115d14f2492fe0170adf21d68a1de108eebc51c1df4f346a091"
+
+[[package]]
+name = "utf8_iter"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be"
+
+[[package]]
+name = "uuid"
+version = "1.22.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a68d3c8f01c0cfa54a75291d83601161799e4a89a39e0929f4b0354d88757a37"
+dependencies = [
+ "getrandom 0.4.2",
+ "js-sys",
+ "serde_core",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "value-bag"
+version = "1.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ba6f5989077681266825251a52748b8c1d8a4ad098cc37e440103d0ea717fc0"
+
+[[package]]
+name = "version-compare"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03c2856837ef78f57382f06b2b8563a2f512f7185d732608fd9176cb3b8edf0e"
+
+[[package]]
+name = "version_check"
+version = "0.9.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a"
+
+[[package]]
+name = "vswhom"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "be979b7f07507105799e854203b470ff7c78a1639e330a58f183b5fea574608b"
+dependencies = [
+ "libc",
+ "vswhom-sys",
+]
+
+[[package]]
+name = "vswhom-sys"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fb067e4cbd1ff067d1df46c9194b5de0e98efd2810bbc95c5d5e5f25a3231150"
+dependencies = [
+ "cc",
+ "libc",
+]
+
+[[package]]
+name = "walkdir"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b"
+dependencies = [
+ "same-file",
+ "winapi-util",
+]
+
+[[package]]
+name = "want"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e"
+dependencies = [
+ "try-lock",
+]
+
+[[package]]
+name = "wasi"
+version = "0.9.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519"
+
+[[package]]
+name = "wasi"
+version = "0.11.1+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ccf3ec651a847eb01de73ccad15eb7d99f80485de043efb2f370cd654f4ea44b"
+
+[[package]]
+name = "wasip2"
+version = "1.0.2+wasi-0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9517f9239f02c069db75e65f174b3da828fe5f5b945c4dd26bd25d89c03ebcf5"
+dependencies = [
+ "wit-bindgen",
+]
+
+[[package]]
+name = "wasip3"
+version = "0.4.0+wasi-0.3.0-rc-2026-01-06"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5428f8bf88ea5ddc08faddef2ac4a67e390b88186c703ce6dbd955e1c145aca5"
+dependencies = [
+ "wit-bindgen",
+]
+
+[[package]]
+name = "wasm-bindgen"
+version = "0.2.114"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6532f9a5c1ece3798cb1c2cfdba640b9b3ba884f5db45973a6f442510a87d38e"
+dependencies = [
+ "cfg-if",
+ "once_cell",
+ "rustversion",
+ "wasm-bindgen-macro",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-futures"
+version = "0.4.64"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e9c5522b3a28661442748e09d40924dfb9ca614b21c00d3fd135720e48b67db8"
+dependencies = [
+ "cfg-if",
+ "futures-util",
+ "js-sys",
+ "once_cell",
+ "wasm-bindgen",
+ "web-sys",
+]
+
+[[package]]
+name = "wasm-bindgen-macro"
+version = "0.2.114"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "18a2d50fcf105fb33bb15f00e7a77b772945a2ee45dcf454961fd843e74c18e6"
+dependencies = [
+ "quote",
+ "wasm-bindgen-macro-support",
+]
+
+[[package]]
+name = "wasm-bindgen-macro-support"
+version = "0.2.114"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03ce4caeaac547cdf713d280eda22a730824dd11e6b8c3ca9e42247b25c631e3"
+dependencies = [
+ "bumpalo",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-shared"
+version = "0.2.114"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75a326b8c223ee17883a4251907455a2431acc2791c98c26279376490c378c16"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "wasm-encoder"
+version = "0.244.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "990065f2fe63003fe337b932cfb5e3b80e0b4d0f5ff650e6985b1048f62c8319"
+dependencies = [
+ "leb128fmt",
+ "wasmparser",
+]
+
+[[package]]
+name = "wasm-metadata"
+version = "0.244.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bb0e353e6a2fbdc176932bbaab493762eb1255a7900fe0fea1a2f96c296cc909"
+dependencies = [
+ "anyhow",
+ "indexmap 2.13.0",
+ "wasm-encoder",
+ "wasmparser",
+]
+
+[[package]]
+name = "wasm-streams"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d1ec4f6517c9e11ae630e200b2b65d193279042e28edd4a2cda233e46670bbb"
+dependencies = [
+ "futures-util",
+ "js-sys",
+ "wasm-bindgen",
+ "wasm-bindgen-futures",
+ "web-sys",
+]
+
+[[package]]
+name = "wasmparser"
+version = "0.244.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "47b807c72e1bac69382b3a6fb3dbe8ea4c0ed87ff5629b8685ae6b9a611028fe"
+dependencies = [
+ "bitflags 2.11.0",
+ "hashbrown 0.15.5",
+ "indexmap 2.13.0",
+ "semver",
+]
+
+[[package]]
+name = "web-sys"
+version = "0.3.91"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "854ba17bb104abfb26ba36da9729addc7ce7f06f5c0f90f3c391f8461cca21f9"
+dependencies = [
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "webkit2gtk"
+version = "2.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1027150013530fb2eaf806408df88461ae4815a45c541c8975e61d6f2fc4793"
+dependencies = [
+ "bitflags 1.3.2",
+ "cairo-rs",
+ "gdk",
+ "gdk-sys",
+ "gio",
+ "gio-sys",
+ "glib",
+ "glib-sys",
+ "gobject-sys",
+ "gtk",
+ "gtk-sys",
+ "javascriptcore-rs",
+ "libc",
+ "once_cell",
+ "soup3",
+ "webkit2gtk-sys",
+]
+
+[[package]]
+name = "webkit2gtk-sys"
+version = "2.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "916a5f65c2ef0dfe12fff695960a2ec3d4565359fdbb2e9943c974e06c734ea5"
+dependencies = [
+ "bitflags 1.3.2",
+ "cairo-sys-rs",
+ "gdk-sys",
+ "gio-sys",
+ "glib-sys",
+ "gobject-sys",
+ "gtk-sys",
+ "javascriptcore-rs-sys",
+ "libc",
+ "pkg-config",
+ "soup3-sys",
+ "system-deps",
+]
+
+[[package]]
+name = "webview2-com"
+version = "0.38.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7130243a7a5b33c54a444e54842e6a9e133de08b5ad7b5861cd8ed9a6a5bc96a"
+dependencies = [
+ "webview2-com-macros",
+ "webview2-com-sys",
+ "windows",
+ "windows-core 0.61.2",
+ "windows-implement",
+ "windows-interface",
+]
+
+[[package]]
+name = "webview2-com-macros"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67a921c1b6914c367b2b823cd4cde6f96beec77d30a939c8199bb377cf9b9b54"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "webview2-com-sys"
+version = "0.38.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "381336cfffd772377d291702245447a5251a2ffa5bad679c99e61bc48bacbf9c"
+dependencies = [
+ "thiserror 2.0.18",
+ "windows",
+ "windows-core 0.61.2",
+]
+
+[[package]]
+name = "winapi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
+dependencies = [
+ "winapi-i686-pc-windows-gnu",
+ "winapi-x86_64-pc-windows-gnu",
+]
+
+[[package]]
+name = "winapi-i686-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
+
+[[package]]
+name = "winapi-util"
+version = "0.1.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c2a7b1c03c876122aa43f3020e6c3c3ee5c05081c9a00739faf7503aeba10d22"
+dependencies = [
+ "windows-sys 0.61.2",
+]
+
+[[package]]
+name = "winapi-x86_64-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
+
+[[package]]
+name = "window-vibrancy"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9bec5a31f3f9362f2258fd0e9c9dd61a9ca432e7306cc78c444258f0dce9a9c"
+dependencies = [
+ "objc2",
+ "objc2-app-kit",
+ "objc2-core-foundation",
+ "objc2-foundation",
+ "raw-window-handle",
+ "windows-sys 0.59.0",
+ "windows-version",
+]
+
+[[package]]
+name = "windows"
+version = "0.61.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9babd3a767a4c1aef6900409f85f5d53ce2544ccdfaa86dad48c91782c6d6893"
+dependencies = [
+ "windows-collections",
+ "windows-core 0.61.2",
+ "windows-future",
+ "windows-link 0.1.3",
+ "windows-numerics",
+]
+
+[[package]]
+name = "windows-collections"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3beeceb5e5cfd9eb1d76b381630e82c4241ccd0d27f1a39ed41b2760b255c5e8"
+dependencies = [
+ "windows-core 0.61.2",
+]
+
+[[package]]
+name = "windows-core"
+version = "0.61.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c0fdd3ddb90610c7638aa2b3a3ab2904fb9e5cdbecc643ddb3647212781c4ae3"
+dependencies = [
+ "windows-implement",
+ "windows-interface",
+ "windows-link 0.1.3",
+ "windows-result 0.3.4",
+ "windows-strings 0.4.2",
+]
+
+[[package]]
+name = "windows-core"
+version = "0.62.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8e83a14d34d0623b51dce9581199302a221863196a1dde71a7663a4c2be9deb"
+dependencies = [
+ "windows-implement",
+ "windows-interface",
+ "windows-link 0.2.1",
+ "windows-result 0.4.1",
+ "windows-strings 0.5.1",
+]
+
+[[package]]
+name = "windows-future"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fc6a41e98427b19fe4b73c550f060b59fa592d7d686537eebf9385621bfbad8e"
+dependencies = [
+ "windows-core 0.61.2",
+ "windows-link 0.1.3",
+ "windows-threading",
+]
+
+[[package]]
+name = "windows-implement"
+version = "0.60.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "053e2e040ab57b9dc951b72c264860db7eb3b0200ba345b4e4c3b14f67855ddf"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "windows-interface"
+version = "0.59.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f316c4a2570ba26bbec722032c4099d8c8bc095efccdc15688708623367e358"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "windows-link"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5e6ad25900d524eaabdbbb96d20b4311e1e7ae1699af4fb28c17ae66c80d798a"
+
+[[package]]
+name = "windows-link"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f0805222e57f7521d6a62e36fa9163bc891acd422f971defe97d64e70d0a4fe5"
+
+[[package]]
+name = "windows-numerics"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9150af68066c4c5c07ddc0ce30421554771e528bde427614c61038bc2c92c2b1"
+dependencies = [
+ "windows-core 0.61.2",
+ "windows-link 0.1.3",
+]
+
+[[package]]
+name = "windows-result"
+version = "0.3.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "56f42bd332cc6c8eac5af113fc0c1fd6a8fd2aa08a0119358686e5160d0586c6"
+dependencies = [
+ "windows-link 0.1.3",
+]
+
+[[package]]
+name = "windows-result"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7781fa89eaf60850ac3d2da7af8e5242a5ea78d1a11c49bf2910bb5a73853eb5"
+dependencies = [
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "windows-strings"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "56e6c93f3a0c3b36176cb1327a4958a0353d5d166c2a35cb268ace15e91d3b57"
+dependencies = [
+ "windows-link 0.1.3",
+]
+
+[[package]]
+name = "windows-strings"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7837d08f69c77cf6b07689544538e017c1bfcf57e34b4c0ff58e6c2cd3b37091"
+dependencies = [
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.45.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75283be5efb2831d37ea142365f009c02ec203cd29a3ebecbc093d52315b66d0"
+dependencies = [
+ "windows-targets 0.42.2",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.59.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b"
+dependencies = [
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.60.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f2f500e4d28234f72040990ec9d39e3a6b950f9f22d3dba18416c35882612bcb"
+dependencies = [
+ "windows-targets 0.53.5",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.61.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ae137229bcbd6cdf0f7b80a31df61766145077ddf49416a728b02cb3921ff3fc"
+dependencies = [
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e5180c00cd44c9b1c88adb3693291f1cd93605ded80c250a75d472756b4d071"
+dependencies = [
+ "windows_aarch64_gnullvm 0.42.2",
+ "windows_aarch64_msvc 0.42.2",
+ "windows_i686_gnu 0.42.2",
+ "windows_i686_msvc 0.42.2",
+ "windows_x86_64_gnu 0.42.2",
+ "windows_x86_64_gnullvm 0.42.2",
+ "windows_x86_64_msvc 0.42.2",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973"
+dependencies = [
+ "windows_aarch64_gnullvm 0.52.6",
+ "windows_aarch64_msvc 0.52.6",
+ "windows_i686_gnu 0.52.6",
+ "windows_i686_gnullvm 0.52.6",
+ "windows_i686_msvc 0.52.6",
+ "windows_x86_64_gnu 0.52.6",
+ "windows_x86_64_gnullvm 0.52.6",
+ "windows_x86_64_msvc 0.52.6",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.53.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4945f9f551b88e0d65f3db0bc25c33b8acea4d9e41163edf90dcd0b19f9069f3"
+dependencies = [
+ "windows-link 0.2.1",
+ "windows_aarch64_gnullvm 0.53.1",
+ "windows_aarch64_msvc 0.53.1",
+ "windows_i686_gnu 0.53.1",
+ "windows_i686_gnullvm 0.53.1",
+ "windows_i686_msvc 0.53.1",
+ "windows_x86_64_gnu 0.53.1",
+ "windows_x86_64_gnullvm 0.53.1",
+ "windows_x86_64_msvc 0.53.1",
+]
+
+[[package]]
+name = "windows-threading"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b66463ad2e0ea3bbf808b7f1d371311c80e115c0b71d60efc142cafbcfb057a6"
+dependencies = [
+ "windows-link 0.1.3",
+]
+
+[[package]]
+name = "windows-version"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e4060a1da109b9d0326b7262c8e12c84df67cc0dbc9e33cf49e01ccc2eb63631"
+dependencies = [
+ "windows-link 0.2.1",
+]
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "597a5118570b68bc08d8d59125332c54f1ba9d9adeedeef5b99b02ba2b0698f8"
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3"
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a9d8416fa8b42f5c947f8482c43e7d89e73a173cead56d044f6a56104a6d1b53"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e08e8864a60f06ef0d0ff4ba04124db8b0fb3be5776a5cd47641e942e58c4d43"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9d782e804c2f632e395708e99a94275910eb9100b2114651e04744e9b125006"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c61d927d8da41da96a81f029489353e68739737d3beca43145c8afec9a31a84f"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "960e6da069d81e09becb0ca57a65220ddff016ff2d6af6a223cf372a506593a3"
+
+[[package]]
+name = "windows_i686_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66"
+
+[[package]]
+name = "windows_i686_gnullvm"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fa7359d10048f68ab8b09fa71c3daccfb0e9b559aed648a8f95469c27057180c"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "44d840b6ec649f480a41c8d80f9c65108b92d89345dd94027bfe06ac444d1060"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e7ac75179f18232fe9c285163565a57ef8d3c89254a30685b57d83a38d326c2"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8de912b8b8feb55c064867cf047dda097f92d51efad5b491dfb98f6bbb70cb36"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c3842cdd74a865a8066ab39c8a7a473c0778a3f29370b5fd6b4b9aa7df4a499"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "26d41b46a36d453748aedef1486d5c7a85db22e56aff34643984ea85514e94a3"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0ffa179e2d07eee8ad8f57493436566c7cc30ac536a3379fdf008f47f6bb7ae1"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.42.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9aec5da331524158c6d1a4ac0ab1541149c0b9505fde06423b02f5ef0106b9f0"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.53.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d6bbff5f0aada427a1e5a6da5f1f98158182f26556f345ac9e04d36d0ebed650"
+
+[[package]]
+name = "winnow"
+version = "0.5.40"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "winnow"
+version = "0.7.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "df79d97927682d2fd8adb29682d1140b343be4ac0f08fd68b7765d9c059d3945"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "winreg"
+version = "0.55.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cb5a765337c50e9ec252c2069be9bf91c7df47afb103b642ba3a53bf8101be97"
+dependencies = [
+ "cfg-if",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "wit-bindgen"
+version = "0.51.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d7249219f66ced02969388cf2bb044a09756a083d0fab1e566056b04d9fbcaa5"
+dependencies = [
+ "wit-bindgen-rust-macro",
+]
+
+[[package]]
+name = "wit-bindgen-core"
+version = "0.51.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ea61de684c3ea68cb082b7a88508a8b27fcc8b797d738bfc99a82facf1d752dc"
+dependencies = [
+ "anyhow",
+ "heck 0.5.0",
+ "wit-parser",
+]
+
+[[package]]
+name = "wit-bindgen-rust"
+version = "0.51.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b7c566e0f4b284dd6561c786d9cb0142da491f46a9fbed79ea69cdad5db17f21"
+dependencies = [
+ "anyhow",
+ "heck 0.5.0",
+ "indexmap 2.13.0",
+ "prettyplease",
+ "syn 2.0.117",
+ "wasm-metadata",
+ "wit-bindgen-core",
+ "wit-component",
+]
+
+[[package]]
+name = "wit-bindgen-rust-macro"
+version = "0.51.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0c0f9bfd77e6a48eccf51359e3ae77140a7f50b1e2ebfe62422d8afdaffab17a"
+dependencies = [
+ "anyhow",
+ "prettyplease",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+ "wit-bindgen-core",
+ "wit-bindgen-rust",
+]
+
+[[package]]
+name = "wit-component"
+version = "0.244.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d66ea20e9553b30172b5e831994e35fbde2d165325bec84fc43dbf6f4eb9cb2"
+dependencies = [
+ "anyhow",
+ "bitflags 2.11.0",
+ "indexmap 2.13.0",
+ "log",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "wasm-encoder",
+ "wasm-metadata",
+ "wasmparser",
+ "wit-parser",
+]
+
+[[package]]
+name = "wit-parser"
+version = "0.244.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ecc8ac4bc1dc3381b7f59c34f00b67e18f910c2c0f50015669dde7def656a736"
+dependencies = [
+ "anyhow",
+ "id-arena",
+ "indexmap 2.13.0",
+ "log",
+ "semver",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "unicode-xid",
+ "wasmparser",
+]
+
+[[package]]
+name = "writeable"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9edde0db4769d2dc68579893f2306b26c6ecfbe0ef499b013d731b7b9247e0b9"
+
+[[package]]
+name = "wry"
+version = "0.54.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bb26159b420aa77684589a744ae9a9461a95395b848764ad12290a14d960a11a"
+dependencies = [
+ "base64 0.22.1",
+ "block2",
+ "cookie",
+ "crossbeam-channel",
+ "dirs",
+ "dpi",
+ "dunce",
+ "gdkx11",
+ "gtk",
+ "html5ever",
+ "http",
+ "javascriptcore-rs",
+ "jni",
+ "kuchikiki",
+ "libc",
+ "ndk",
+ "objc2",
+ "objc2-app-kit",
+ "objc2-core-foundation",
+ "objc2-foundation",
+ "objc2-ui-kit",
+ "objc2-web-kit",
+ "once_cell",
+ "percent-encoding",
+ "raw-window-handle",
+ "sha2",
+ "soup3",
+ "tao-macros",
+ "thiserror 2.0.18",
+ "url",
+ "webkit2gtk",
+ "webkit2gtk-sys",
+ "webview2-com",
+ "windows",
+ "windows-core 0.61.2",
+ "windows-version",
+ "x11-dl",
+]
+
+[[package]]
+name = "wyz"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "05f360fc0b24296329c78fda852a1e9ae82de9cf7b27dae4b7f62f118f77b9ed"
+dependencies = [
+ "tap",
+]
+
+[[package]]
+name = "x11"
+version = "2.21.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "502da5464ccd04011667b11c435cb992822c2c0dbde1770c988480d312a0db2e"
+dependencies = [
+ "libc",
+ "pkg-config",
+]
+
+[[package]]
+name = "x11-dl"
+version = "2.21.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38735924fedd5314a6e548792904ed8c6de6636285cb9fec04d5b1db85c1516f"
+dependencies = [
+ "libc",
+ "once_cell",
+ "pkg-config",
+]
+
+[[package]]
+name = "yoke"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72d6e5c6afb84d73944e5cedb052c4680d5657337201555f9f2a16b7406d4954"
+dependencies = [
+ "stable_deref_trait",
+ "yoke-derive",
+ "zerofrom",
+]
+
+[[package]]
+name = "yoke-derive"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b659052874eb698efe5b9e8cf382204678a0086ebf46982b79d6ca3182927e5d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+ "synstructure",
+]
+
+[[package]]
+name = "zerocopy"
+version = "0.8.40"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a789c6e490b576db9f7e6b6d661bcc9799f7c0ac8352f56ea20193b2681532e5"
+dependencies = [
+ "zerocopy-derive",
+]
+
+[[package]]
+name = "zerocopy-derive"
+version = "0.8.40"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f65c489a7071a749c849713807783f70672b28094011623e200cb86dcb835953"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "zerofrom"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "50cc42e0333e05660c3587f3bf9d0478688e15d870fab3346451ce7f8c9fbea5"
+dependencies = [
+ "zerofrom-derive",
+]
+
+[[package]]
+name = "zerofrom-derive"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d71e5d6e06ab090c67b5e44993ec16b72dcbaabc526db883a360057678b48502"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+ "synstructure",
+]
+
+[[package]]
+name = "zerotrie"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2a59c17a5562d507e4b54960e8569ebee33bee890c70aa3fe7b97e85a9fd7851"
+dependencies = [
+ "displaydoc",
+ "yoke",
+ "zerofrom",
+]
+
+[[package]]
+name = "zerovec"
+version = "0.11.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6c28719294829477f525be0186d13efa9a3c602f7ec202ca9e353d310fb9a002"
+dependencies = [
+ "yoke",
+ "zerofrom",
+ "zerovec-derive",
+]
+
+[[package]]
+name = "zerovec-derive"
+version = "0.11.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eadce39539ca5cb3985590102671f2567e659fca9666581ad3411d59207951f3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.117",
+]
+
+[[package]]
+name = "zmij"
+version = "1.0.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8848ee67ecc8aedbaf3e4122217aff892639231befc6a1b58d29fff4c2cabaa"
diff --git a/apps/desktop/src-tauri/Cargo.toml b/apps/desktop/src-tauri/Cargo.toml
new file mode 100644
index 00000000..5d994068
--- /dev/null
+++ b/apps/desktop/src-tauri/Cargo.toml
@@ -0,0 +1,23 @@
+[package]
+name = "btca-desktop"
+version = "0.0.1"
+description = "Desktop shell for the btca web app"
+authors = ["Ben Davis"]
+license = ""
+repository = "https://github.com/bmdavis419/better-context"
+edition = "2021"
+rust-version = "1.77.2"
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+
+[lib]
+name = "btca_desktop_lib"
+crate-type = ["staticlib", "cdylib", "rlib"]
+
+[build-dependencies]
+tauri-build = { version = "2.5.6", features = [] }
+
+[dependencies]
+log = "0.4"
+tauri = { version = "2.10.3", features = [] }
+tauri-plugin-log = "2"
diff --git a/apps/desktop/src-tauri/build.rs b/apps/desktop/src-tauri/build.rs
new file mode 100644
index 00000000..d860e1e6
--- /dev/null
+++ b/apps/desktop/src-tauri/build.rs
@@ -0,0 +1,3 @@
+fn main() {
+ tauri_build::build()
+}
diff --git a/apps/desktop/src-tauri/capabilities/default.json b/apps/desktop/src-tauri/capabilities/default.json
new file mode 100644
index 00000000..25be56e0
--- /dev/null
+++ b/apps/desktop/src-tauri/capabilities/default.json
@@ -0,0 +1,7 @@
+{
+ "$schema": "../gen/schemas/desktop-schema.json",
+ "identifier": "default",
+ "description": "enables the default permissions",
+ "windows": ["main"],
+ "permissions": ["core:default"]
+}
diff --git a/apps/desktop/src-tauri/icons/128x128.png b/apps/desktop/src-tauri/icons/128x128.png
new file mode 100644
index 00000000..077f94d8
Binary files /dev/null and b/apps/desktop/src-tauri/icons/128x128.png differ
diff --git a/apps/desktop/src-tauri/icons/128x128@2x.png b/apps/desktop/src-tauri/icons/128x128@2x.png
new file mode 100644
index 00000000..6205c907
Binary files /dev/null and b/apps/desktop/src-tauri/icons/128x128@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/32x32.png b/apps/desktop/src-tauri/icons/32x32.png
new file mode 100644
index 00000000..9716982d
Binary files /dev/null and b/apps/desktop/src-tauri/icons/32x32.png differ
diff --git a/apps/desktop/src-tauri/icons/64x64.png b/apps/desktop/src-tauri/icons/64x64.png
new file mode 100644
index 00000000..ed567efd
Binary files /dev/null and b/apps/desktop/src-tauri/icons/64x64.png differ
diff --git a/apps/desktop/src-tauri/icons/Square107x107Logo.png b/apps/desktop/src-tauri/icons/Square107x107Logo.png
new file mode 100644
index 00000000..2bf9d9b6
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square107x107Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square142x142Logo.png b/apps/desktop/src-tauri/icons/Square142x142Logo.png
new file mode 100644
index 00000000..01405445
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square142x142Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square150x150Logo.png b/apps/desktop/src-tauri/icons/Square150x150Logo.png
new file mode 100644
index 00000000..d9fa250e
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square150x150Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square284x284Logo.png b/apps/desktop/src-tauri/icons/Square284x284Logo.png
new file mode 100644
index 00000000..bd507c8d
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square284x284Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square30x30Logo.png b/apps/desktop/src-tauri/icons/Square30x30Logo.png
new file mode 100644
index 00000000..528e8c6e
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square30x30Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square310x310Logo.png b/apps/desktop/src-tauri/icons/Square310x310Logo.png
new file mode 100644
index 00000000..499c3cdf
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square310x310Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square44x44Logo.png b/apps/desktop/src-tauri/icons/Square44x44Logo.png
new file mode 100644
index 00000000..5deddee5
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square44x44Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square71x71Logo.png b/apps/desktop/src-tauri/icons/Square71x71Logo.png
new file mode 100644
index 00000000..fbac4107
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square71x71Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/Square89x89Logo.png b/apps/desktop/src-tauri/icons/Square89x89Logo.png
new file mode 100644
index 00000000..061a8b25
Binary files /dev/null and b/apps/desktop/src-tauri/icons/Square89x89Logo.png differ
diff --git a/apps/desktop/src-tauri/icons/StoreLogo.png b/apps/desktop/src-tauri/icons/StoreLogo.png
new file mode 100644
index 00000000..a4bd0a1e
Binary files /dev/null and b/apps/desktop/src-tauri/icons/StoreLogo.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml b/apps/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml
new file mode 100644
index 00000000..2ffbf24b
--- /dev/null
+++ b/apps/desktop/src-tauri/icons/android/mipmap-anydpi-v26/ic_launcher.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 00000000..dddf504b
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png
new file mode 100644
index 00000000..98c8f4df
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_foreground.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png
new file mode 100644
index 00000000..d8acd53a
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-hdpi/ic_launcher_round.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png
new file mode 100644
index 00000000..2ea088a1
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png
new file mode 100644
index 00000000..e4763a8f
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_foreground.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png
new file mode 100644
index 00000000..d02a8bb8
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-mdpi/ic_launcher_round.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 00000000..3b2e3135
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png
new file mode 100644
index 00000000..dffa11b7
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_foreground.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png
new file mode 100644
index 00000000..1efc95cf
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xhdpi/ic_launcher_round.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 00000000..6c59d988
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png
new file mode 100644
index 00000000..66d993c3
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_foreground.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png
new file mode 100644
index 00000000..5773582a
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxhdpi/ic_launcher_round.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 00000000..c3428131
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png
new file mode 100644
index 00000000..93bce634
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_foreground.png differ
diff --git a/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png
new file mode 100644
index 00000000..3dba5d79
Binary files /dev/null and b/apps/desktop/src-tauri/icons/android/mipmap-xxxhdpi/ic_launcher_round.png differ
diff --git a/apps/desktop/src-tauri/icons/android/values/ic_launcher_background.xml b/apps/desktop/src-tauri/icons/android/values/ic_launcher_background.xml
new file mode 100644
index 00000000..ea9c223a
--- /dev/null
+++ b/apps/desktop/src-tauri/icons/android/values/ic_launcher_background.xml
@@ -0,0 +1,4 @@
+
+
+ #fff
+
\ No newline at end of file
diff --git a/apps/desktop/src-tauri/icons/icon.icns b/apps/desktop/src-tauri/icons/icon.icns
new file mode 100644
index 00000000..fcd80f36
Binary files /dev/null and b/apps/desktop/src-tauri/icons/icon.icns differ
diff --git a/apps/desktop/src-tauri/icons/icon.ico b/apps/desktop/src-tauri/icons/icon.ico
new file mode 100644
index 00000000..f0313f81
Binary files /dev/null and b/apps/desktop/src-tauri/icons/icon.ico differ
diff --git a/apps/desktop/src-tauri/icons/icon.png b/apps/desktop/src-tauri/icons/icon.png
new file mode 100644
index 00000000..23e92d34
Binary files /dev/null and b/apps/desktop/src-tauri/icons/icon.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@1x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@1x.png
new file mode 100644
index 00000000..f1f6de46
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@1x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x-1.png b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x-1.png
new file mode 100644
index 00000000..b884b264
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x-1.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x.png
new file mode 100644
index 00000000..b884b264
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@3x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@3x.png
new file mode 100644
index 00000000..3b5ed7ca
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-20x20@3x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@1x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@1x.png
new file mode 100644
index 00000000..ac88f1b0
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@1x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x-1.png b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x-1.png
new file mode 100644
index 00000000..99e15f12
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x-1.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x.png
new file mode 100644
index 00000000..99e15f12
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@3x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@3x.png
new file mode 100644
index 00000000..75583d61
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-29x29@3x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@1x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@1x.png
new file mode 100644
index 00000000..b884b264
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@1x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x-1.png b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x-1.png
new file mode 100644
index 00000000..36f11bc6
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x-1.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x.png
new file mode 100644
index 00000000..36f11bc6
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@3x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@3x.png
new file mode 100644
index 00000000..4103a77c
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-40x40@3x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-512@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-512@2x.png
new file mode 100644
index 00000000..66685f89
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-512@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@2x.png
new file mode 100644
index 00000000..4103a77c
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@3x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@3x.png
new file mode 100644
index 00000000..b1933697
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-60x60@3x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@1x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@1x.png
new file mode 100644
index 00000000..8dd6453d
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@1x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@2x.png
new file mode 100644
index 00000000..aa3f0e45
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-76x76@2x.png differ
diff --git a/apps/desktop/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png b/apps/desktop/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png
new file mode 100644
index 00000000..7929c187
Binary files /dev/null and b/apps/desktop/src-tauri/icons/ios/AppIcon-83.5x83.5@2x.png differ
diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs
new file mode 100644
index 00000000..69f4491c
--- /dev/null
+++ b/apps/desktop/src-tauri/src/lib.rs
@@ -0,0 +1,16 @@
+#[cfg_attr(mobile, tauri::mobile_entry_point)]
+pub fn run() {
+ tauri::Builder::default()
+ .setup(|app| {
+ if cfg!(debug_assertions) {
+ app.handle().plugin(
+ tauri_plugin_log::Builder::default()
+ .level(log::LevelFilter::Info)
+ .build(),
+ )?;
+ }
+ Ok(())
+ })
+ .run(tauri::generate_context!())
+ .expect("error while running tauri application");
+}
diff --git a/apps/desktop/src-tauri/src/main.rs b/apps/desktop/src-tauri/src/main.rs
new file mode 100644
index 00000000..c736c21a
--- /dev/null
+++ b/apps/desktop/src-tauri/src/main.rs
@@ -0,0 +1,6 @@
+// Prevents additional console window on Windows in release, DO NOT REMOVE!!
+#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
+
+fn main() {
+ btca_desktop_lib::run();
+}
diff --git a/apps/desktop/src-tauri/tauri.conf.json b/apps/desktop/src-tauri/tauri.conf.json
new file mode 100644
index 00000000..7b153251
--- /dev/null
+++ b/apps/desktop/src-tauri/tauri.conf.json
@@ -0,0 +1,42 @@
+{
+ "$schema": "../node_modules/@tauri-apps/cli/config.schema.json",
+ "productName": "btca Desktop",
+ "version": "0.0.1",
+ "identifier": "dev.btca.desktop",
+ "build": {
+ "frontendDist": "../build",
+ "devUrl": "http://localhost:1420",
+ "beforeDevCommand": "bun run dev:ui",
+ "beforeBuildCommand": "bun run build:ui"
+ },
+ "app": {
+ "windows": [
+ {
+ "label": "main",
+ "title": "btca",
+ "url": "/app",
+ "width": 1440,
+ "height": 960,
+ "minWidth": 1200,
+ "minHeight": 780,
+ "resizable": true,
+ "fullscreen": false,
+ "center": true
+ }
+ ],
+ "security": {
+ "csp": null
+ }
+ },
+ "bundle": {
+ "active": true,
+ "targets": "all",
+ "icon": [
+ "icons/32x32.png",
+ "icons/128x128.png",
+ "icons/128x128@2x.png",
+ "icons/icon.icns",
+ "icons/icon.ico"
+ ]
+ }
+}
diff --git a/apps/desktop/src/app.d.ts b/apps/desktop/src/app.d.ts
new file mode 100644
index 00000000..da08e6da
--- /dev/null
+++ b/apps/desktop/src/app.d.ts
@@ -0,0 +1,13 @@
+// See https://svelte.dev/docs/kit/types#app.d.ts
+// for information about these interfaces
+declare global {
+ namespace App {
+ // interface Error {}
+ // interface Locals {}
+ // interface PageData {}
+ // interface PageState {}
+ // interface Platform {}
+ }
+}
+
+export {};
diff --git a/apps/desktop/src/app.html b/apps/desktop/src/app.html
new file mode 100644
index 00000000..21441d3b
--- /dev/null
+++ b/apps/desktop/src/app.html
@@ -0,0 +1,25 @@
+
+
+
+
+
+
+
+ %sveltekit.head%
+
+
+ %sveltekit.body%
+
+
diff --git a/apps/desktop/src/lib/CopyButton.svelte b/apps/desktop/src/lib/CopyButton.svelte
new file mode 100644
index 00000000..e8566453
--- /dev/null
+++ b/apps/desktop/src/lib/CopyButton.svelte
@@ -0,0 +1,30 @@
+
+
+
+ {#if copied}
+
+ {:else}
+
+ {/if}
+
diff --git a/apps/desktop/src/lib/assets/cli-showcase.png b/apps/desktop/src/lib/assets/cli-showcase.png
new file mode 100644
index 00000000..85b99882
Binary files /dev/null and b/apps/desktop/src/lib/assets/cli-showcase.png differ
diff --git a/apps/desktop/src/lib/assets/favicon.svg b/apps/desktop/src/lib/assets/favicon.svg
new file mode 100644
index 00000000..ea88bdf2
--- /dev/null
+++ b/apps/desktop/src/lib/assets/favicon.svg
@@ -0,0 +1,11 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/apps/desktop/src/lib/assets/mcp-showcase.png b/apps/desktop/src/lib/assets/mcp-showcase.png
new file mode 100644
index 00000000..86b82c7a
Binary files /dev/null and b/apps/desktop/src/lib/assets/mcp-showcase.png differ
diff --git a/apps/desktop/src/lib/assets/og.png b/apps/desktop/src/lib/assets/og.png
new file mode 100644
index 00000000..4e10c65d
Binary files /dev/null and b/apps/desktop/src/lib/assets/og.png differ
diff --git a/apps/desktop/src/lib/assets/rules/better_context.mdc b/apps/desktop/src/lib/assets/rules/better_context.mdc
new file mode 100644
index 00000000..e4c38c5e
--- /dev/null
+++ b/apps/desktop/src/lib/assets/rules/better_context.mdc
@@ -0,0 +1,235 @@
+---
+description: Use btca to fetch up-to-date answers about external libraries/frameworks by searching their official repos locally (source-first, no web browsing).
+alwaysApply: true
+---
+
+# btca – Better Context Agent
+
+btca queries local clones of library/framework repositories to answer questions with up-to-date, source-first information. Use it instead of web searches when the user needs accurate, current details about technologies.
+
+## When to Use btca
+
+- User asks "how does X work in Svelte 5?" or similar framework/library questions
+- User needs current API behavior that may differ from your training data
+- User explicitly says "use btca" or references a configured resource
+- You need to verify syntax or patterns for a specific library version
+
+## Agent Behavior
+
+- **Prefer `btca ask`** over the interactive TUI (agents can't operate TUIs reliably)
+- **Ask for missing information** – if the user wants to add/change resources or models but doesn't provide all required details, ask them before running commands
+- Run `btca resources` first if you're unsure which resources are available
+
+---
+
+- If unsure what's configured: `btca resources`
+- If missing: add it with `btca add --name svelte --type git --url https://github.com/sveltejs/svelte --branch main`
+
+### `btca ask` (Primary Command for Agents)
+
+- `btca ask --resource --question ''`
+- For multiple resources: `btca ask --resource svelte --resource effect --question ''`
+- You can also use @mentions in the question: `btca ask --question 'How do @svelte stores work?'`
+
+| Flag | Required | Description |
+| -------------------------- | -------- | ---------------------------------------------------------------------------- |
+| `-q, --question ` | Yes | The question to ask |
+| `-r, --resource ` | No | Resource(s) to search; repeatable. If omitted, uses all configured resources |
+
+**Features:**
+
+- Supports `@mentions` in the question: `btca ask -q "@svelte @tailwind How do I style components?"`
+- Combine `-r` flags with `@mentions`; duplicates are deduplicated
+
+# **Examples:**
+
+- `btca ask --question '' [--resource ...]`
+ - Ask one question and stream the answer.
+ - `--resource` can be repeated for multiple resources.
+ - Supports @mentions in the question to specify resources.
+
+```bash
+# Single resource
+btca ask -r svelte -q "How does the $state rune work?"
+
+# One-shot GitHub repository
+btca ask -r https://github.com/sveltejs/svelte.dev -q "How do I setup a new sveltekit project?"
+
+# Multiple resources
+btca ask -r react -r typescript -q "How do I type useState?"
+
+# Using @mentions instead of -r flags
+btca ask -q "@svelte @effect How do I integrate Effect with Svelte?"
+```
+
+### Anonymous URL resources
+
+`-r` also accepts HTTPS Git URLs directly (for one-shot use). These URLs are cached under your normal resources directory and are not written to config.
+
+---
+
+### `btca resources`
+
+List all configured resources. **Run this first** to see what's available.
+
+```bash
+btca resources
+```
+
+---
+
+### `btca add`
+
+Add a new resource. **Ask the user for missing details** if they don't provide them.
+
+```bash
+# Git resource (all flags shown)
+btca add -n -t git -u -b [--search-path ] [--notes ]
+
+# Local resource
+btca add -n -t local --path [--notes ]
+```
+
+| Flag | Required | Description |
+| ----------------------- | ----------- | ------------------------------------------------------------------------------------ |
+| `-n, --name ` | Yes | Unique resource name (Camelcase! and alphanumeric + hyphens, must start with letter) |
+| `-t, --type ` | Yes | `git` or `local` |
+| `-u, --url ` | Yes (git) | HTTPS git repository URL |
+| `-b, --branch ` | No (git) | Branch to clone (default: `main`) |
+| `--path ` | Yes (local) | Absolute path to local directory |
+| `--search-path ` | No | Subdirectory within repo to focus searches on |
+| `--notes ` | No | Hints for the AI about this resource (e.g., "Focus on the docs/ directory") |
+
+**If user says "add svelte" without details, ask:**
+
+> Which repository URL should I use? Common choice for Svelte docs: `https://github.com/sveltejs/svelte.dev` with `searchPath: apps/svelte.dev`. Should I use that, or do you have a different repo in mind? Ask for each required information if missing! Do not assume!
+
+---
+
+### `btca remove`
+
+Remove a resource.
+
+```bash
+btca remove
+```
+
+---
+
+### `btca connect`
+
+Change the AI model/provider.
+
+```bash
+btca connect -p -m
+```
+
+| Flag | Required | Description |
+| --------------------- | -------- | ------------------------------------------------- |
+| `-p, --provider ` | Yes | Provider ID (e.g., `opencode`, `anthropic`) |
+| `-m, --model ` | Yes | Model ID (e.g., `claude-haiku-4-5`, `big-pickle`) |
+
+**If user says "change the model" without specifying which, ask:**
+
+> Which model would you like to use? Recommended options with `opencode` provider:
+>
+> - `claude-haiku-4-5` – fast, highly recommended
+> - `big-pickle` – free, surprisingly good (default)
+> - `minimax-m2.1-free` – very fast, very cheap
+> - `kimi-k2` – no reasoning
+
+---
+
+### `btca clear`
+
+Clear all locally cloned git repositories (frees disk space, forces fresh clone on next use).
+
+```bash
+btca clear
+```
+
+---
+
+### `btca serve`
+
+Start btca server as a persistent process (for external integrations).
+
+```bash
+btca serve [-p ]
+```
+
+Default port: 8080. Server runs until `Ctrl+C`.
+
+---
+
+### Global Flags
+
+These apply to all subcommands:
+
+| Flag | Description |
+| ---------------- | --------------------------------------------------------------- |
+| `--server ` | Connect to an existing btca server instead of auto-starting one |
+| `--port ` | Port for the auto-started server (default: random 3000-6000) |
+| `-v, --version` | Print version number |
+
+---
+
+## Config Locations
+
+| Location | Purpose |
+| ---------------------------------- | --------------------------------------------- |
+| `~/.config/btca/btca.config.jsonc` | Global config (used when no project config) |
+| `./btca.config.jsonc` | Project config (overrides global per-project) |
+
+When both exist, resources merge by name (project overrides global).
+
+---
+
+## TUI Quick Reference
+
+If the user wants to use the interactive TUI (`btca` without subcommands):
+
+| Feature | How to use |
+| --------- | ------------------------------------------------ |
+| @Mentions | `@svelte How do I...?` – first message needs one |
+| Commands | Press `/` to open command palette (connect, add, clear, resume) |
+| `Escape` | Cancel streaming (press twice to confirm) |
+| `Ctrl+Q` | Quit |
+
+---
+
+## Troubleshooting
+
+| Problem | Solution |
+| -------------------------- | -------------------------------------------------------------------------- |
+| "No resources configured" | Run `btca add ...` or check if `btca.config.jsonc` exists |
+| "Provider not connected" | Run `opencode auth` in terminal to configure provider credentials |
+| Stale/corrupted cache | Run `btca clear` to reset |
+| Deprecated `-t` flag error | Use `-r` instead: `btca ask -r svelte ...` |
+
+---
+
+## Quick Reference
+
+```bash
+# Ask a question
+btca ask -r -q ""
+
+# List resources
+btca resources
+
+# Add git resource
+btca add -n -t git -u -b
+
+# Add local resource
+btca add -n -t local --path
+
+# Remove resource
+btca remove
+
+# Change model
+btca connect -p opencode -m claude-haiku-4-5
+
+# Clear cache
+btca clear
+```
diff --git a/apps/desktop/src/lib/assets/web-showcase.png b/apps/desktop/src/lib/assets/web-showcase.png
new file mode 100644
index 00000000..2f6a5439
Binary files /dev/null and b/apps/desktop/src/lib/assets/web-showcase.png differ
diff --git a/apps/desktop/src/lib/billing/plans.ts b/apps/desktop/src/lib/billing/plans.ts
new file mode 100644
index 00000000..a9213ac1
--- /dev/null
+++ b/apps/desktop/src/lib/billing/plans.ts
@@ -0,0 +1,24 @@
+import { PRO_AI_BUDGET_MICROS, PRO_AI_BUDGET_USD } from '@btca/convex/aiBudget';
+import { WEB_SANDBOX_MODELS } from '@btca/convex/webSandboxModels';
+
+export const BILLING_PLAN = {
+ id: 'btca_pro',
+ name: 'Pro',
+ priceUsd: 8,
+ interval: 'month',
+ aiBudgetUsd: PRO_AI_BUDGET_USD,
+ aiBudgetMicros: PRO_AI_BUDGET_MICROS,
+ models: WEB_SANDBOX_MODELS.map(({ id, label, tier, description }) => ({
+ id,
+ label,
+ tier,
+ description
+ }))
+} as const;
+
+export const FEATURE_IDS = {
+ aiBudget: 'ai_budget',
+ chatMessages: 'chat_messages'
+} as const;
+
+export const SUPPORT_URL = 'https://x.com/davis7';
diff --git a/apps/desktop/src/lib/billing/types.ts b/apps/desktop/src/lib/billing/types.ts
new file mode 100644
index 00000000..b7d28354
--- /dev/null
+++ b/apps/desktop/src/lib/billing/types.ts
@@ -0,0 +1,33 @@
+export type UsageMetric = {
+ remainingPct: number;
+ usedPct: number;
+ isDepleted: boolean;
+};
+
+export type BillingSummary = {
+ plan: 'pro' | 'free' | 'none';
+ status: 'active' | 'trialing' | 'canceled' | 'none';
+ currentPeriodEnd?: number;
+ canceledAt?: number;
+ customer?: {
+ name?: string | null;
+ email?: string | null;
+ };
+ paymentMethod?: {
+ type: string;
+ card?: {
+ brand: string;
+ last4: string;
+ exp_month: number;
+ exp_year: number;
+ };
+ } | null;
+ usage: {
+ aiBudget: UsageMetric;
+ };
+ freeMessages?: {
+ used: number;
+ total: number;
+ remaining: number;
+ };
+};
diff --git a/apps/desktop/src/lib/clerk.ts b/apps/desktop/src/lib/clerk.ts
new file mode 100644
index 00000000..ea3b67ab
--- /dev/null
+++ b/apps/desktop/src/lib/clerk.ts
@@ -0,0 +1,60 @@
+import { PUBLIC_CLERK_PUBLISHABLE_KEY } from '$env/static/public';
+import { Clerk } from '@clerk/clerk-js';
+import { dev } from '$app/environment';
+
+let clerkInstance: Clerk | null = null;
+let initPromise: Promise | null = null;
+
+function suppressBeforeUnloadInDev() {
+ if (!dev || typeof window === 'undefined') return;
+
+ const originalAddEventListener = window.addEventListener.bind(window);
+ window.addEventListener = ((
+ type: string,
+ listener: EventListenerOrEventListenerObject,
+ options?: boolean | AddEventListenerOptions
+ ) => {
+ if (type === 'beforeunload') return;
+ return originalAddEventListener(type, listener, options);
+ }) as typeof window.addEventListener;
+
+ Object.defineProperty(window, 'onbeforeunload', {
+ set: () => {},
+ get: () => null
+ });
+}
+
+export async function initializeClerk(): Promise {
+ if (clerkInstance?.loaded) {
+ return clerkInstance;
+ }
+
+ if (initPromise) {
+ return initPromise;
+ }
+
+ suppressBeforeUnloadInDev();
+
+ initPromise = (async () => {
+ clerkInstance = new Clerk(PUBLIC_CLERK_PUBLISHABLE_KEY);
+ await clerkInstance.load();
+
+ return clerkInstance;
+ })();
+
+ return initPromise;
+}
+
+/**
+ * Get the Clerk instance (must be initialized first)
+ */
+export function getClerk(): Clerk | null {
+ return clerkInstance;
+}
+
+/**
+ * Check if Clerk is loaded
+ */
+export function isClerkLoaded(): boolean {
+ return clerkInstance?.loaded ?? false;
+}
diff --git a/apps/desktop/src/lib/components/AddResourceModal.svelte b/apps/desktop/src/lib/components/AddResourceModal.svelte
new file mode 100644
index 00000000..d193bfba
--- /dev/null
+++ b/apps/desktop/src/lib/components/AddResourceModal.svelte
@@ -0,0 +1,453 @@
+
+
+{#if isOpen}
+ {
+ if (event.key === 'Enter' || event.key === ' ') {
+ event.preventDefault();
+ closeModal();
+ }
+ }}
+ >
+
+
event.stopPropagation()}
+ onkeydown={handleKeydown}
+ >
+
+
+
+ {#if resourceType === 'git'}
+
+ {:else}
+
+ {/if}
+
+
+
Add a resource
+
+ {resourceType === 'git'
+ ? 'Connect a git repository so btca can index it.'
+ : 'Attach an npm package so btca can use its published docs and metadata.'}
+
+
+
+
+
+
+
+
+
+
+
resetForm('git')}
+ >
+
+
+ Git Repository
+
+ Repo URL and branch.
+
+
resetForm('npm')}
+ >
+
+ Package name and optional version.
+
+
+
+ {#if resourceType === 'git'}
+
+
+ Git URL
+
+
+ {#if parseError}
+
{parseError}
+ {/if}
+ {#if detectedRepo}
+
+
+
+ Detected {detectedRepo} · default branch {branchName}
+
+
+ {/if}
+
+ {:else}
+
+
+ npm package
+
+
+
+ {/if}
+
+
+
+
+ Resource name
+
+
+
Use @mention: @{resourceName || 'resource'}
+
+ {#if resourceType === 'git'}
+
+ {:else}
+
+
+ Version or tag
+
+
+
+ {/if}
+
+
+ {#if submitError}
+
+ {submitError}
+
+ {/if}
+
+
+
+
+ {#if selectedProject}
+ This resource will be added to {selectedProject.name}
+ and synced onto your instance after the next chat mention.
+ {:else}
+ We'll sync this resource onto your instance after the next chat mention.
+ {/if}
+
+
+
+ Cancel
+
+
+ {#if isSubmitting}
+
+ Adding...
+ {:else}
+ Add {resourceType === 'git' ? 'repository' : 'package'}
+ {/if}
+
+
+
+
+
+{/if}
diff --git a/apps/desktop/src/lib/components/ChatMessages.svelte b/apps/desktop/src/lib/components/ChatMessages.svelte
new file mode 100644
index 00000000..90a87726
--- /dev/null
+++ b/apps/desktop/src/lib/components/ChatMessages.svelte
@@ -0,0 +1,561 @@
+
+
+
+
+
+
+ {#if !isAtBottom}
+
scrollToBottom()}
+ class="absolute bottom-4 left-1/2 flex -translate-x-1/2 items-center gap-1.5 border border-[hsl(var(--bc-border))] bg-[hsl(var(--bc-surface))] px-3 py-1.5 text-xs font-medium shadow-md transition-colors hover:border-[hsl(var(--bc-fg))] hover:bg-[hsl(var(--bc-surface-2))]"
+ >
+
+ Jump to bottom
+
+ {/if}
+
diff --git a/apps/desktop/src/lib/components/CommandPalette.svelte b/apps/desktop/src/lib/components/CommandPalette.svelte
new file mode 100644
index 00000000..3065fccc
--- /dev/null
+++ b/apps/desktop/src/lib/components/CommandPalette.svelte
@@ -0,0 +1,335 @@
+
+
+{#if isOpen}
+
+
e.stopPropagation()}
+ onkeydown={handleKeydown}
+ >
+
+
+
+
+ Esc
+
+
+
+
+ {#if allItems.length === 0}
+
+ No results for "{searchQuery} "
+
+ {:else}
+ {#each groupedItems as group (group.name)}
+
+ {group.name}
+
+ {#each group.items as item (item.id)}
+ {@const isSelected = item.globalIndex === selectedIndex}
+ {@const Icon = item.icon}
+
item.onSelect()}
+ onmouseenter={() => (selectedIndex = item.globalIndex)}
+ >
+
+
+ {item.label}
+
+ {#if item.sublabel}
+
+ {item.sublabel}
+
+ {/if}
+
+ {/each}
+ {/each}
+ {/if}
+
+
+
+
+ ↑↓
+ navigate
+
+
+
+
+
+ select
+
+
+ Esc
+ close
+
+
+
+
+{/if}
diff --git a/apps/desktop/src/lib/components/CreateProjectModal.svelte b/apps/desktop/src/lib/components/CreateProjectModal.svelte
new file mode 100644
index 00000000..887d9553
--- /dev/null
+++ b/apps/desktop/src/lib/components/CreateProjectModal.svelte
@@ -0,0 +1,91 @@
+
+
+{#if projectStore.showCreateModal}
+ e.key === 'Escape' && closeModal()}
+ >
+
+
Create New Project
+
+
+
+
+{/if}
diff --git a/apps/desktop/src/lib/components/Dropdown.svelte b/apps/desktop/src/lib/components/Dropdown.svelte
new file mode 100644
index 00000000..577f0b83
--- /dev/null
+++ b/apps/desktop/src/lib/components/Dropdown.svelte
@@ -0,0 +1,178 @@
+
+
+
+
+ {displayLabel}
+
+
+
+ {#if open}
+
+ {/if}
+
+
+
diff --git a/apps/desktop/src/lib/components/InstanceCard.svelte b/apps/desktop/src/lib/components/InstanceCard.svelte
new file mode 100644
index 00000000..8a27c168
--- /dev/null
+++ b/apps/desktop/src/lib/components/InstanceCard.svelte
@@ -0,0 +1,475 @@
+
+
+
+
+
+
(isExpanded = !isExpanded)}
+ >
+
+
+
+
+
+
Instance
+
Your dedicated btca runtime
+
+
+
+
+
+
+ {stateInfo.label}
+
+ {#if instanceStore.updateAvailable}
+ Update available
+ {/if}
+
+
+
+
+
+
+ {#if isExpanded}
+
(isExpanded = false)}
+ aria-label="Close dropdown"
+ >
+
+ {#if instanceStore.isLoading || instanceStore.isBootstrapping}
+
+
+ {instanceStore.isBootstrapping
+ ? 'Setting up your instance...'
+ : 'Loading instance data...'}
+
+ {:else if !hasInstance}
+
+
+
+
+
+
+
Instance not found
+
+ Your instance hasn't been set up yet. This usually happens automatically.
+
+
instanceStore.ensureExists()}
+ >
+
+ Set up instance
+
+ {#if instanceStore.error}
+
{instanceStore.error}
+ {/if}
+
+
+
+ {:else}
+ {#if instanceStore.state === 'error'}
+
+
+
+
+
Instance needs attention
+
{errorText}
+
+ {#if isDiskFullError}
+ Reset your instance to clear cached repos and keep going.
+ {:else}
+ Reset your instance to rebuild it from scratch.
+ {/if}
+
+
runAction('reset')}
+ disabled={!canReset || pendingAction !== null}
+ >
+ {#if pendingAction === 'reset'}
+
+ Resetting...
+ {:else}
+
+ Reset instance
+ {/if}
+
+
+
+
+ {/if}
+
+
+
+
+
Status
+
{stateInfo.description}
+
State: {instanceStore.state}
+
+
+
+
+
+
+
+
+
+ Last active: {formatDateTime(instanceStore.instance?.lastActiveAt)}
+
+
+ {#if formatServerHost(instanceStore.instance?.serverUrl)}
+
+
+ {formatServerHost(instanceStore.instance?.serverUrl)}
+
+ {/if}
+
+
+
+
+
Controls
+
+
runAction('wake')}
+ disabled={!canWake || isTransitioning || pendingAction !== null}
+ >
+ {#if pendingAction === 'wake'}
+
+ Waking...
+ {:else}
+
+ Wake instance
+ {/if}
+
+
runAction('stop')}
+ disabled={!canStop || isTransitioning || pendingAction !== null}
+ >
+ {#if pendingAction === 'stop'}
+
+ Stopping...
+ {:else}
+
+ Stop instance
+ {/if}
+
+
runAction('update')}
+ disabled={!canUpdate || isTransitioning || pendingAction !== null}
+ >
+ {#if pendingAction === 'update'}
+
+ Updating...
+ {:else}
+
+ Update packages
+ {/if}
+
+
+ {#if instanceStore.state !== 'error' && (instanceStore.error || instanceStore.instance?.errorMessage)}
+
+
+
{instanceStore.error ?? instanceStore.instance?.errorMessage}
+
+ {/if}
+
+
+
+
+
Versions
+
+
+
btca
+
+ {instanceStore.btcaVersion ?? 'Unknown'}
+ {#if instanceStore.btcaUpdateAvailable}
+
+ → {instanceStore.latestBtcaVersion}
+
+ {/if}
+
+
+
+
opencode
+
+ {instanceStore.opencodeVersion ?? 'Unknown'}
+ {#if instanceStore.opencodeUpdateAvailable}
+
+ → {instanceStore.latestOpencodeVersion}
+
+ {/if}
+
+
+
+
+ Last check: {formatDateTime(instanceStore.instance?.lastVersionCheck)}
+
+
+
+
+
+
+
+
Cached resources
+
+
Manage
+
+ {#if instanceStore.cachedResources.length === 0}
+
+ No cached resources yet. Mention a resource in chat to cache it.
+
+ {:else}
+
+ {#each displayedResources as resource (resource._id)}
+
+
+ @{resource.name}
+ {getCachedResourceMeta(resource)}
+
+
+ Last used: {formatDateTime(resource.lastUsedAt)}
+
+
+ {/each}
+ {#if instanceStore.cachedResources.length > displayedResources.length}
+
+ +{instanceStore.cachedResources.length - displayedResources.length} more cached
+
+ {/if}
+
+ {/if}
+
+ {/if}
+
+ {/if}
+
diff --git a/apps/desktop/src/lib/components/InstanceStatus.svelte b/apps/desktop/src/lib/components/InstanceStatus.svelte
new file mode 100644
index 00000000..c24ed51b
--- /dev/null
+++ b/apps/desktop/src/lib/components/InstanceStatus.svelte
@@ -0,0 +1,338 @@
+
+
+
+
+
+
(isExpanded = !isExpanded)}
+ >
+
+
+
+
+
+
Instance
+
Dedicated runtime
+
+
+
+
+
+ {stateInfo.label}
+
+ {#if instanceStore.updateAvailable}
+ Update
+ {/if}
+
+
+
+ {#if isExpanded}
+
+ {/if}
+
diff --git a/apps/desktop/src/lib/components/ProjectModelPicker.svelte b/apps/desktop/src/lib/components/ProjectModelPicker.svelte
new file mode 100644
index 00000000..0c41c536
--- /dev/null
+++ b/apps/desktop/src/lib/components/ProjectModelPicker.svelte
@@ -0,0 +1,194 @@
+
+
+{#if selectedProject}
+
+
+ {#if isSaving}
+
+ {/if}
+ {activeModel.label}
+ {activeModel.description}
+
+
+
+ {#if isOpen}
+
+ {/if}
+
+{/if}
+
+
diff --git a/apps/desktop/src/lib/components/ProjectSelector.svelte b/apps/desktop/src/lib/components/ProjectSelector.svelte
new file mode 100644
index 00000000..facb5a2d
--- /dev/null
+++ b/apps/desktop/src/lib/components/ProjectSelector.svelte
@@ -0,0 +1,174 @@
+
+
+
+
+
+
+
+
+
+ {#if projectStore.isLoading}
+ Loading...
+ {:else if projectStore.selectedProject}
+ {projectStore.selectedProject.name}
+ {#if projectStore.selectedProject.isDefault}
+ (default)
+ {/if}
+ {:else}
+ No project
+ {/if}
+
+
+
+
+
+ {#if isOpen}
+
+ {#each projectStore.projects as project (project._id)}
+
selectProject(project._id)}
+ >
+
+ {project.name}
+ {#if project.isDefault}
+ (default)
+ {/if}
+
+ {/each}
+
+
+
+
+
+ Create new project
+
+
+ {/if}
+
+
+{#if showCreateModal}
+
+ e.key === 'Escape' && (showCreateModal = false)}
+ >
+
+
Create New Project
+
+
+
+
+{/if}
diff --git a/apps/desktop/src/lib/components/ProvisioningModal.svelte b/apps/desktop/src/lib/components/ProvisioningModal.svelte
new file mode 100644
index 00000000..d91f0aef
--- /dev/null
+++ b/apps/desktop/src/lib/components/ProvisioningModal.svelte
@@ -0,0 +1,166 @@
+
+
+{#if isVisible}
+
+
+
+
+
+
+
+
+
Provisioning your instance
+
This happens once for new accounts. Keep this tab open.
+
+
+
+ {statusLabel}
+
+
+
+
+
+ {progressCopy}
+ {progressPercent}%
+
+
+
+
+
+ {#each steps as step}
+
+
+ {#if step.isComplete}
+
+ {:else if step.isActive}
+
+ {:else}
+
+ {/if}
+
+
+
{step.title}
+
{step.description}
+
+
+ {/each}
+
+
+
+ We will close this once provisioning completes. You can keep exploring in another tab.
+
+
+
+{/if}
diff --git a/apps/desktop/src/lib/components/ResourceLogo.svelte b/apps/desktop/src/lib/components/ResourceLogo.svelte
new file mode 100644
index 00000000..ef1086e6
--- /dev/null
+++ b/apps/desktop/src/lib/components/ResourceLogo.svelte
@@ -0,0 +1,85 @@
+
+
+
+ {#if Logo}
+
+ {:else}
+
+
+
+
+
+
+
+ {/if}
+
+
+
diff --git a/apps/desktop/src/lib/components/Sidebar.svelte b/apps/desktop/src/lib/components/Sidebar.svelte
new file mode 100644
index 00000000..d8e7e368
--- /dev/null
+++ b/apps/desktop/src/lib/components/Sidebar.svelte
@@ -0,0 +1,443 @@
+
+
+
+
+
diff --git a/apps/desktop/src/lib/components/ToolCallSummary.svelte b/apps/desktop/src/lib/components/ToolCallSummary.svelte
new file mode 100644
index 00000000..b7861c47
--- /dev/null
+++ b/apps/desktop/src/lib/components/ToolCallSummary.svelte
@@ -0,0 +1,40 @@
+
+
+{#if summaryItems.length > 0}
+
+{/if}
diff --git a/apps/desktop/src/lib/components/pricing/PricingPlans.svelte b/apps/desktop/src/lib/components/pricing/PricingPlans.svelte
new file mode 100644
index 00000000..8ba921fd
--- /dev/null
+++ b/apps/desktop/src/lib/components/pricing/PricingPlans.svelte
@@ -0,0 +1,148 @@
+
+
+
+
+
+
+
+
+
+ For developers who want grounded answers from real codebases
+
+
+ Search repos, save threads, and use btca as your codebase research layer.
+
+
+
+
+
+
+
+
+
Try btca on a real codebase
+
+
+
+ Limited messages to test the workflow
+
+
+
+ Grounded codebase search
+
+
+
+ Pick from low, balanced, and high-end web models
+
+
+
+ Good for evaluating the product before you commit to a paid workflow.
+
+
+
+
+
+
+
Pro
+
${BILLING_PLAN.priceUsd}
+
per month
+
+
Cancel anytime
+
+
For solo developers doing ongoing codebase research
+
+ {#each features as feature}
+
+
+ {feature}
+
+ {/each}
+
+
+ Monthly AI usage is shared across all models. Higher-end models use it faster.
+
+ {#if errorMessage}
+
{errorMessage}
+ {/if}
+ {#if isSubscribed}
+
View usage
+ {:else}
+
+ {#if isRedirecting}
+
+ Starting checkout...
+ {:else if !isSignedIn}
+ Start with Free
+ {:else}
+ Upgrade to Pro
+ {/if}
+
+ {/if}
+
+
+
diff --git a/apps/desktop/src/lib/config.ts b/apps/desktop/src/lib/config.ts
new file mode 100644
index 00000000..25de80e4
--- /dev/null
+++ b/apps/desktop/src/lib/config.ts
@@ -0,0 +1,11 @@
+import { dev } from '$app/environment';
+import { env } from '$env/dynamic/public';
+
+const trimTrailingSlash = (value: string) => value.replace(/\/+$/, '');
+
+const configuredBackendBaseUrl = trimTrailingSlash(env.PUBLIC_BACKEND_BASE_URL ?? '');
+const defaultBackendBaseUrl = dev ? 'http://localhost:5173' : 'https://btca.dev';
+
+export const backendBaseUrl = configuredBackendBaseUrl || defaultBackendBaseUrl;
+
+export const mcpBaseUrl = `${backendBaseUrl}/api/mcp`;
diff --git a/apps/desktop/src/lib/result/http.ts b/apps/desktop/src/lib/result/http.ts
new file mode 100644
index 00000000..468745c7
--- /dev/null
+++ b/apps/desktop/src/lib/result/http.ts
@@ -0,0 +1,172 @@
+import { Result } from 'better-result';
+
+import { toWebError, type WebError } from '@btca/convex/errors';
+
+type ConvexResultError = WebError | Error | string;
+
+type ConvexResultObject = { ok: boolean; error?: unknown; [key: string]: unknown };
+
+type ConvexActionResult = T extends { ok: false }
+ ? never
+ : T extends { ok: true }
+ ? Omit
+ : T;
+
+const toErrorMessage = (error: unknown): string => {
+ if (typeof error === 'string') {
+ return error;
+ }
+
+ if (error instanceof Error) {
+ return error.message || 'Unknown error';
+ }
+
+ if (
+ error &&
+ typeof error === 'object' &&
+ 'message' in error &&
+ typeof error.message === 'string'
+ ) {
+ return error.message;
+ }
+
+ return 'Unknown error';
+};
+
+const getConvexErrorStatus = (cause: unknown): number | null => {
+ if (!cause || typeof cause !== 'object') return null;
+
+ const raw = cause as {
+ code?: unknown;
+ error?: unknown;
+ message?: unknown;
+ };
+
+ const code = typeof raw.code === 'string' ? raw.code.toLowerCase() : '';
+ if (!code) return null;
+
+ if (code === 'unauthorized' || code === 'forbidden') return code === 'forbidden' ? 403 : 401;
+ if (code === 'not_found') return 404;
+ if (code === 'already_exists') return 409;
+ return null;
+};
+
+export const toResult = (operation: () => T) => {
+ try {
+ return Result.ok(operation());
+ } catch (error) {
+ return Result.err(toWebError(error));
+ }
+};
+
+export const toResultAsync = (operation: () => Promise) => {
+ return operation()
+ .then((value) => Result.ok(value))
+ .catch((error) => Result.err(toWebError(error)));
+};
+
+const isConvexResultObject = (result: unknown): result is ConvexResultObject => {
+ if (!result || typeof result !== 'object') {
+ return false;
+ }
+ return 'ok' in result;
+};
+
+const toActionSuccess = (result: T): ConvexActionResult => {
+ const value = result as ConvexResultObject;
+ if (!isConvexResultObject(result)) {
+ return result as ConvexActionResult;
+ }
+
+ if (value.ok !== true) {
+ return undefined as unknown as ConvexActionResult;
+ }
+
+ const withoutOk = { ...value } as Record;
+ delete withoutOk.ok;
+ return withoutOk as ConvexActionResult;
+};
+
+export const extractApiKey = (request: Request): string | null => {
+ const authHeader = request.headers.get('Authorization');
+ if (!authHeader?.startsWith('Bearer ')) {
+ return null;
+ }
+ return authHeader.slice(7) || null;
+};
+
+export const jsonResponse = (data: unknown, status = 200): Response => {
+ return new Response(JSON.stringify(data), {
+ status,
+ headers: {
+ 'Content-Type': 'application/json'
+ }
+ });
+};
+
+export const jsonError = (status: number, error: string): Response =>
+ jsonResponse({ error }, status);
+
+export const mapCliErrorStatus = (error: string): number => {
+ const lower = error.toLowerCase();
+ if (lower.includes('valid') || lower.includes('unauthorized')) return 401;
+ if (lower.includes('not found')) return 404;
+ if (lower.includes('forbidden')) return 403;
+ if (lower.includes('already exists')) return 409;
+ return 400;
+};
+
+export const runConvexActionResult = async (
+ action: () => Promise
+): Promise, ConvexResultError>> => {
+ const wrapped = await toResultAsync(action);
+ if (Result.isError(wrapped)) {
+ return Result.err(toWebError(wrapped.error));
+ }
+
+ const value = wrapped.value;
+ if (isConvexResultObject(value) && value.ok === false) {
+ return Result.err(toWebError(toErrorMessage(value.error ?? 'Unknown error')));
+ }
+
+ return Result.ok(toActionSuccess(wrapped.value));
+};
+
+export const runConvexQueryResult = async (
+ query: () => Promise
+): Promise, ConvexResultError>> => {
+ const wrapped = await toResultAsync(query);
+ if (Result.isError(wrapped)) {
+ return Result.err(toWebError(wrapped.error));
+ }
+
+ const value = wrapped.value;
+ if (isConvexResultObject(value) && value.ok === false) {
+ return Result.err(toWebError(toErrorMessage(value.error ?? 'Unknown error')));
+ }
+
+ return Result.ok(toActionSuccess(wrapped.value));
+};
+
+export const handleConvexRouteResult = (
+ result: Result,
+ options: {
+ onOk: (value: T) => unknown;
+ mapErrorStatus?: (error: string, cause?: unknown) => number;
+ }
+): Response => {
+ if (Result.isError(result)) {
+ const message = toErrorMessage(result.error);
+ const status =
+ (options.mapErrorStatus ?? mapCliErrorStatus)(message, result.error) ??
+ getConvexErrorStatus(result.error) ??
+ mapCliErrorStatus(message);
+ return jsonError(status, message);
+ }
+
+ const body = options.onOk(result.value);
+ return body instanceof Response ? body : jsonResponse(body);
+};
+
+export const toWebErrorResponse = (error: WebError, fallback = 500): Response =>
+ jsonError(fallback, error.message);
diff --git a/apps/desktop/src/lib/shiki/chatHighlighter.ts b/apps/desktop/src/lib/shiki/chatHighlighter.ts
new file mode 100644
index 00000000..c491e080
--- /dev/null
+++ b/apps/desktop/src/lib/shiki/chatHighlighter.ts
@@ -0,0 +1,49 @@
+import { createHighlighter, type Highlighter } from 'shiki';
+
+let highlighterPromise: Promise | null = null;
+let highlighterInstance: Highlighter | null = null;
+let generation = 0;
+
+export const getChatHighlighter = () => {
+ if (!highlighterPromise) {
+ generation += 1;
+ const current = generation;
+ highlighterPromise = createHighlighter({
+ themes: ['dark-plus', 'light-plus'],
+ langs: [
+ 'elixir',
+ 'typescript',
+ 'tsx',
+ 'svelte',
+ 'json',
+ 'text',
+ 'javascript',
+ 'jsx',
+ 'html',
+ 'css',
+ 'bash',
+ 'shell',
+ 'python',
+ 'rust',
+ 'go'
+ ]
+ }).then((highlighter) => {
+ if (current !== generation) {
+ highlighter.dispose();
+ return highlighter;
+ }
+ highlighterInstance = highlighter;
+ return highlighter;
+ });
+ }
+
+ return highlighterPromise;
+};
+
+export const disposeChatHighlighter = () => {
+ if (!highlighterPromise && !highlighterInstance) return;
+ generation += 1;
+ highlighterInstance?.dispose();
+ highlighterInstance = null;
+ highlighterPromise = null;
+};
diff --git a/apps/desktop/src/lib/stores/ShikiStore.svelte.ts b/apps/desktop/src/lib/stores/ShikiStore.svelte.ts
new file mode 100644
index 00000000..430e3eb2
--- /dev/null
+++ b/apps/desktop/src/lib/stores/ShikiStore.svelte.ts
@@ -0,0 +1,80 @@
+import { Result } from 'better-result';
+import { createContext, onMount } from 'svelte';
+import { createHighlighterCore, type HighlighterCore } from 'shiki/core';
+import { createJavaScriptRegexEngine } from 'shiki/engine/javascript';
+import bash from '@shikijs/langs/bash';
+import json from '@shikijs/langs/json';
+import toml from '@shikijs/langs/toml';
+import darkPlus from '@shikijs/themes/dark-plus';
+import lightPlus from '@shikijs/themes/light-plus';
+import { WebValidationError } from '@btca/convex/errors';
+
+let highlighterPromise: Promise | null = null;
+let highlighterInstance: HighlighterCore | null = null;
+let generation = 0;
+
+const getCoreHighlighter = () => {
+ if (!highlighterPromise) {
+ generation += 1;
+ const current = generation;
+ highlighterPromise = createHighlighterCore({
+ langs: [bash, json, toml],
+ themes: [darkPlus, lightPlus],
+ engine: createJavaScriptRegexEngine()
+ }).then((highlighter) => {
+ if (current !== generation) {
+ highlighter.dispose();
+ return highlighter;
+ }
+ highlighterInstance = highlighter;
+ return highlighter;
+ });
+ }
+
+ return highlighterPromise;
+};
+
+class ShikiStore {
+ highlighter = $state(null);
+
+ constructor() {
+ onMount(async () => {
+ this.highlighter = await getCoreHighlighter();
+ });
+ }
+}
+
+const [internalGet, internalSet] = createContext();
+
+export const getShikiStore = () => {
+ const missingShikiStoreError = () =>
+ new WebValidationError({
+ message: 'ShikiStore not found, did you call setShikiStore() in a parent component?'
+ });
+
+ const getShikiStoreResult = (): Result => {
+ const store = internalGet();
+ if (!store) return Result.err(missingShikiStoreError());
+ return Result.ok(store);
+ };
+
+ return Result.match(getShikiStoreResult(), {
+ ok: (store) => store,
+ err: (error) => {
+ throw error;
+ }
+ });
+};
+
+export const setShikiStore = () => {
+ const newStore = new ShikiStore();
+ return internalSet(newStore);
+};
+
+export const disposeShikiStoreHighlighter = () => {
+ if (!highlighterPromise && !highlighterInstance) return;
+ generation += 1;
+ highlighterInstance?.dispose();
+ highlighterInstance = null;
+ highlighterPromise = null;
+};
diff --git a/apps/desktop/src/lib/stores/analytics.svelte.ts b/apps/desktop/src/lib/stores/analytics.svelte.ts
new file mode 100644
index 00000000..02dff0b2
--- /dev/null
+++ b/apps/desktop/src/lib/stores/analytics.svelte.ts
@@ -0,0 +1,62 @@
+import posthog from 'posthog-js';
+import { browser } from '$app/environment';
+import { env } from '$env/dynamic/public';
+
+let initialized = false;
+
+export function initAnalytics(): void {
+ if (!browser || initialized) return;
+
+ const posthogId = env.PUBLIC_POSTHOG_ID;
+ if (!posthogId) {
+ console.warn('PUBLIC_POSTHOG_ID is not set');
+ return;
+ }
+
+ const analyticsHost = env.PUBLIC_ANALYTICS_HOST;
+ if (!analyticsHost) {
+ console.warn('PUBLIC_ANALYTICS_HOST is not set');
+ return;
+ }
+
+ posthog.init(posthogId, {
+ api_host: analyticsHost,
+ ui_host: 'https://us.posthog.com',
+ capture_pageview: true,
+ capture_pageleave: true,
+ persistence: 'localStorage'
+ });
+
+ initialized = true;
+}
+
+export function identifyUser(userId: string, properties?: Record): void {
+ if (!browser || !initialized) return;
+ posthog.identify(userId, properties);
+}
+
+export function resetUser(): void {
+ if (!browser || !initialized) return;
+ posthog.reset();
+}
+
+export function trackEvent(event: string, properties?: Record): void {
+ if (!browser || !initialized) return;
+ posthog.capture(event, {
+ ...properties,
+ source: 'client'
+ });
+}
+
+export const ClientAnalyticsEvents = {
+ USER_SIGNED_IN: 'user_signed_in',
+ USER_SIGNED_OUT: 'user_signed_out',
+ CHECKOUT_BUTTON_CLICKED: 'checkout_button_clicked',
+ API_KEY_CREATED: 'api_key_created',
+ API_KEY_REVOKED: 'api_key_revoked',
+ STREAM_CANCELLED: 'stream_cancelled',
+ INSTANCE_WAKE_REQUESTED: 'instance_wake_requested',
+ INSTANCE_STOP_REQUESTED: 'instance_stop_requested',
+ INSTANCE_UPDATE_REQUESTED: 'instance_update_requested',
+ INSTANCE_RESET_REQUESTED: 'instance_reset_requested'
+} as const;
diff --git a/apps/desktop/src/lib/stores/auth.svelte.ts b/apps/desktop/src/lib/stores/auth.svelte.ts
new file mode 100644
index 00000000..f1408eb0
--- /dev/null
+++ b/apps/desktop/src/lib/stores/auth.svelte.ts
@@ -0,0 +1,139 @@
+import type { Clerk } from '@clerk/clerk-js';
+import type { OAuthScope, SessionResource, UserResource } from '@clerk/types';
+import type { Id } from '@btca/convex/dataModel';
+
+// Auth state
+let clerk = $state(null);
+let isLoaded = $state(false);
+let isSignedIn = $state(false);
+let user = $state(null);
+let session = $state(null);
+let instanceId = $state | null>(null);
+
+/**
+ * Set the Clerk instance (alias: setAuthState)
+ */
+export function setClerk(instance: Clerk) {
+ clerk = instance;
+ isLoaded = instance.loaded ?? false;
+ isSignedIn = !!instance.user;
+ user = instance.user ?? null;
+ session = instance.session ?? null;
+
+ // Listen for auth changes
+ instance.addListener(() => {
+ isLoaded = instance.loaded ?? false;
+ isSignedIn = !!instance.user;
+ user = instance.user ?? null;
+ session = instance.session ?? null;
+ });
+}
+
+// Alias for setClerk
+export const setAuthState = setClerk;
+
+/**
+ * Set the instance ID
+ */
+export function setInstanceId(id: Id<'instances'> | null) {
+ instanceId = id;
+}
+
+/**
+ * Get the current auth state
+ */
+export function getAuthState() {
+ return {
+ get clerk() {
+ return clerk;
+ },
+ get isLoaded() {
+ return isLoaded;
+ },
+ get isSignedIn() {
+ return isSignedIn;
+ },
+ get user() {
+ return user;
+ },
+ get session() {
+ return session;
+ },
+ get hasSession() {
+ return !!session;
+ },
+ get instanceId() {
+ return instanceId;
+ },
+ set instanceId(id: Id<'instances'> | null) {
+ instanceId = id;
+ },
+ async getToken(options?: { template?: string }) {
+ if (!clerk?.session) return null;
+ return clerk.session.getToken(options);
+ }
+ } as {
+ readonly clerk: Clerk | null;
+ readonly isLoaded: boolean;
+ readonly isSignedIn: boolean;
+ readonly user: UserResource | null;
+ readonly session: SessionResource | null;
+ readonly hasSession: boolean;
+ instanceId: Id<'instances'> | null;
+ getToken: (options?: { template?: string }) => Promise;
+ };
+}
+
+/**
+ * Sign out the user
+ */
+export async function signOut() {
+ if (clerk) {
+ await clerk.signOut();
+ session = null;
+ instanceId = null;
+ }
+}
+
+/**
+ * Open the sign-in modal
+ */
+export function openSignIn() {
+ if (clerk) {
+ clerk.openSignIn({ redirectUrl: '/app' });
+ }
+}
+
+/**
+ * Open the sign-up modal
+ */
+export function openSignUp(redirectUrl?: string) {
+ if (clerk) {
+ clerk.openSignUp({ redirectUrl });
+ }
+}
+
+/**
+ * Open the user profile modal
+ */
+export function openUserProfile(additionalOAuthScopes?: Partial>) {
+ if (clerk) {
+ clerk.openUserProfile(additionalOAuthScopes ? { additionalOAuthScopes } : undefined);
+ }
+}
+
+export async function reconnectGitHub(scopes: OAuthScope[] = ['repo']) {
+ const githubAccount = user?.externalAccounts.find((account) => account.provider === 'github');
+ if (!githubAccount) return false;
+
+ try {
+ await githubAccount.reauthorize({
+ additionalScopes: scopes,
+ redirectUrl: globalThis.location?.href
+ });
+ return true;
+ } catch (error) {
+ console.error('Failed to reauthorize GitHub account:', error);
+ return false;
+ }
+}
diff --git a/apps/desktop/src/lib/stores/billing.svelte.ts b/apps/desktop/src/lib/stores/billing.svelte.ts
new file mode 100644
index 00000000..349719f9
--- /dev/null
+++ b/apps/desktop/src/lib/stores/billing.svelte.ts
@@ -0,0 +1,132 @@
+import { Result } from 'better-result';
+import { createContext } from 'svelte';
+import { useConvexClient } from 'convex-svelte';
+import { api } from '@btca/convex/api';
+import type { Id } from '@btca/convex/dataModel';
+import type { BillingSummary } from '$lib/billing/types';
+import { WebValidationError } from '@btca/convex/errors';
+
+class BillingStore {
+ private _client = useConvexClient();
+ private _summary = $state(null);
+ private _isLoading = $state(false);
+ private _error = $state(null);
+ private instanceId = $state | null>(null);
+
+ get summary() {
+ return this._summary;
+ }
+
+ get isLoading() {
+ return this._isLoading;
+ }
+
+ get error() {
+ return this._error;
+ }
+
+ get isSubscribed() {
+ return (
+ this._summary?.plan === 'pro' &&
+ (this._summary.status === 'active' || this._summary.status === 'trialing')
+ );
+ }
+
+ get isCanceling() {
+ return this.isSubscribed && !!this._summary?.canceledAt;
+ }
+
+ get canceledAt() {
+ return this._summary?.canceledAt ?? null;
+ }
+
+ get isOnFreePlan() {
+ return this._summary?.plan === 'free';
+ }
+
+ get freeMessagesRemaining() {
+ return this._summary?.freeMessages?.remaining ?? 0;
+ }
+
+ get freeMessagesTotal() {
+ return this._summary?.freeMessages?.total ?? 5;
+ }
+
+ get freeMessagesUsed() {
+ return this._summary?.freeMessages?.used ?? 0;
+ }
+
+ get canChat() {
+ if (this._summary?.plan === 'pro') {
+ return this._summary.status === 'active' || this._summary.status === 'trialing';
+ }
+ if (this._summary?.plan === 'free') {
+ return (this._summary.freeMessages?.remaining ?? 0) > 0;
+ }
+ return false;
+ }
+
+ get isOverLimit() {
+ if (!this._summary) return false;
+ if (this._summary.plan === 'free') {
+ return (this._summary.freeMessages?.remaining ?? 0) <= 0;
+ }
+ return this._summary.usage.aiBudget.isDepleted;
+ }
+
+ get hasSummary() {
+ return !!this._summary;
+ }
+
+ setUserId(instanceId: Id<'instances'> | null) {
+ this.instanceId = instanceId;
+ if (!instanceId) {
+ this._summary = null;
+ this._error = null;
+ this._isLoading = false;
+ return;
+ }
+ void this.fetch();
+ }
+
+ async fetch() {
+ if (!this.instanceId) return;
+ this._isLoading = true;
+ this._error = null;
+ try {
+ this._summary = (await this._client.action(api.usage.getBillingSummary, {
+ instanceId: this.instanceId
+ })) as BillingSummary;
+ } catch (err) {
+ this._error = err instanceof Error ? err.message : 'Failed to load billing';
+ } finally {
+ this._isLoading = false;
+ }
+ }
+
+ refetch = async () => {
+ await this.fetch();
+ };
+}
+
+const [internalGetStore, internalSetStore] = createContext();
+
+export const getBillingStore = () => {
+ const missingBillingStoreError = () =>
+ new WebValidationError({ message: 'Billing store not found. Did you call setBillingStore?' });
+
+ const getBillingStoreResult = (): Result => {
+ const store = internalGetStore();
+ if (!store) return Result.err(missingBillingStoreError());
+ return Result.ok(store);
+ };
+
+ return Result.match(getBillingStoreResult(), {
+ ok: (store) => store,
+ err: (error) => {
+ throw error;
+ }
+ });
+};
+
+export const setBillingStore = () => internalSetStore(new BillingStore());
diff --git a/apps/desktop/src/lib/stores/instance.svelte.ts b/apps/desktop/src/lib/stores/instance.svelte.ts
new file mode 100644
index 00000000..17eadbfc
--- /dev/null
+++ b/apps/desktop/src/lib/stores/instance.svelte.ts
@@ -0,0 +1,251 @@
+import { Result } from 'better-result';
+import { createContext } from 'svelte';
+import { useQuery, useConvexClient } from 'convex-svelte';
+import { instances } from '@btca/convex/apiHelpers';
+import type { Doc, Id } from '@btca/convex/dataModel';
+import { trackEvent, ClientAnalyticsEvents } from './analytics.svelte';
+import { WebValidationError } from '@btca/convex/errors';
+
+type InstanceStatus = {
+ instance: Doc<'instances'>;
+ cachedResources: Doc<'cachedResources'>[];
+ expectedSnapshotName: string;
+ migrationNeeded: boolean;
+} | null;
+
+type InstanceActionResponse = {
+ success?: boolean;
+ serverUrl?: string;
+ stopped?: boolean;
+ updated?: boolean;
+ applied?: boolean;
+ appliesOnWake?: boolean;
+ error?: string;
+};
+
+type EnsureInstanceResult = {
+ instanceId: Id<'instances'>;
+ status: 'created' | 'exists' | 'provisioning';
+};
+
+class InstanceStore {
+ private _query = useQuery(instances.queries.getStatus, {});
+ private _client = useConvexClient();
+ private _error = $state(null);
+ private _isBootstrapping = $state(false);
+ private _hasBootstrapped = $state(false);
+ private _ensureStatus = $state(null);
+
+ get status(): InstanceStatus {
+ return this._query.data ?? null;
+ }
+
+ get instance() {
+ return this.status?.instance ?? null;
+ }
+
+ get cachedResources() {
+ return this.status?.cachedResources ?? [];
+ }
+
+ get expectedSnapshotName() {
+ return this.status?.expectedSnapshotName ?? null;
+ }
+
+ get migrationNeeded() {
+ return this.status?.migrationNeeded ?? false;
+ }
+
+ get state() {
+ return this.status?.instance.state ?? null;
+ }
+
+ get errorKind() {
+ return this.status?.instance.errorKind ?? null;
+ }
+
+ get btcaVersion() {
+ return this.status?.instance.btcaVersion ?? null;
+ }
+
+ get opencodeVersion() {
+ return this.status?.instance.opencodeVersion ?? null;
+ }
+
+ get latestBtcaVersion() {
+ return this.status?.instance.latestBtcaVersion ?? null;
+ }
+
+ get latestOpencodeVersion() {
+ return this.status?.instance.latestOpencodeVersion ?? null;
+ }
+
+ get btcaUpdateAvailable() {
+ const current = this.btcaVersion;
+ const latest = this.latestBtcaVersion;
+ return Boolean(current && latest && current !== latest);
+ }
+
+ get opencodeUpdateAvailable() {
+ const current = this.opencodeVersion;
+ const latest = this.latestOpencodeVersion;
+ return Boolean(current && latest && current !== latest);
+ }
+
+ get updateAvailable() {
+ return this.btcaUpdateAvailable;
+ }
+
+ get storageUsedBytes() {
+ return this.status?.instance.storageUsedBytes ?? null;
+ }
+
+ get error() {
+ return this._error ?? this._query.error?.message ?? null;
+ }
+
+ get isLoading() {
+ return this._query.isLoading;
+ }
+
+ get isBootstrapping() {
+ return this._isBootstrapping;
+ }
+
+ get hasBootstrapped() {
+ return this._hasBootstrapped;
+ }
+
+ get ensureStatus() {
+ return this._ensureStatus;
+ }
+
+ get needsBootstrap() {
+ if (this._query.isLoading || this._hasBootstrapped) {
+ return false;
+ }
+
+ if (!this.instance) {
+ return true;
+ }
+
+ return this.migrationNeeded || this.state === 'unprovisioned' || this.state === 'provisioning';
+ }
+
+ async ensureExists(): Promise {
+ if (this._isBootstrapping || this._hasBootstrapped) {
+ return null;
+ }
+
+ this._isBootstrapping = true;
+ this._error = null;
+
+ try {
+ const result = await this._client.action(instances.actions.ensureInstanceExists, {});
+ this._ensureStatus = (result as EnsureInstanceResult).status;
+ this._hasBootstrapped = true;
+ return result as EnsureInstanceResult;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Failed to create instance';
+ return null;
+ } finally {
+ this._isBootstrapping = false;
+ }
+ }
+
+ async wake(projectId?: Id<'projects'>): Promise {
+ this._error = null;
+ trackEvent(ClientAnalyticsEvents.INSTANCE_WAKE_REQUESTED, {
+ instanceId: this.instance?._id,
+ projectId
+ });
+ try {
+ const result = await this._client.action(instances.actions.wakeMyInstance, { projectId });
+ return result as InstanceActionResponse;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Instance wake failed';
+ return { error: this._error };
+ }
+ }
+
+ async applyProjectRuntimeConfig(projectId: Id<'projects'>): Promise {
+ this._error = null;
+ try {
+ const result = await this._client.action(instances.actions.applyProjectRuntimeConfig, {
+ projectId
+ });
+ return result as InstanceActionResponse;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Failed to apply project config';
+ return { error: this._error };
+ }
+ }
+
+ async stop(): Promise {
+ this._error = null;
+ trackEvent(ClientAnalyticsEvents.INSTANCE_STOP_REQUESTED, {
+ instanceId: this.instance?._id
+ });
+ try {
+ const result = await this._client.action(instances.actions.stopMyInstance, {});
+ return result as InstanceActionResponse;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Instance stop failed';
+ return { error: this._error };
+ }
+ }
+
+ async update(): Promise {
+ this._error = null;
+ trackEvent(ClientAnalyticsEvents.INSTANCE_UPDATE_REQUESTED, {
+ instanceId: this.instance?._id
+ });
+ try {
+ const result = await this._client.action(instances.actions.updateMyInstance, {});
+ return result as InstanceActionResponse;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Instance update failed';
+ return { error: this._error };
+ }
+ }
+
+ async reset(): Promise {
+ this._error = null;
+ trackEvent(ClientAnalyticsEvents.INSTANCE_RESET_REQUESTED, {
+ instanceId: this.instance?._id
+ });
+ try {
+ const result = await this._client.action(instances.actions.resetMyInstance, {});
+ return result as InstanceActionResponse;
+ } catch (error) {
+ this._error =
+ error instanceof Error ? error.message : 'Instance reset failed. Please contact support.';
+ return { error: this._error };
+ }
+ }
+}
+
+const [internalGetStore, internalSetStore] = createContext();
+
+const missingInstanceStoreError = () =>
+ new WebValidationError({ message: 'Instance store not found. Did you call setInstanceStore?' });
+
+const getInstanceStoreResult = (): Result => {
+ const store = internalGetStore();
+ if (!store) return Result.err(missingInstanceStoreError());
+ return Result.ok(store);
+};
+
+export const getInstanceStore = () =>
+ Result.match(getInstanceStoreResult(), {
+ ok: (store) => store,
+ err: (error) => {
+ throw error;
+ }
+ });
+
+export const setInstanceStore = () => {
+ const store = new InstanceStore();
+ internalSetStore(store);
+ return store;
+};
diff --git a/apps/desktop/src/lib/stores/project.svelte.ts b/apps/desktop/src/lib/stores/project.svelte.ts
new file mode 100644
index 00000000..cee3b0f0
--- /dev/null
+++ b/apps/desktop/src/lib/stores/project.svelte.ts
@@ -0,0 +1,170 @@
+import { Result } from 'better-result';
+import { getContext, setContext } from 'svelte';
+import { useQuery, useConvexClient } from 'convex-svelte';
+import { goto } from '$app/navigation';
+import { api } from '@btca/convex/api';
+import type { Id } from '@btca/convex/dataModel';
+import { WebValidationError } from '@btca/convex/errors';
+
+type Project = {
+ _id: Id<'projects'>;
+ name: string;
+ model?: string;
+ isDefault: boolean;
+ createdAt: number;
+};
+
+const PROJECT_STORE_KEY = Symbol('project-store');
+
+export class ProjectStore {
+ private _query = useQuery(api.projects.list, {});
+ private _client = useConvexClient();
+ private _selectedProjectId = $state | null>(null);
+ private _error = $state(null);
+ private _initialized = $state(false);
+ private _showCreateModal = $state(false);
+
+ get projects(): Project[] {
+ return (this._query.data as Project[] | undefined) ?? [];
+ }
+
+ get isLoading() {
+ return this._query.isLoading;
+ }
+
+ get error() {
+ return this._error ?? this._query.error?.message ?? null;
+ }
+
+ get showCreateModal() {
+ return this._showCreateModal;
+ }
+
+ set showCreateModal(value: boolean) {
+ this._showCreateModal = value;
+ }
+
+ get selectedProjectId() {
+ return this._selectedProjectId;
+ }
+
+ get selectedProject(): Project | null {
+ if (!this._selectedProjectId) {
+ return this.projects.find((p) => p.isDefault) ?? this.projects[0] ?? null;
+ }
+ return this.projects.find((p) => p._id === this._selectedProjectId) ?? null;
+ }
+
+ get defaultProject(): Project | null {
+ return this.projects.find((p) => p.isDefault) ?? null;
+ }
+
+ get initialized() {
+ return this._initialized;
+ }
+
+ selectProject(projectId: Id<'projects'> | null, updateUrl = true) {
+ this._selectedProjectId = projectId;
+ if (updateUrl && typeof window !== 'undefined') {
+ this.updateUrlParam(projectId);
+ }
+ }
+
+ private updateUrlParam(projectId: Id<'projects'> | null) {
+ const url = new URL(window.location.href);
+ if (projectId) {
+ url.searchParams.set('project', projectId);
+ } else {
+ url.searchParams.delete('project');
+ }
+ window.history.replaceState({}, '', url.toString());
+ }
+
+ initFromUrl(urlProjectId: string | null) {
+ if (this._initialized) return;
+
+ if (urlProjectId && this.projects.some((p) => p._id === urlProjectId)) {
+ this._selectedProjectId = urlProjectId as Id<'projects'>;
+ }
+ this._initialized = true;
+ }
+
+ async selectProjectWithNavigation(projectId: Id<'projects'> | null) {
+ this._selectedProjectId = projectId;
+ if (typeof window !== 'undefined') {
+ const url = new URL(window.location.href);
+ if (projectId) {
+ url.searchParams.set('project', projectId);
+ } else {
+ url.searchParams.delete('project');
+ }
+ await goto(url.toString(), { replaceState: true, keepFocus: true });
+ }
+ }
+
+ async createProject(name: string, model?: string): Promise | null> {
+ this._error = null;
+ try {
+ const projectId = await this._client.mutation(api.projects.create, {
+ name,
+ model
+ });
+ return projectId;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Failed to create project';
+ return null;
+ }
+ }
+
+ async updateProjectModel(projectId: Id<'projects'>, model?: string): Promise {
+ this._error = null;
+ try {
+ await this._client.mutation(api.projects.updateModel, {
+ projectId,
+ model
+ });
+ return true;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Failed to update project';
+ return false;
+ }
+ }
+
+ async deleteProject(projectId: Id<'projects'>): Promise {
+ this._error = null;
+ try {
+ await this._client.mutation(api.projects.remove, { projectId });
+ if (this._selectedProjectId === projectId) {
+ this.selectProject(null);
+ }
+ return true;
+ } catch (error) {
+ this._error = error instanceof Error ? error.message : 'Failed to delete project';
+ return false;
+ }
+ }
+}
+
+const missingProjectStoreError = () =>
+ new WebValidationError({ message: 'Project store not found. Did you call setProjectStore?' });
+
+const getProjectStoreResult = (): Result => {
+ const store = getContext(PROJECT_STORE_KEY);
+ if (!store) return Result.err(missingProjectStoreError());
+ return Result.ok(store);
+};
+
+export const getProjectStore = (): ProjectStore => {
+ return Result.match(getProjectStoreResult(), {
+ ok: (store) => store,
+ err: (error) => {
+ throw error;
+ }
+ });
+};
+
+export const setProjectStore = (): ProjectStore => {
+ const store = new ProjectStore();
+ setContext(PROJECT_STORE_KEY, store);
+ return store;
+};
diff --git a/apps/desktop/src/lib/stores/theme.svelte.ts b/apps/desktop/src/lib/stores/theme.svelte.ts
new file mode 100644
index 00000000..4ecc33fa
--- /dev/null
+++ b/apps/desktop/src/lib/stores/theme.svelte.ts
@@ -0,0 +1,70 @@
+import { Result } from 'better-result';
+import { getContext, setContext } from 'svelte';
+import { browser } from '$app/environment';
+import { WebValidationError } from '@btca/convex/errors';
+
+const THEME_KEY = Symbol('theme');
+
+export type Theme = 'light' | 'dark';
+
+class ThemeStore {
+ private _theme = $state('dark');
+
+ constructor() {
+ if (browser) {
+ const stored = localStorage.getItem('theme') as Theme | null;
+ const prefersDark = window.matchMedia?.('(prefers-color-scheme: dark)').matches;
+ const initialTheme =
+ stored === 'light' || stored === 'dark' ? stored : prefersDark ? 'dark' : 'light';
+ this.applyTheme(initialTheme, false);
+ }
+ }
+
+ get theme() {
+ return this._theme;
+ }
+
+ toggle() {
+ this.applyTheme(this._theme === 'dark' ? 'light' : 'dark');
+ }
+
+ set(newTheme: Theme) {
+ this.applyTheme(newTheme);
+ }
+
+ private applyTheme(newTheme: Theme, persist = true) {
+ this._theme = newTheme;
+ if (browser) {
+ document.documentElement.classList.toggle('dark', newTheme === 'dark');
+ if (persist) {
+ localStorage.setItem('theme', newTheme);
+ }
+ }
+ }
+}
+
+export const setThemeStore = () => {
+ const store = new ThemeStore();
+ setContext(THEME_KEY, store);
+ return store;
+};
+
+export const getThemeStore = (): ThemeStore => {
+ const missingThemeStoreError = () =>
+ new WebValidationError({
+ message: 'Theme store not found. Did you call setThemeStore() in a parent component?'
+ });
+
+ const getThemeStoreResult = (): Result => {
+ const store = getContext(THEME_KEY);
+ if (!store) return Result.err(missingThemeStoreError());
+ return Result.ok(store);
+ };
+
+ return Result.match(getThemeStoreResult(), {
+ ok: (store) => store,
+ err: (error) => {
+ throw error;
+ }
+ });
+};
diff --git a/apps/desktop/src/lib/stores/threadPreload.svelte.ts b/apps/desktop/src/lib/stores/threadPreload.svelte.ts
new file mode 100644
index 00000000..58baea7d
--- /dev/null
+++ b/apps/desktop/src/lib/stores/threadPreload.svelte.ts
@@ -0,0 +1,54 @@
+import type { Doc, Id } from '@btca/convex/dataModel';
+
+export type PreloadedThreadWithMessages =
+ | ((Doc<'threads'> & {
+ messages: Array>;
+ resources: string[];
+ threadResources: string[];
+ activeStream: {
+ sessionId: string;
+ messageId: Id<'messages'>;
+ startedAt: number;
+ } | null;
+ }) & { projectId?: Id<'projects'> | undefined })
+ | null;
+
+type CacheEntry = {
+ data: PreloadedThreadWithMessages;
+ loadedAt: number;
+};
+
+const CACHE_TTL_MS = 2 * 60 * 1000;
+const preloadCache = new Map();
+
+const normalizeThreadId = (threadId: string) => threadId;
+
+const isFresh = (entry: CacheEntry) => Date.now() - entry.loadedAt < CACHE_TTL_MS;
+const getEntry = (threadId: string) => preloadCache.get(normalizeThreadId(threadId));
+const getFreshData = (threadId: string) => {
+ const entry = getEntry(threadId);
+ if (!entry) return null;
+ if (!isFresh(entry)) {
+ preloadCache.delete(normalizeThreadId(threadId));
+ return null;
+ }
+ return entry.data;
+};
+
+export const threadPreloadStore = {
+ get(threadId: string) {
+ return getFreshData(threadId);
+ },
+ set(threadId: string, data: PreloadedThreadWithMessages) {
+ preloadCache.set(normalizeThreadId(threadId), {
+ data,
+ loadedAt: Date.now()
+ });
+ },
+ has(threadId: string) {
+ return getFreshData(threadId) !== null;
+ },
+ markConsumed(threadId: string) {
+ preloadCache.delete(normalizeThreadId(threadId));
+ }
+};
diff --git a/apps/desktop/src/lib/svgs/ClerkSvg.svelte b/apps/desktop/src/lib/svgs/ClerkSvg.svelte
new file mode 100644
index 00000000..e1f403cb
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/ClerkSvg.svelte
@@ -0,0 +1,9 @@
+
diff --git a/apps/desktop/src/lib/svgs/ConvexSvg.svelte b/apps/desktop/src/lib/svgs/ConvexSvg.svelte
new file mode 100644
index 00000000..4b3a8977
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/ConvexSvg.svelte
@@ -0,0 +1,12 @@
+
diff --git a/apps/desktop/src/lib/svgs/DaytonaSvg.svelte b/apps/desktop/src/lib/svgs/DaytonaSvg.svelte
new file mode 100644
index 00000000..5d1803d6
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/DaytonaSvg.svelte
@@ -0,0 +1,14 @@
+
+
+
+
+
diff --git a/apps/desktop/src/lib/svgs/HonoSvg.svelte b/apps/desktop/src/lib/svgs/HonoSvg.svelte
new file mode 100644
index 00000000..fc82694c
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/HonoSvg.svelte
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
+
+
diff --git a/apps/desktop/src/lib/svgs/OpencodeSvg.svelte b/apps/desktop/src/lib/svgs/OpencodeSvg.svelte
new file mode 100644
index 00000000..537dcf91
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/OpencodeSvg.svelte
@@ -0,0 +1,10 @@
+
diff --git a/apps/desktop/src/lib/svgs/SolidSvg.svelte b/apps/desktop/src/lib/svgs/SolidSvg.svelte
new file mode 100644
index 00000000..7cbc857f
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/SolidSvg.svelte
@@ -0,0 +1,50 @@
+
diff --git a/apps/desktop/src/lib/svgs/SvelteSvg.svelte b/apps/desktop/src/lib/svgs/SvelteSvg.svelte
new file mode 100644
index 00000000..64dcc1ab
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/SvelteSvg.svelte
@@ -0,0 +1,14 @@
+
diff --git a/apps/desktop/src/lib/svgs/TailwindSvg.svelte b/apps/desktop/src/lib/svgs/TailwindSvg.svelte
new file mode 100644
index 00000000..1f376379
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/TailwindSvg.svelte
@@ -0,0 +1,11 @@
+
diff --git a/apps/desktop/src/lib/svgs/TurboSvg.svelte b/apps/desktop/src/lib/svgs/TurboSvg.svelte
new file mode 100644
index 00000000..e8d46a2c
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/TurboSvg.svelte
@@ -0,0 +1,21 @@
+
diff --git a/apps/desktop/src/lib/svgs/ViteSvg.svelte b/apps/desktop/src/lib/svgs/ViteSvg.svelte
new file mode 100644
index 00000000..bb96fc10
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/ViteSvg.svelte
@@ -0,0 +1,369 @@
+
diff --git a/apps/desktop/src/lib/svgs/ZodSvg.svelte b/apps/desktop/src/lib/svgs/ZodSvg.svelte
new file mode 100644
index 00000000..1467b0e8
--- /dev/null
+++ b/apps/desktop/src/lib/svgs/ZodSvg.svelte
@@ -0,0 +1,39 @@
+
diff --git a/apps/desktop/src/lib/types/index.ts b/apps/desktop/src/lib/types/index.ts
new file mode 100644
index 00000000..08e32b9e
--- /dev/null
+++ b/apps/desktop/src/lib/types/index.ts
@@ -0,0 +1,223 @@
+import { z } from 'zod';
+
+// Re-export shared types
+export type {
+ TextChunk,
+ ReasoningChunk,
+ ToolChunk,
+ FileChunk,
+ BtcaChunk,
+ AssistantContent,
+ ThreadMessage,
+ CancelState
+} from '@btca/shared';
+
+export { formatConversationHistory, extractMessageText } from '@btca/shared';
+
+import type { BtcaChunk, AssistantContent } from '@btca/shared';
+
+// Resource types
+export interface Resource {
+ name: string;
+ type: 'git' | 'local' | 'npm';
+ url?: string;
+ branch?: string;
+ path?: string;
+ package?: string;
+ version?: string;
+ searchPath?: string;
+ specialNotes?: string;
+}
+
+export interface MessageStats {
+ durationMs?: number;
+ inputTokens?: number;
+ outputTokens?: number;
+ cachedTokens?: number;
+ totalTokens?: number;
+ tokensPerSecond?: number;
+ totalPriceUsd?: number;
+}
+
+// Message types (using AssistantContent from shared)
+export type Message =
+ | {
+ id: string;
+ role: 'user';
+ content: string;
+ resources: string[];
+ }
+ | {
+ id: string;
+ role: 'assistant';
+ content: AssistantContent;
+ canceled?: boolean;
+ stats?: MessageStats;
+ }
+ | {
+ id: string;
+ role: 'system';
+ content: string;
+ };
+
+// Session types
+export interface ChatSession {
+ id: string;
+ sandboxId: string;
+ serverUrl: string;
+ messages: Message[];
+ threadResources: string[];
+ createdAt: Date;
+ lastActivityAt: Date;
+ status: 'pending' | 'creating' | 'cloning' | 'starting' | 'active' | 'error' | 'destroyed';
+ error?: string;
+}
+
+// Stream event types (from btca server - matching apps/server/src/stream/types.ts)
+const BtcaModelSchema = z.object({
+ provider: z.string(),
+ model: z.string()
+});
+
+const BtcaCollectionInfoSchema = z.object({
+ key: z.string(),
+ path: z.string()
+});
+
+const BtcaToolStateSchema = z.object({
+ status: z.enum(['pending', 'running', 'completed', 'error']),
+ input: z.record(z.string(), z.unknown()).optional(),
+ output: z.string().optional(),
+ title: z.string().optional(),
+ metadata: z.record(z.string(), z.unknown()).optional(),
+ error: z.string().optional(),
+ raw: z.string().optional(),
+ time: z
+ .object({
+ start: z.number(),
+ end: z.number().optional(),
+ compacted: z.number().optional()
+ })
+ .optional()
+});
+
+export const BtcaStreamEventSchema = z.discriminatedUnion('type', [
+ z.object({
+ type: z.literal('meta'),
+ model: BtcaModelSchema,
+ resources: z.array(z.string()),
+ collection: BtcaCollectionInfoSchema
+ }),
+ z.object({
+ type: z.literal('text.delta'),
+ delta: z.string()
+ }),
+ z.object({
+ type: z.literal('reasoning.delta'),
+ delta: z.string()
+ }),
+ z.object({
+ type: z.literal('tool.updated'),
+ callID: z.string(),
+ tool: z.string(),
+ state: BtcaToolStateSchema
+ }),
+ z.object({
+ type: z.literal('done'),
+ text: z.string(),
+ reasoning: z.string(),
+ tools: z.array(
+ z.object({
+ callID: z.string(),
+ tool: z.string(),
+ state: BtcaToolStateSchema
+ })
+ ),
+ usage: z
+ .object({
+ inputTokens: z.number().optional(),
+ outputTokens: z.number().optional(),
+ reasoningTokens: z.number().optional(),
+ totalTokens: z.number().optional()
+ })
+ .optional(),
+ metrics: z
+ .object({
+ timing: z
+ .object({
+ totalMs: z.number().optional(),
+ genMs: z.number().optional()
+ })
+ .optional(),
+ throughput: z
+ .object({
+ outputTokensPerSecond: z.number().optional(),
+ totalTokensPerSecond: z.number().optional()
+ })
+ .optional(),
+ pricing: z
+ .object({
+ source: z.literal('models.dev'),
+ modelKey: z.string().optional(),
+ ratesUsdPerMTokens: z
+ .object({
+ input: z.number().optional(),
+ output: z.number().optional(),
+ reasoning: z.number().optional(),
+ cacheRead: z.number().optional(),
+ cacheWrite: z.number().optional()
+ })
+ .optional(),
+ costUsd: z
+ .object({
+ input: z.number().optional(),
+ output: z.number().optional(),
+ reasoning: z.number().optional(),
+ total: z.number().optional()
+ })
+ .optional()
+ })
+ .optional()
+ })
+ .optional()
+ }),
+ z.object({
+ type: z.literal('error'),
+ tag: z.string(),
+ message: z.string()
+ })
+]);
+
+export type BtcaStreamEvent = z.infer;
+
+// Command palette types
+export type CommandMode = 'add-resource' | 'switch-session' | 'clear' | 'destroy-sandbox';
+
+export interface Command {
+ name: string;
+ description: string;
+ mode: CommandMode;
+ shortcut?: string;
+}
+
+export const COMMANDS: Command[] = [
+ {
+ name: 'Clear Chat',
+ description: 'Clear the current chat messages',
+ mode: 'clear',
+ shortcut: 'Ctrl+L'
+ },
+ {
+ name: 'Switch Session',
+ description: 'Switch to a different chat session',
+ mode: 'switch-session',
+ shortcut: 'Ctrl+K'
+ },
+ {
+ name: 'Destroy Sandbox',
+ description: 'Destroy the current sandbox and end the session',
+ mode: 'destroy-sandbox'
+ }
+];
+
+// CancelState is re-exported from @btca/shared above
diff --git a/apps/desktop/src/lib/utils/stream.ts b/apps/desktop/src/lib/utils/stream.ts
new file mode 100644
index 00000000..40b4ee23
--- /dev/null
+++ b/apps/desktop/src/lib/utils/stream.ts
@@ -0,0 +1,68 @@
+import { BtcaStreamEventSchema, type BtcaStreamEvent } from '../types/index.ts';
+
+/**
+ * Parse a Server-Sent Events stream from a Response
+ */
+export async function* parseSSEStream(response: Response): AsyncGenerator {
+ if (!response.body) {
+ return;
+ }
+
+ const reader = response.body.getReader();
+ const decoder = new TextDecoder();
+ let buffer = '';
+
+ try {
+ while (true) {
+ const { done, value } = await reader.read();
+ if (done) break;
+
+ buffer += decoder.decode(value, { stream: true });
+
+ // Process complete events from buffer
+ const lines = buffer.split('\n');
+ buffer = lines.pop() ?? '';
+
+ let eventData = '';
+
+ for (const line of lines) {
+ if (line.startsWith('data: ')) {
+ eventData = line.slice(6);
+ } else if (line === '' && eventData) {
+ try {
+ const parsed = JSON.parse(eventData) as unknown;
+ const validated = BtcaStreamEventSchema.parse(parsed);
+ yield validated;
+ } catch (error) {
+ console.error('Failed to parse SSE event:', error);
+ }
+ eventData = '';
+ }
+ }
+ }
+
+ // Process any remaining data
+ if (buffer.trim()) {
+ const lines = buffer.split('\n');
+ let eventData = '';
+
+ for (const line of lines) {
+ if (line.startsWith('data: ')) {
+ eventData = line.slice(6);
+ }
+ }
+
+ if (eventData) {
+ try {
+ const parsed = JSON.parse(eventData) as unknown;
+ const validated = BtcaStreamEventSchema.parse(parsed);
+ yield validated;
+ } catch {
+ // Ignore incomplete final event
+ }
+ }
+ }
+ } finally {
+ reader.releaseLock();
+ }
+}
diff --git a/apps/desktop/src/routes/+layout.svelte b/apps/desktop/src/routes/+layout.svelte
new file mode 100644
index 00000000..182416d9
--- /dev/null
+++ b/apps/desktop/src/routes/+layout.svelte
@@ -0,0 +1,48 @@
+
+
+
+
+ btca Desktop
+
+
+
+
+
+
+
+
+{@render children()}
diff --git a/apps/desktop/src/routes/+layout.ts b/apps/desktop/src/routes/+layout.ts
new file mode 100644
index 00000000..a3d15781
--- /dev/null
+++ b/apps/desktop/src/routes/+layout.ts
@@ -0,0 +1 @@
+export const ssr = false;
diff --git a/apps/desktop/src/routes/app/+layout.svelte b/apps/desktop/src/routes/app/+layout.svelte
new file mode 100644
index 00000000..abf29f34
--- /dev/null
+++ b/apps/desktop/src/routes/app/+layout.svelte
@@ -0,0 +1,212 @@
+
+
+
+
+
+ btca | App
+
+
+
+
+
+
+ {#if auth.isSignedIn}
+
(sidebarOpen = true)}
+ aria-label="Open sidebar"
+ >
+
+
+
+
+
+ {#if sidebarOpen}
+
(sidebarOpen = false)}
+ aria-label="Close sidebar"
+ >
+ {/if}
+ {/if}
+
+
+ {#if isInitializing}
+
+
+
+ {:else}
+ {@render children()}
+ {/if}
+
+
+
+
+
+ (commandPaletteOpen = false)}
+ onOpenAddResource={() => (addResourceModalOpen = true)}
+/>
+ (addResourceModalOpen = false)} />
diff --git a/apps/desktop/src/routes/app/+layout.ts b/apps/desktop/src/routes/app/+layout.ts
new file mode 100644
index 00000000..fc69c27b
--- /dev/null
+++ b/apps/desktop/src/routes/app/+layout.ts
@@ -0,0 +1,3 @@
+// SPA mode - disable SSR and prerendering for the app directory
+export const ssr = false;
+export const prerender = false;
diff --git a/apps/desktop/src/routes/app/+page.svelte b/apps/desktop/src/routes/app/+page.svelte
new file mode 100644
index 00000000..56a6ce4a
--- /dev/null
+++ b/apps/desktop/src/routes/app/+page.svelte
@@ -0,0 +1,45 @@
+
+
+{#if !auth.isSignedIn}
+
+
+
+
+
+
+
Welcome to btca Chat
+
+ Ask questions about your favorite frameworks and libraries. Sign in to get started.
+
+
+
auth.clerk?.openSignIn()}>
+ Sign in to get started
+
+
+{:else}
+
+
+
+
+
+
Pick up where you left off
+
+ Select a thread from the sidebar or start a new one to begin chatting.
+
+
+
+
+ New Thread
+
+
+{/if}
diff --git a/apps/desktop/src/routes/app/chat/[id]/+page.svelte b/apps/desktop/src/routes/app/chat/[id]/+page.svelte
new file mode 100644
index 00000000..135f38fc
--- /dev/null
+++ b/apps/desktop/src/routes/app/chat/[id]/+page.svelte
@@ -0,0 +1,988 @@
+
+
+
+ {#if !isNewThread && showLoadingSpinner}
+
+
+
+ {:else if !isNewThread && !thread}
+
+ {:else}
+
+
+
+
+
+ {/if}
+
diff --git a/apps/desktop/src/routes/app/checkout/cancel/+page.svelte b/apps/desktop/src/routes/app/checkout/cancel/+page.svelte
new file mode 100644
index 00000000..c0017913
--- /dev/null
+++ b/apps/desktop/src/routes/app/checkout/cancel/+page.svelte
@@ -0,0 +1,19 @@
+
+
+
+
+
+
+
+
Checkout canceled
+
+ No worries - your card wasn't charged. You can restart checkout whenever you're ready.
+
+
+
+
diff --git a/apps/desktop/src/routes/app/checkout/success/+page.svelte b/apps/desktop/src/routes/app/checkout/success/+page.svelte
new file mode 100644
index 00000000..0f466948
--- /dev/null
+++ b/apps/desktop/src/routes/app/checkout/success/+page.svelte
@@ -0,0 +1,25 @@
+
+
+
+
+
+
+
+
You're subscribed
+
Checkout complete. Your plan is now active and ready to use.
+
+
+
diff --git a/apps/desktop/src/routes/app/settings/+page.svelte b/apps/desktop/src/routes/app/settings/+page.svelte
new file mode 100644
index 00000000..803a67ba
--- /dev/null
+++ b/apps/desktop/src/routes/app/settings/+page.svelte
@@ -0,0 +1,1007 @@
+
+
+
+
+
+
+
Settings
+
Manage your account and preferences.
+
+
+
+
+ {#each tabs as tab}
+ setActiveTab(tab.id)}
+ >
+
+ {tab.label}
+
+ {/each}
+
+
+
+
+
+ {#if activeTab === 'account'}
+
+
+
+ {#if auth.user?.imageUrl}
+
+ {:else}
+
+
+
+ {/if}
+
+
{auth.user?.fullName ?? 'User'}
+
+ {auth.user?.primaryEmailAddress?.emailAddress ?? ''}
+
+ {#if billingStore.isSubscribed}
+
+ Pro Plan
+
+ {:else if billingStore.isOnFreePlan}
+
+ Free Plan
+
+ {/if}
+
+
openUserProfile()}>
+ Edit Profile
+
+
+
+
+ {#if billingStore.isOnFreePlan}
+
+
+
+
Free Plan
+
+ {billingStore.freeMessagesRemaining} / {billingStore.freeMessagesTotal} free messages
+ remaining
+
+
+
setActiveTab('billing')}
+ class="bc-btn bc-btn-primary text-sm">Upgrade to Pro
+
+
+ {:else if billingStore.isSubscribed && usage}
+
+
+
+
+
+
Monthly AI usage
+
{formatPercent(maxUsedPct)} used
+
+
+
+
+
Higher-end models use this faster.
+
+ {/if}
+
+ {#if selectedProject}
+
+
+
+
Model
+
+ Higher-end models use your monthly AI usage faster.
+
+
+
+
+
+ {/if}
+
+ {:else if activeTab === 'mcp'}
+
+ {#if billingStore.isOnFreePlan}
+
+
+
+
Free Plan
+
+ {billingStore.freeMessagesRemaining} / {billingStore.freeMessagesTotal} free messages
+ remaining
+
+
+ {#if billingStore.freeMessagesRemaining === 0}
+
setActiveTab('billing')}
+ class="bc-btn bc-btn-primary text-xs">Upgrade to Pro
+ {:else}
+
setActiveTab('billing')}
+ class="bc-btn bc-btn-primary text-xs">Upgrade
+ {/if}
+
+
+ {/if}
+
+
+
+
+
API Keys
+
Manage your API keys for MCP access.
+
+
(showCreateModal = true)}
+ >
+
+ Create Key
+
+
+
+ {#if isLoadingKeys}
+
+
+
+ {:else if clerkApiKeys.length === 0}
+
+
+
No API keys yet
+
Create one to connect your MCP clients.
+
+ {:else}
+
+
+
+
+ Name
+ ID
+
+ Created
+
+
+ Last Used
+
+
+ Usage
+
+
+
+
+
+ {#each clerkApiKeys as key}
+ {@const usage = usageByKeyId.get(key.id)}
+
+ {key.name ?? 'Unnamed key'}
+ {key.id}
+ {formatDate(key.createdAt)}
+
+ {usage?.lastUsedAt ? formatDate(usage.lastUsedAt) : 'Never'}
+
+ {usage?.usageCount ?? 0}
+
+ handleRevokeKey(key.id)}
+ title="Revoke key"
+ >
+
+
+
+
+ {/each}
+
+
+
+
+ The ID shown is for reference only — it's not your actual API key.
+
+ {/if}
+
+
+
+
+
Setup Guide
+
Configure your MCP client to use Better Context.
+
+
+
+
+
1. Add to your MCP configuration
+
+
+ {#each Object.entries(toolConfigs) as [key, tool]}
+ (selectedTool = key as McpTool)}
+ >
+ {tool.name}
+
+ {/each}
+
+
+
+
+ {#if currentConfig.kind === 'command'}
+ Run this command in your terminal:
+ {:else}
+ Add this to {currentConfig.filename}:
+ {/if}
+
+
+ Docs
+
+
+
+
+
+
+ {#if shikiStore.highlighter}
+ {@html shikiStore.highlighter.codeToHtml(currentConfig.config, {
+ theme: shikiTheme,
+ lang: currentConfig.language,
+ rootStyle: 'background-color: transparent; padding: 0; margin: 0;'
+ })}
+ {:else}
+
{currentConfig.config}
+ {/if}
+
+
+
+
+ {#if currentConfig.hasKeyInConfig}
+
+ Replace YOUR_API_KEY with an API key from above.
+
+ {/if}
+ {#if currentConfig.envVarSnippet}
+
+
Set environment variable:
+
+
+
+ {#if shikiStore.highlighter}
+ {@html shikiStore.highlighter.codeToHtml(currentConfig.envVarSnippet, {
+ theme: shikiTheme,
+ lang: currentConfig.envVarLanguage ?? 'bash',
+ rootStyle: 'background-color: transparent; padding: 0; margin: 0;'
+ })}
+ {:else}
+
{currentConfig.envVarSnippet}
+ {/if}
+
+
+
+
+
+ Add this to .zshenv or your shell profile.
+
+ {#if currentConfig.hasKeyInEnv}
+
+ Replace YOUR_API_KEY with an API key from above.
+
+ {/if}
+
+ {/if}
+
+
+
+
2. MCP Audit (optional)
+
+ Add this to your AGENTS.md or system prompt:
+
+
+
+
+
+
+ {:else if activeTab === 'usage'}
+
+ {#if billingStore.isLoading}
+
+
+
+ {:else if !billingStore.isSubscribed}
+
+ {:else}
+
+
+
+
+
+
+
+
Monthly AI usage
+
Higher-end models use this faster.
+
+
+
+ {formatPercent(maxUsedPct)} used
+
+
+
+
+
+ {#if billingStore.isOverLimit}
+
+ You've hit your monthly usage limits. Contact
+
{SUPPORT_URL} to raise them.
+
+ {/if}
+ {/if}
+
+ {:else if activeTab === 'billing'}
+
+ {#if billingStore.isLoading}
+
+
+
+ {:else if !billingStore.isSubscribed}
+
+ {:else}
+
+
+
Plan
+
{BILLING_PLAN.name}
+
${BILLING_PLAN.priceUsd} per month
+ {#if billingStore.isCanceling && formattedEndDate}
+
+ Cancels on {formattedEndDate}
+
+ {:else if formattedEndDate}
+
Renews on {formattedEndDate}
+ {/if}
+
+
+
+
Payment
+ {#if billingStore.summary?.paymentMethod?.card}
+
+
+
+ {billingStore.summary.paymentMethod.card.brand.toUpperCase()} ending in
+ {billingStore.summary.paymentMethod.card.last4}
+
+
+ Expires {billingStore.summary.paymentMethod.card.exp_month}/
+ {billingStore.summary.paymentMethod.card.exp_year}
+
+
+
+
+ {:else}
+
No payment method on file yet.
+ {/if}
+
+
+
+
+
+
+
Manage subscription
+
Update payment method, cancel, or view invoices.
+
+
+ {#if isRedirecting}
+
+ Opening...
+ {:else}
+ Open portal
+
+ {/if}
+
+
+ {#if errorMessage}
+
{errorMessage}
+ {/if}
+
+ {/if}
+
+ {/if}
+
+
+
+
+{#if showCreateModal}
+
+
e.key === 'Escape' && closeCreateModal()}
+ >
+
+ {#if newlyCreatedKey}
+ {@const modalConfig = getConfigWithKey(modalSelectedTool, newlyCreatedKey)}
+ {@const modalEnvVarSnippet = getEnvVarWithKey(modalSelectedTool, newlyCreatedKey)}
+
API Key Created
+
+ Copy your configuration below. You won't be able to see this key again.
+
+
+
+ {#each Object.entries(toolConfigs) as [key, tool]}
+ (modalSelectedTool = key as McpTool)}
+ >
+ {tool.name}
+
+ {/each}
+
+
+
+
+
+ {#if shikiStore.highlighter}
+ {@html shikiStore.highlighter.codeToHtml(modalConfig, {
+ theme: shikiTheme,
+ lang: toolConfigs[modalSelectedTool].language,
+ rootStyle: 'background-color: transparent; padding: 0; margin: 0;'
+ })}
+ {:else}
+
{modalConfig}
+ {/if}
+
+
+
+
+
+
+ {#if toolConfigs[modalSelectedTool].kind === 'command'}
+ Run this command in your terminal.
+ {:else}
+ Add this to {toolConfigs[modalSelectedTool].filename}
+ {/if}
+
+
+ {#if modalEnvVarSnippet}
+
+
Set environment variable:
+
+
+
+ {#if shikiStore.highlighter}
+ {@html shikiStore.highlighter.codeToHtml(modalEnvVarSnippet, {
+ theme: shikiTheme,
+ lang: toolConfigs[modalSelectedTool].envVarLanguage ?? 'bash',
+ rootStyle: 'background-color: transparent; padding: 0; margin: 0;'
+ })}
+ {:else}
+
{modalEnvVarSnippet}
+ {/if}
+
+
+
+
+
+ Add this to .zshenv
+ or your shell profile.
+
+
+ {/if}
+
+
+ {#if copiedAgents}
+ Copied AGENTS.md instructions
+ {:else}
+ Copy AGENTS.md instructions
+ {/if}
+
+
Done
+ {:else}
+
Create API Key
+
+ Give your key a name to help you remember what it's used for.
+
+
e.key === 'Enter' && handleCreateKey()}
+ />
+
+ Cancel
+
+ {#if isCreating}
+
+ Creating...
+ {:else}
+ Create
+ {/if}
+
+
+ {/if}
+
+
+{/if}
diff --git a/apps/desktop/src/routes/app/settings/billing/+page.svelte b/apps/desktop/src/routes/app/settings/billing/+page.svelte
new file mode 100644
index 00000000..a2a64303
--- /dev/null
+++ b/apps/desktop/src/routes/app/settings/billing/+page.svelte
@@ -0,0 +1,181 @@
+
+
+
+
+
+
Billing
+
Manage your subscription and payment details.
+
+
+ {#if billingStore.isLoading}
+
+
+
+ {:else if !billingStore.isSubscribed}
+
+ {:else}
+
+
+
Plan
+
{BILLING_PLAN.name}
+
${BILLING_PLAN.priceUsd} per month
+ {#if billingStore.isCanceling && formattedEndDate}
+
+ Cancels on {formattedEndDate}
+
+ {:else if formattedEndDate}
+
Renews on {formattedEndDate}
+ {/if}
+
+
+
+
Payment
+ {#if billingStore.summary?.paymentMethod?.card}
+
+
+
+ {billingStore.summary.paymentMethod.card.brand.toUpperCase()} ending in
+ {billingStore.summary.paymentMethod.card.last4}
+
+
+ Expires {billingStore.summary.paymentMethod.card.exp_month}/
+ {billingStore.summary.paymentMethod.card.exp_year}
+
+
+
+
+ {:else}
+
No payment method on file yet.
+ {/if}
+
+
+
+
+
+
+
+
+
+
+
Monthly AI usage
+
Higher-end models use this faster.
+
+
+
{Math.round(usedPct)}% used
+
+
+
+
+
+
+
+
Manage subscription
+
Update payment method, cancel, or view invoices.
+
+
+ {#if isRedirecting}
+
+ Opening...
+ {:else}
+ Open portal
+
+ {/if}
+
+
+ {#if errorMessage}
+
{errorMessage}
+ {/if}
+
+ {/if}
+
+
diff --git a/apps/desktop/src/routes/app/settings/questions/+page.svelte b/apps/desktop/src/routes/app/settings/questions/+page.svelte
new file mode 100644
index 00000000..6a91387a
--- /dev/null
+++ b/apps/desktop/src/routes/app/settings/questions/+page.svelte
@@ -0,0 +1,727 @@
+
+
+
+
+
+
+
MCP Questions
+
+ Questions asked via MCP tools for the
+ {#if selectedProject}
+ {selectedProject.name}
+ {:else}
+ selected
+ {/if}
+ project.
+
+
+
goto('/app/settings?tab=mcp')}>
+
+ Configure
+
+
+
+ {#if selectedProject}
+
+
+
+
+
+
+
+ Resource
+
+ ({
+ value: r.name,
+ label: `@${r.name} (${r.count})`
+ }))
+ ]}
+ />
+
+
+
+
+ Sort
+
+
+ {#if sortDirection === 'desc'}
+
+ Newest first
+ {:else}
+
+ Oldest first
+ {/if}
+
+
+
+
+
+ Per page
+
+
+ 10
+ 20
+ 50
+
+
+
+
+
+ Reset
+
+
+
+
+
+
+ {#if total === 0}
+ No results
+ {:else}
+ Showing {startIndex}-{endIndex} of {total}
+ {#if totalAll !== total}
+ (filtered from {totalAll})
+ {/if}
+ {/if}
+
+ {#if hasQueryFilters}
+
+ {#if searchQuery}
+ Search: "{searchQuery}"
+ {/if}
+ {#if resourceFilter}
+ Resource: @{resourceFilter}
+ {/if}
+
+ {/if}
+
+
+
+ {/if}
+
+ {#if !selectedProject}
+
+
+
+
+
No project selected
+
+ Select a project from the sidebar to view its questions.
+
+
+ {:else if isLoading}
+
+
+
+ {:else if totalAll === 0}
+
+
+
+
+
No questions yet
+
+ Questions asked via MCP will appear here. Use the ask tool from
+ your MCP client to get started.
+
+
+ {:else if total === 0}
+
+
+
+
+
No results for these filters
+
+ Try removing a filter or adjusting your search to see more questions.
+
+
+ Reset filters
+
+
+ {:else}
+
+ {#each questions as question (question._id)}
+ {@const isExpanded = expandedQuestions.has(question._id)}
+ {@const needsExpand = shouldShowExpand(question.answer)}
+
+
+
+
+
+
+ {#if isExpanded || !needsExpand}
+
+ {@html renderMarkdown(question.answer)}
+
+ {:else}
+
+ {getPreviewText(question.answer)}
+
+ {/if}
+
+ {#if needsExpand}
+
toggleExpanded(question._id)}
+ >
+ {#if isExpanded}
+
+ Show less
+ {:else}
+
+ Show full answer
+ {/if}
+
+ {/if}
+
+
+ {/each}
+
+
+
+
+ Page {currentPage} of {totalPages}
+
+
+ goToPage(currentPage - 1)}
+ disabled={currentPage <= 1}
+ >
+
+ Previous
+
+ goToPage(currentPage + 1)}
+ disabled={currentPage >= totalPages}
+ >
+ Next
+
+
+
+
+ {/if}
+
+
+
+
diff --git a/apps/desktop/src/routes/app/settings/resources/+page.svelte b/apps/desktop/src/routes/app/settings/resources/+page.svelte
new file mode 100644
index 00000000..5de5fa24
--- /dev/null
+++ b/apps/desktop/src/routes/app/settings/resources/+page.svelte
@@ -0,0 +1,978 @@
+
+
+
+
+
+
+
Resources
+
+ Manage your available documentation resources. Use @mentions in chat to query them.
+
+
+
+
+
+
+
+
+
GitHub Private Repos
+
+
+ Private GitHub repositories work only in the signed-in web app sandbox. Your local CLI
+ still uses the git auth on your own machine.
+
+
+
+ {#if githubConnection?.status === 'connected'}
+ Connected to @{githubConnection.githubLogin}
+ {:else if githubConnection?.status === 'missing_scope'}
+ GitHub connected, but private repo access is missing
+ {:else}
+ GitHub not connected
+ {/if}
+
+ {#if githubConnection?.status === 'connected'}
+ Scopes: {githubConnection.scopes.join(', ')}
+ {/if}
+
+
+ If GitHub is already connected without private repo access, btca will try a direct
+ reconnect first and fall back to the Clerk profile modal with the required scope. You
+ can also refresh GitHub access later if you need newly granted org access to show up.
+
+ {#if githubSyncError}
+
{githubSyncError}
+ {/if}
+
+
+ {#if githubConnection?.status === 'connected'}
+
+ {#if isReauthorizingGitHub}
+
+ {:else}
+
+ {/if}
+ Refresh GitHub Access
+
+ {/if}
+
+ {#if isSyncingGitHub}
+
+ {:else}
+
+ {/if}
+ Refresh Status
+
+
+ {#if isReauthorizingGitHub}
+
+ {:else}
+
+ {/if}
+ {githubConnection?.status === 'missing_scope'
+ ? 'Reconnect GitHub'
+ : githubConnection?.status === 'connected'
+ ? 'Manage GitHub'
+ : 'Connect GitHub'}
+
+
+
+
+
+
+
+
+
+
+
Your Custom Resources
+
+
+
+
+
+ All Projects
+
+
(showAddForm ? closeAddForm() : openAddForm('git'))}
+ >
+
+ Add Resource
+
+
+
+
+ Add your own git repositories or npm packages as documentation resources.
+
+
+ Public repos can be added directly. Private GitHub repos require a connected GitHub account
+ with private repository access. npm packages can be added with an optional pinned version.
+
+
+
+
+
+
+
Quick Add Git Repo
+
+
+ e.key === 'Enter' && quickAddUrl.trim() && handleQuickAdd()}
+ />
+
+ {#if isParsingUrl}
+
+ {:else}
+ Add
+ {/if}
+
+
+ {#if parseError}
+
{parseError}
+ {/if}
+
+
+
+ {#if showConfirmation}
+
+
+
+
Confirm Resource Details
+
+
+ {#if formError}
+
+ {formError}
+
+ {/if}
+
+
+
+
Name *
+
+
Used as @mention (e.g., @{formName || 'name'})
+
+
+
+ Git URL *
+
+
+
+
+
+
+ Notes
+
+
+
+
+
+
+ Cancel
+
+
+ {#if isSubmitting}
+
+ Adding...
+ {:else}
+
+ Confirm & Add
+ {/if}
+
+
+
+
+ {/if}
+
+
+ {#if showAddForm}
+
+
Add Custom Resource
+
+ {#if formError}
+
+ {formError}
+
+ {/if}
+
+
+
+
resetForm('git')}
+ >
+
+
+ Git Repository
+
+ Repo URL, branch, and optional docs path.
+
+
resetForm('npm')}
+ >
+
+ Package name and an optional pinned version.
+
+
+
+
+
Name *
+
+
+ Used as @mention (e.g., @myFramework). Allowed: letters, numbers, ., _, -, / (no
+ spaces)
+
+
+
+ {#if formType === 'git'}
+
+ Git URL *
+
+
+
+
+ {:else}
+
+ npm Package *
+
+
+
+
+
Version or Tag
+
+
+ Leave blank to use the latest published version.
+
+
+ {/if}
+
+
+ Notes
+
+
+
+
+
+ Cancel
+
+
+ {#if isSubmitting}
+
+ Adding...
+ {:else}
+ Add {formType === 'git' ? 'Repository' : 'Package'}
+ {/if}
+
+
+
+
+ {/if}
+
+
+ {#if userResourcesQuery?.isLoading}
+
+
+
+ {:else if userResourcesQuery?.data && userResourcesQuery.data.length > 0}
+
+ {#each userResourcesQuery.data as resource (resource._id)}
+
+
+
+
@{resource.name}
+ {#if resource.type === 'npm'}
+
+
+ npm
+
+ {/if}
+ {#if resource.visibility === 'private'}
+
+
+ Private
+
+ {/if}
+
+
+ {#if resource.url || resource.package}
+
+
+
+ {/if}
+
handleRemoveResource(resource._id)}
+ >
+
+
+
+
+
+ {getResourceLabel(resource)}
+
+
+ {getResourceSummary(resource)}
+
+ {#if resource.specialNotes}
+
{resource.specialNotes}
+ {/if}
+
+ {/each}
+
+ {:else}
+
+
No custom resources added yet
+
openAddForm('git')}
+ >
+
+ Add Your First Resource
+
+
+ {/if}
+
+
+
+
+
+
Global Catalog
+
+ Click a resource to add it to your instance.
+
+ {#if globalAddError}
+
+ {globalAddError}
+
+ {/if}
+
+ {#if GLOBAL_RESOURCES.length > 0}
+
+ {#each GLOBAL_RESOURCES as resource (resource.name)}
+
handleAddGlobalResource(resource)}
+ disabled={userResourceNames.has(resource.name) || addingGlobal === resource.name}
+ >
+
+
+
@{resource.name}
+ {#if addingGlobal === resource.name}
+
+ {:else if userResourceNames.has(resource.name)}
+
+ {:else}
+
+ {/if}
+
+
+ {/each}
+
+ {:else}
+
+
No global resources available
+
+ {/if}
+
+
+
+
+
diff --git a/apps/desktop/src/routes/app/support/+page.svelte b/apps/desktop/src/routes/app/support/+page.svelte
new file mode 100644
index 00000000..da015145
--- /dev/null
+++ b/apps/desktop/src/routes/app/support/+page.svelte
@@ -0,0 +1,50 @@
+
+
+
+
+
+
+
+
+
FAQs
+
Stubbed answers for now. Replace as needed.
+
+ {#each faqs as faq}
+
+
{faq.question}
+
{faq.answer}
+
+ {/each}
+
+
+
+
+
diff --git a/apps/desktop/src/routes/layout.css b/apps/desktop/src/routes/layout.css
new file mode 100644
index 00000000..7b71f3ea
--- /dev/null
+++ b/apps/desktop/src/routes/layout.css
@@ -0,0 +1,1244 @@
+@import 'tailwindcss';
+@plugin '@tailwindcss/typography';
+
+@custom-variant dark (&:where(.dark, .dark *));
+
+@font-face {
+ font-family: 'GeistMono';
+ src: url('/GeistMono[wght].woff2') format('woff2');
+ font-weight: 100 900;
+ font-style: normal;
+}
+
+@font-face {
+ font-family: 'Geist';
+ src: url('/Geist[wght].woff2') format('woff2');
+ font-weight: 100 900;
+ font-style: normal;
+}
+
+@theme {
+ --font-family-geist:
+ 'Geist', ui-sans-serif, system-ui, -apple-system, 'BlinkMacSystemFont', 'Segoe UI', sans-serif;
+ --font-family-geist-mono:
+ 'GeistMono', ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono',
+ 'Courier New', monospace;
+ --font-family-ui: var(--font-family-geist);
+}
+
+@layer base {
+ :root {
+ color-scheme: light;
+
+ --btca-brand: #4783eb;
+ --btca-brand-light: #9abbf4;
+ --btca-brand-dark: #1450b8;
+
+ --btca-n0: #fafafa;
+ --btca-n50: #f0f0f0;
+ --btca-n100: #e6e6e6;
+ --btca-n200: #d6d6d6;
+ --btca-n300: #c1c1c1;
+ --btca-n400: #a5a5a5;
+ --btca-n500: #7f7f7f;
+ --btca-n600: #616161;
+ --btca-n700: #474747;
+ --btca-n800: #303030;
+ --btca-n900: #1a1a1a;
+
+ --btca-success: #1e7d4f;
+ --btca-warning: #b87416;
+ --btca-error: #b13a3a;
+ --btca-info: #2b6cb0;
+
+ --btca-radius-xs: 0px;
+ --btca-radius-sm: 0px;
+ --btca-radius-md: 0px;
+ --btca-radius-lg: 0px;
+
+ --btca-elevation-1: 0 1px 2px rgba(0, 0, 0, 0.06);
+ --btca-elevation-2: 0 4px 12px rgba(0, 0, 0, 0.08);
+ --btca-elevation-3: 0 10px 24px rgba(0, 0, 0, 0.12);
+
+ --bc-bg: 0 0% 94%;
+ --bc-bg-2: 0 0% 90%;
+ --bc-bg-muted: 0 0% 90%;
+ --bc-fg: 0 0% 10%;
+ --bc-fg-muted: 0 0% 38%;
+ --bc-text: var(--bc-fg);
+ --bc-muted: var(--bc-fg-muted);
+ --bc-border: 0 0% 76%;
+ --bc-surface: 0 0% 98%;
+ --bc-surface-2: 0 0% 90%;
+ --bc-bg-secondary: var(--bc-surface-2);
+
+ --bc-accent: 218 80% 60%;
+ --bc-accent-2: 0 0% 12.2%;
+ --bc-accent-3: 0 0% 12.2%;
+
+ --bc-success: 150.9 61.3% 30.4%;
+ --bc-warning: 34.8 78.6% 40.4%;
+ --bc-error: 0 50.6% 46.1%;
+ --bc-info: 210.7 60.7% 42.9%;
+
+ --bc-code-bg: var(--bc-surface);
+ --bc-code-fg: var(--bc-fg);
+ --bc-code-accent: var(--bc-accent);
+
+ --bc-shadow: 0 0% 0%;
+ }
+
+ :root.dark {
+ color-scheme: dark;
+
+ --btca-brand: #4783eb;
+ --btca-brand-light: #9abbf4;
+ --btca-brand-dark: #1450b8;
+
+ --btca-n0: #161616;
+ --btca-n50: #2a2a2a;
+ --btca-n100: #3b3b3b;
+ --btca-n200: #4f4f4f;
+ --btca-n300: #6a6a6a;
+ --btca-n400: #888888;
+ --btca-n500: #b0b0b0;
+ --btca-n600: #d0d0d0;
+ --btca-n700: #e6e6e6;
+ --btca-n800: #f0f0f0;
+ --btca-n900: #fafafa;
+
+ --btca-success: #1e7d4f;
+ --btca-warning: #b87416;
+ --btca-error: #b13a3a;
+ --btca-info: #2b6cb0;
+
+ --btca-radius-xs: 0px;
+ --btca-radius-sm: 0px;
+ --btca-radius-md: 0px;
+ --btca-radius-lg: 0px;
+
+ --btca-elevation-1: 0 1px 2px rgba(0, 0, 0, 0.3);
+ --btca-elevation-2: 0 4px 12px rgba(0, 0, 0, 0.35);
+ --btca-elevation-3: 0 10px 24px rgba(0, 0, 0, 0.45);
+
+ --bc-bg: 0 0% 10%;
+ --bc-bg-2: 0 0% 18%;
+ --bc-bg-muted: 0 0% 26%;
+ --bc-fg: 0 0% 96%;
+ --bc-fg-muted: 0 0% 68%;
+ --bc-text: var(--bc-fg);
+ --bc-muted: var(--bc-fg-muted);
+ --bc-border: 0 0% 26%;
+ --bc-surface: 0 0% 18%;
+ --bc-surface-2: 0 0% 26%;
+ --bc-bg-secondary: var(--bc-surface-2);
+
+ --bc-accent: 218 80% 60%;
+ --bc-accent-2: 0 0% 96.9%;
+ --bc-accent-3: 0 0% 96.9%;
+
+ --bc-success: 150.9 61.3% 30.4%;
+ --bc-warning: 34.8 78.6% 40.4%;
+ --bc-error: 0 50.6% 46.1%;
+ --bc-info: 210.7 60.7% 42.9%;
+
+ --bc-code-bg: var(--bc-surface);
+ --bc-code-fg: var(--bc-fg);
+ --bc-code-accent: var(--bc-accent);
+
+ --bc-shadow: 0 0% 0%;
+ }
+
+ html,
+ body {
+ height: 100%;
+ font-family: var(--font-family-ui);
+ font-variation-settings: 'opsz' 28;
+ text-rendering: geometricPrecision;
+ }
+
+ body {
+ background: hsl(var(--bc-bg));
+ color: hsl(var(--bc-fg));
+ @apply antialiased;
+ }
+
+ ::selection {
+ background: color-mix(in oklab, hsl(var(--bc-accent)) 22%, transparent);
+ }
+
+ a {
+ @apply underline underline-offset-4;
+ text-decoration-color: color-mix(in oklab, hsl(var(--bc-border)) 60%, transparent);
+ }
+
+ a:hover {
+ text-decoration-color: color-mix(in oklab, hsl(var(--bc-accent)) 65%, transparent);
+ }
+
+ :focus-visible {
+ outline: 2px solid color-mix(in oklab, hsl(var(--bc-accent)) 85%, white);
+ outline-offset: 2px;
+ border-radius: 0;
+ }
+
+ .rounded,
+ .rounded-sm,
+ .rounded-md,
+ .rounded-lg,
+ .rounded-xl,
+ .rounded-2xl,
+ .rounded-3xl,
+ .rounded-full {
+ border-radius: 0 !important;
+ }
+
+ code,
+ pre,
+ textarea {
+ font-family: var(--font-family-geist-mono);
+ font-variation-settings: 'opsz' 28;
+ }
+}
+
+@layer components {
+ .bc-container {
+ @apply mx-auto w-full max-w-5xl px-6;
+ }
+
+ .bc-skip {
+ position: absolute;
+ left: 16px;
+ top: 12px;
+ z-index: 50;
+ transform: translateY(-140%);
+ transition: transform 180ms ease;
+ }
+
+ .bc-skip:focus-within {
+ transform: translateY(0);
+ }
+
+ .bc-skipLink {
+ display: inline-flex;
+ align-items: center;
+ gap: 8px;
+ padding: 10px 12px;
+ border-radius: 0;
+ background: hsl(var(--bc-surface));
+ border: 1px solid hsl(var(--bc-border));
+ color: hsl(var(--bc-fg));
+ text-decoration: none;
+ font-weight: 600;
+ letter-spacing: 0;
+ }
+
+ .bc-appBg {
+ background: hsl(var(--bc-bg));
+ }
+
+ .bc-header {
+ border-bottom: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-bg));
+ backdrop-filter: none;
+ }
+
+ .bc-logoMark {
+ display: grid;
+ place-items: center;
+ width: 42px;
+ height: 42px;
+ border-radius: 0;
+ background: hsl(var(--bc-surface));
+ border: 1px solid hsl(var(--bc-border));
+ color: hsl(var(--bc-fg));
+ }
+
+ .bc-title {
+ font-weight: 650;
+ letter-spacing: -0.03em;
+ }
+
+ .bc-subtitle {
+ color: hsl(var(--bc-fg-muted));
+ letter-spacing: 0.01em;
+ }
+
+ .bc-navLink {
+ display: inline-flex;
+ align-items: center;
+ gap: 8px;
+ padding: 10px 12px;
+ border-radius: 0;
+ text-decoration: none;
+ color: hsl(var(--bc-fg));
+ font-weight: 600;
+ font-size: 13px;
+ letter-spacing: 0.02em;
+ background: transparent;
+ transition:
+ background 120ms ease,
+ box-shadow 120ms ease;
+ }
+
+ .bc-navLink:hover {
+ background: color-mix(in oklab, hsl(var(--bc-surface)) 55%, transparent);
+ box-shadow: 0 1px 0 color-mix(in oklab, hsl(var(--bc-border)) 80%, transparent);
+ }
+
+ .bc-navLink-active {
+ color: hsl(var(--bc-accent));
+ box-shadow: inset 0 -2px 0 hsl(var(--bc-accent));
+ }
+
+ .bc-chip {
+ display: inline-flex;
+ align-items: center;
+ gap: 10px;
+ padding: 10px 12px;
+ border-radius: 0;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ color: hsl(var(--bc-fg));
+ text-decoration: none;
+ transition:
+ background 120ms ease,
+ border-color 120ms ease,
+ box-shadow 120ms ease;
+ }
+
+ .bc-chip:hover {
+ border-color: hsl(var(--bc-fg));
+ background: hsl(var(--bc-surface-2));
+ box-shadow: 0 1px 0 color-mix(in oklab, hsl(var(--bc-border)) 80%, transparent);
+ }
+
+ .bc-chatPattern {
+ background-image:
+ radial-gradient(
+ color-mix(in oklab, hsl(var(--bc-fg)) 7%, transparent) 0.6px,
+ transparent 0.6px
+ ),
+ radial-gradient(
+ color-mix(in oklab, hsl(var(--bc-fg)) 4%, transparent) 0.6px,
+ transparent 0.6px
+ );
+ background-size:
+ 26px 26px,
+ 18px 18px;
+ background-position:
+ 0 0,
+ 8px 10px;
+ }
+
+ .bc-card {
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ border-radius: 0;
+ }
+
+ .bc-sidebar {
+ background: hsl(var(--bc-bg));
+ border-right: 1px solid hsl(var(--bc-border));
+ }
+
+ .bc-sidebar-inner {
+ display: flex;
+ flex-direction: column;
+ height: 100%;
+ }
+
+ .bc-sidebar-section {
+ padding: 16px;
+ border-bottom: 1px solid hsl(var(--bc-border));
+ }
+
+ .bc-sidebar-actions {
+ display: grid;
+ grid-template-columns: repeat(2, minmax(0, 1fr));
+ gap: 8px;
+ }
+
+ .bc-sidebar-search {
+ display: flex;
+ align-items: center;
+ gap: 8px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ padding: 8px 10px;
+ }
+
+ .bc-sidebar-search-input {
+ flex: 1;
+ background: transparent;
+ border: none;
+ font-size: 12px;
+ color: hsl(var(--bc-fg));
+ }
+
+ .bc-sidebar-search-input:focus {
+ outline: none;
+ }
+
+ .bc-sidebar-search-input::placeholder {
+ color: hsl(var(--bc-fg-muted));
+ }
+
+ .bc-sidebar-clear {
+ display: inline-flex;
+ align-items: center;
+ justify-content: center;
+ width: 28px;
+ height: 28px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ color: hsl(var(--bc-fg-muted));
+ cursor: pointer;
+ }
+
+ .bc-sidebar-clear:hover {
+ color: hsl(var(--bc-fg));
+ border-color: hsl(var(--bc-fg));
+ background: hsl(var(--bc-surface));
+ }
+
+ .bc-thread-list {
+ flex: 1;
+ overflow-y: auto;
+ padding: 12px;
+ display: flex;
+ flex-direction: column;
+ gap: 8px;
+ scrollbar-width: thin;
+ scrollbar-color: hsl(var(--bc-border)) transparent;
+ }
+
+ .bc-thread-list::-webkit-scrollbar {
+ width: 4px;
+ }
+
+ .bc-thread-list::-webkit-scrollbar-track {
+ background: transparent;
+ }
+
+ .bc-thread-list::-webkit-scrollbar-thumb {
+ background-color: hsl(var(--bc-border));
+ border-radius: 0;
+ }
+
+ .bc-thread-list::-webkit-scrollbar-thumb:hover {
+ background-color: hsl(var(--bc-fg-muted));
+ }
+
+ .bc-thread-item {
+ display: flex;
+ align-items: center;
+ gap: 10px;
+ padding: 10px 12px;
+ border: 1px solid transparent;
+ text-decoration: none;
+ color: hsl(var(--bc-fg));
+ background: transparent;
+ transition:
+ border-color 120ms ease,
+ background 120ms ease;
+ }
+
+ .bc-thread-item:hover {
+ border-color: hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .bc-thread-item-active {
+ border-color: hsl(var(--bc-accent));
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .bc-thread-item-delete {
+ opacity: 0;
+ border: none;
+ background: transparent;
+ color: hsl(var(--bc-fg-muted));
+ transition:
+ opacity 120ms ease,
+ color 120ms ease;
+ cursor: pointer;
+ }
+
+ .bc-thread-item:hover .bc-thread-item-delete,
+ .bc-thread-item-active .bc-thread-item-delete {
+ opacity: 1;
+ }
+
+ .bc-thread-item-delete:hover {
+ color: hsl(var(--bc-error));
+ }
+
+ .bc-sidebar-footer {
+ margin-top: auto;
+ }
+
+ .bc-sidebar-dropdown {
+ box-shadow: 0 12px 24px hsl(var(--bc-shadow) / 0.12);
+ }
+
+ .bc-sidebar-menu-item {
+ display: flex;
+ align-items: center;
+ gap: 8px;
+ width: 100%;
+ padding: 8px 10px;
+ border: none;
+ background: transparent;
+ text-decoration: none;
+ color: hsl(var(--bc-fg));
+ cursor: pointer;
+ font-size: 12px;
+ }
+
+ .bc-sidebar-menu-item:hover {
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .bc-cardHover {
+ transition:
+ box-shadow 220ms ease,
+ border-color 220ms ease,
+ background 220ms ease;
+ }
+
+ .bc-cardHover:hover {
+ border-color: color-mix(in oklab, hsl(var(--bc-accent-2)) 30%, hsl(var(--bc-border)));
+ background: color-mix(in oklab, hsl(var(--bc-surface)) 82%, transparent);
+ box-shadow: 0 12px 24px -18px hsl(var(--bc-shadow) / 0.35);
+ }
+
+ .bc-btn {
+ display: inline-flex;
+ align-items: center;
+ justify-content: center;
+ gap: 8px;
+ padding: 10px 16px;
+ border-radius: 0;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ color: hsl(var(--bc-fg));
+ font-weight: 600;
+ font-size: 13px;
+ text-decoration: none;
+ cursor: pointer;
+ transition:
+ background 120ms ease,
+ border-color 120ms ease;
+ }
+
+ .bc-btn:hover {
+ border-color: hsl(var(--bc-fg));
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .bc-btn:disabled {
+ opacity: 0.5;
+ cursor: not-allowed;
+ }
+
+ .bc-btn-primary {
+ background: hsl(var(--bc-accent));
+ color: #ffffff;
+ border-color: hsl(var(--bc-accent));
+ }
+
+ .bc-btn-primary:hover {
+ background: color-mix(in oklab, hsl(var(--bc-accent)) 90%, black);
+ border-color: hsl(var(--bc-accent));
+ }
+
+ .bc-btnPrimary {
+ background: hsl(var(--bc-accent));
+ color: #ffffff;
+ border: 1px solid hsl(var(--bc-accent));
+ }
+
+ .bc-btnPrimary:hover {
+ background: color-mix(in oklab, hsl(var(--bc-accent)) 90%, black);
+ border-color: hsl(var(--bc-accent));
+ color: #ffffff;
+ filter: saturate(1.05) contrast(1.03);
+ }
+
+ .bc-iconBtn {
+ display: inline-flex;
+ align-items: center;
+ justify-content: center;
+ width: 38px;
+ height: 38px;
+ border-radius: 0;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ color: hsl(var(--bc-fg));
+ transition:
+ background 120ms ease,
+ border-color 120ms ease,
+ box-shadow 120ms ease;
+ }
+
+ .bc-iconBtn:hover {
+ border-color: color-mix(in oklab, hsl(var(--bc-accent-2)) 30%, hsl(var(--bc-border)));
+ background: color-mix(in oklab, hsl(var(--bc-surface)) 88%, transparent);
+ box-shadow: 0 1px 0 color-mix(in oklab, hsl(var(--bc-border)) 80%, transparent);
+ }
+
+ .bc-copied {
+ border-color: hsl(var(--bc-accent));
+ background: hsl(var(--bc-surface));
+ }
+
+ .bc-input {
+ width: 100%;
+ padding: 12px 16px;
+ border-radius: 0;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ color: hsl(var(--bc-fg));
+ font-size: 14px;
+ transition: border-color 120ms ease;
+ }
+
+ .bc-input:focus {
+ outline: none;
+ border-color: hsl(var(--bc-accent));
+ }
+
+ .bc-input::placeholder {
+ color: hsl(var(--bc-fg-muted) / 0.35);
+ }
+
+ .bc-muted {
+ color: hsl(var(--bc-fg-muted));
+ }
+
+ .bc-badge {
+ display: inline-flex;
+ align-items: center;
+ gap: 6px;
+ padding: 4px 8px;
+ border-radius: 0;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ font-size: 12px;
+ font-weight: 600;
+ }
+
+ .bc-badge-success {
+ border-color: hsl(var(--bc-success));
+ color: hsl(var(--bc-success));
+ }
+
+ .bc-badge-warning {
+ border-color: hsl(var(--bc-warning));
+ color: hsl(var(--bc-warning));
+ }
+
+ .bc-badge-error {
+ border-color: hsl(var(--bc-error));
+ color: hsl(var(--bc-error));
+ }
+
+ .bc-badgeAccent {
+ border-color: hsl(var(--bc-accent));
+ background: hsl(var(--bc-surface));
+ }
+
+ .bc-kicker {
+ display: inline-flex;
+ align-items: center;
+ gap: 10px;
+ font-size: 12px;
+ letter-spacing: 0.02em;
+ text-transform: none;
+ font-weight: 600;
+ color: hsl(var(--bc-fg-muted));
+ }
+
+ .bc-kickerDot {
+ width: 8px;
+ height: 8px;
+ border-radius: 0;
+ background: hsl(var(--bc-accent));
+ }
+
+ .bc-h1 {
+ font-weight: 650;
+ letter-spacing: -0.04em;
+ line-height: 1.05;
+ }
+
+ .bc-prose {
+ color: hsl(var(--bc-fg-muted));
+ line-height: 1.7;
+ max-width: 70ch;
+ }
+
+ .bc-inlineCode {
+ display: inline-flex;
+ align-items: center;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ border-radius: 0;
+ padding: 0.15rem 0.4rem;
+ font-size: 0.85em;
+ font-weight: 600;
+ color: hsl(var(--bc-fg));
+ }
+
+ .bc-tag {
+ display: inline-flex;
+ align-items: center;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ border-radius: 0;
+ padding: 0.35rem 0.6rem;
+ font-size: 0.875rem;
+ font-weight: 650;
+ color: hsl(var(--bc-fg));
+ }
+
+ .bc-codeFrame {
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ border-radius: 0;
+ overflow: hidden;
+ position: relative;
+ max-width: 100%;
+ }
+
+ .bc-codeFrame pre {
+ overflow-x: auto;
+ scrollbar-width: thin;
+ }
+
+ .bc-codeFrame pre::-webkit-scrollbar {
+ height: 6px;
+ }
+
+ .bc-codeFrame pre::-webkit-scrollbar-thumb {
+ background-color: hsl(var(--bc-border));
+ border-radius: 0;
+ }
+
+ .bc-codeFrame pre::-webkit-scrollbar-track {
+ background-color: transparent;
+ }
+
+ .bc-codeFrame::after {
+ content: '';
+ position: absolute;
+ inset: 0;
+ background: linear-gradient(90deg, transparent, hsl(var(--bc-fg) / 0.06), transparent);
+ transform: translateX(-120%);
+ animation: bc-sheen 7.5s ease-in-out infinite;
+ pointer-events: none;
+ }
+
+ .bc-ring {
+ box-shadow: none;
+ }
+
+ /* Chat-specific styles */
+ .chat-messages {
+ display: flex;
+ flex-direction: column;
+ gap: 16px;
+ overflow-y: auto;
+ padding: 20px;
+ max-width: 800px;
+ margin: 0 auto;
+ width: 100%;
+ }
+
+ .chat-message {
+ display: flex;
+ flex-direction: column;
+ padding: 12px 16px;
+ }
+
+ .chat-message-user {
+ background: hsl(var(--bc-surface-2));
+ border: 1px solid hsl(var(--bc-border));
+ }
+
+ .chat-message-assistant {
+ background: transparent;
+ padding: 6px 0;
+ }
+
+ .chat-message-system {
+ background: hsl(var(--bc-surface));
+ border: 1px solid hsl(var(--bc-border));
+ padding: 12px 16px;
+ }
+
+ .chat-input-container {
+ padding: 16px 20px;
+ border-top: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-bg));
+ max-width: 64rem;
+ margin: 0 auto;
+ width: 100%;
+ }
+
+ .input-wrapper {
+ position: relative;
+ display: flex;
+ align-items: flex-start;
+ gap: 8px;
+ }
+
+ .chat-input-highlight-wrapper {
+ flex: 1;
+ position: relative;
+ }
+
+ .chat-input-backdrop {
+ position: absolute;
+ inset: 0;
+ padding: 12px 14px;
+ border: 1px solid transparent;
+ font-size: 15px;
+ font-family: var(--font-family-ui);
+ line-height: 1.5;
+ white-space: pre-wrap;
+ word-wrap: break-word;
+ overflow: hidden;
+ pointer-events: none;
+ color: transparent;
+ }
+
+ .mention-highlight {
+ color: hsl(var(--bc-accent));
+ background: hsl(var(--bc-accent) / 0.15);
+ }
+
+ .chat-input {
+ position: relative;
+ width: 100%;
+ padding: 12px 14px;
+ border: 1px solid hsl(var(--bc-accent));
+ background: transparent;
+ color: hsl(var(--bc-fg));
+ caret-color: hsl(var(--bc-fg));
+ font-size: 15px;
+ font-family: var(--font-family-ui);
+ resize: none;
+ line-height: 1.5;
+ field-sizing: content;
+ min-height: calc(2lh + 24px);
+ max-height: calc(4lh + 24px);
+ overflow-y: auto;
+ }
+
+ .chat-input:focus {
+ outline: none;
+ border-color: hsl(var(--bc-accent));
+ }
+
+ .chat-input::placeholder {
+ color: hsl(var(--bc-fg-muted));
+ }
+
+ .chat-input:disabled {
+ opacity: 0.5;
+ cursor: not-allowed;
+ }
+
+ .send-btn {
+ flex-shrink: 0;
+ width: 42px;
+ height: 42px;
+ padding: 0;
+ background: hsl(var(--bc-fg));
+ color: hsl(var(--bc-bg));
+ border: none;
+ cursor: pointer;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ }
+
+ .send-btn:hover:not(:disabled) {
+ background: hsl(var(--bc-accent));
+ }
+
+ .send-btn:disabled {
+ opacity: 0.5;
+ cursor: not-allowed;
+ }
+
+ .mention-menu {
+ position: absolute;
+ bottom: calc(100% + 4px);
+ left: 0;
+ right: 0;
+ background: hsl(var(--bc-surface));
+ border: 1px solid hsl(var(--bc-accent));
+ max-height: 200px;
+ overflow-y: auto;
+ z-index: 50;
+ }
+
+ .mention-item {
+ display: flex;
+ align-items: center;
+ justify-content: space-between;
+ padding: 10px 14px;
+ cursor: pointer;
+ border-bottom: 1px solid hsl(var(--bc-border));
+ }
+
+ .mention-item:last-child {
+ border-bottom: none;
+ }
+
+ .mention-item:hover,
+ .mention-item-selected {
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .mention-item-selected {
+ color: hsl(var(--bc-accent));
+ }
+
+ /* Markdown prose styling */
+ .prose {
+ --tw-prose-body: hsl(var(--bc-fg));
+ --tw-prose-headings: hsl(var(--bc-fg));
+ --tw-prose-lead: hsl(var(--bc-fg));
+ --tw-prose-links: hsl(var(--bc-accent));
+ --tw-prose-bold: hsl(var(--bc-fg));
+ --tw-prose-counters: hsl(var(--bc-fg-muted));
+ --tw-prose-bullets: hsl(var(--bc-fg-muted));
+ --tw-prose-hr: hsl(var(--bc-border));
+ --tw-prose-quotes: hsl(var(--bc-fg));
+ --tw-prose-quote-borders: hsl(var(--bc-border));
+ --tw-prose-captions: hsl(var(--bc-fg-muted));
+ --tw-prose-code: hsl(var(--bc-fg));
+ --tw-prose-pre-code: hsl(var(--bc-fg));
+ --tw-prose-pre-bg: hsl(var(--bc-surface-2));
+ --tw-prose-th-borders: hsl(var(--bc-border));
+ --tw-prose-td-borders: hsl(var(--bc-border));
+
+ color: hsl(var(--bc-fg));
+ font-size: 15px;
+ line-height: 1.8;
+ }
+
+ .prose p {
+ margin-bottom: 1em;
+ }
+
+ .prose ul,
+ .prose ol {
+ margin-top: 0.75em;
+ margin-bottom: 0.75em;
+ }
+
+ .prose li {
+ margin-top: 0.375em;
+ margin-bottom: 0.375em;
+ }
+
+ .prose :where(code):not(:where(pre code))::before,
+ .prose :where(code):not(:where(pre code))::after {
+ content: none;
+ }
+
+ .prose :where(code):not(:where(pre code)) {
+ font-size: 0.875em;
+ padding: 0.2em 0.4em;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ }
+
+ .code-block-wrapper {
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ margin: 6px 0;
+ overflow: hidden;
+ }
+
+ .code-block-header {
+ display: flex;
+ align-items: center;
+ justify-content: space-between;
+ padding: 4px 8px;
+ border-bottom: 1px solid hsl(var(--bc-border));
+ font-size: 12px;
+ }
+
+ .code-lang {
+ color: hsl(var(--bc-fg-muted));
+ text-transform: uppercase;
+ letter-spacing: 0.05em;
+ font-weight: 600;
+ }
+
+ .copy-btn {
+ display: flex;
+ align-items: center;
+ gap: 4px;
+ padding: 4px 8px;
+ background: transparent;
+ border: 1px solid hsl(var(--bc-border));
+ color: hsl(var(--bc-fg-muted));
+ font-size: 11px;
+ cursor: pointer;
+ transition: all 100ms ease;
+ }
+
+ .copy-btn:hover {
+ background: hsl(var(--bc-surface-2));
+ color: hsl(var(--bc-fg));
+ border-color: hsl(var(--bc-fg-muted));
+ }
+
+ .code-content {
+ overflow-x: auto;
+ padding: 4px 6px;
+ }
+
+ /* Shiki code blocks */
+ .prose pre.shiki,
+ .prose pre.shiki code,
+ pre.shiki,
+ pre.shiki code {
+ font-family: var(--font-family-geist-mono);
+ font-size: 14px !important;
+ line-height: 1.6;
+ }
+
+ pre.shiki {
+ background: transparent !important;
+ margin: 0;
+ padding: 0;
+ }
+
+ pre.shiki code {
+ padding: 0;
+ border: none;
+ background: transparent;
+ }
+
+ pre.shiki span {
+ color: var(--shiki-light, inherit);
+ }
+
+ :root.dark pre.shiki span {
+ color: var(--shiki-dark, inherit);
+ }
+
+ .prose pre:not(.shiki):not(.code-content pre) {
+ padding: 12px 14px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface));
+ overflow-x: auto;
+ font-size: 14px !important;
+ line-height: 1.6;
+ margin: 12px 0;
+ }
+
+ .prose :where(pre):not(:where([class~='not-prose'], [class~='not-prose'] *)) {
+ margin-top: 2px !important;
+ margin-bottom: 2px !important;
+ padding: 1 !important;
+ }
+
+ .prose pre code {
+ padding: 0;
+ border: none;
+ background: transparent;
+ font-size: inherit;
+ }
+
+ .copy-answer-btn {
+ display: flex;
+ align-items: center;
+ gap: 4px;
+ padding: 4px 8px;
+ background: transparent;
+ border: 1px solid hsl(var(--bc-border));
+ color: hsl(var(--bc-fg-muted));
+ font-size: 11px;
+ cursor: pointer;
+ transition: all 100ms ease;
+ }
+
+ .copy-answer-btn:hover {
+ background: hsl(var(--bc-surface-2));
+ color: hsl(var(--bc-fg));
+ border-color: hsl(var(--bc-fg-muted));
+ }
+
+ .sandbox-status-indicator {
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ width: 36px;
+ height: 36px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ flex-shrink: 0;
+ }
+
+ .sandbox-progress-bar {
+ height: 4px;
+ width: 200px;
+ background: hsl(var(--bc-surface-2));
+ border: 1px solid hsl(var(--bc-border));
+ overflow: hidden;
+ }
+
+ .sandbox-progress-fill {
+ height: 100%;
+ background: hsl(var(--bc-accent));
+ transition: width 300ms ease;
+ }
+
+ .reasoning-block {
+ padding: 8px 12px;
+ border-left: 2px solid hsl(var(--bc-fg-muted) / 0.4);
+ background: hsl(var(--bc-surface-2) / 0.5);
+ color: hsl(var(--bc-fg-muted));
+ font-size: 12px;
+ line-height: 1.5;
+ }
+
+ .tool-indicator {
+ display: flex;
+ align-items: center;
+ gap: 8px;
+ padding: 6px 10px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ font-size: 12px;
+ width: fit-content;
+ }
+
+ .tool-dot {
+ width: 8px;
+ height: 8px;
+ border-radius: 0;
+ }
+
+ .tool-dot-pending {
+ background: hsl(var(--bc-fg-muted));
+ }
+
+ .tool-dot-completed {
+ background: hsl(var(--bc-success));
+ }
+
+ .tool-summary {
+ display: flex;
+ flex-wrap: wrap;
+ align-items: center;
+ gap: 8px;
+ padding: 6px 10px;
+ border: 1px solid hsl(var(--bc-border));
+ background: hsl(var(--bc-surface-2));
+ font-size: 12px;
+ margin-bottom: 12px;
+ }
+
+ .tool-summary-label {
+ font-weight: 600;
+ color: hsl(var(--bc-fg));
+ }
+
+ .tool-summary-items {
+ display: flex;
+ flex-wrap: wrap;
+ gap: 6px;
+ color: hsl(var(--bc-fg-muted));
+ }
+
+ .tool-summary-item {
+ white-space: nowrap;
+ }
+
+ .tool-summary-sep {
+ color: hsl(var(--bc-border));
+ }
+
+ .bc-scrollbar {
+ scrollbar-width: thin;
+ scrollbar-color: hsl(var(--bc-border)) transparent;
+ }
+
+ .bc-scrollbar::-webkit-scrollbar {
+ width: 4px;
+ }
+
+ .bc-scrollbar::-webkit-scrollbar-track {
+ background: transparent;
+ }
+
+ .bc-scrollbar::-webkit-scrollbar-thumb {
+ background-color: hsl(var(--bc-border));
+ border-radius: 0;
+ }
+
+ .bc-scrollbar::-webkit-scrollbar-thumb:hover {
+ background-color: hsl(var(--bc-fg-muted));
+ }
+
+ .bc-message-divider {
+ height: 1px;
+ background: hsl(var(--bc-border));
+ opacity: 0.5;
+ margin: 4px 0 8px;
+ }
+
+ @keyframes spin {
+ to {
+ transform: rotate(360deg);
+ }
+ }
+
+ .animate-spin {
+ animation: spin 1s linear infinite;
+ }
+}
+
+@layer utilities {
+ .bc-reveal {
+ animation: bc-enter 220ms ease-out both;
+ animation-delay: var(--delay, 0ms);
+ }
+
+ @media (prefers-reduced-motion: reduce) {
+ .bc-reveal {
+ animation: none;
+ }
+ }
+}
+
+@keyframes bc-enter {
+ from {
+ opacity: 0;
+ transform: translateY(4px);
+ }
+ to {
+ opacity: 1;
+ transform: translateY(0);
+ }
+}
+
+@keyframes bc-sheen {
+ 0%,
+ 100% {
+ transform: translateX(-120%);
+ }
+ 50% {
+ transform: translateX(120%);
+ }
+}
diff --git a/apps/desktop/static/GeistMono[wght].woff2 b/apps/desktop/static/GeistMono[wght].woff2
new file mode 100644
index 00000000..b6234c3f
Binary files /dev/null and b/apps/desktop/static/GeistMono[wght].woff2 differ
diff --git a/apps/desktop/static/Geist[wght].woff2 b/apps/desktop/static/Geist[wght].woff2
new file mode 100644
index 00000000..659d6021
Binary files /dev/null and b/apps/desktop/static/Geist[wght].woff2 differ
diff --git a/apps/desktop/static/favicon.svg b/apps/desktop/static/favicon.svg
new file mode 100644
index 00000000..ea88bdf2
--- /dev/null
+++ b/apps/desktop/static/favicon.svg
@@ -0,0 +1,11 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/apps/desktop/svelte.config.js b/apps/desktop/svelte.config.js
new file mode 100644
index 00000000..ad2e37d6
--- /dev/null
+++ b/apps/desktop/svelte.config.js
@@ -0,0 +1,25 @@
+import adapter from '@sveltejs/adapter-static';
+import { vitePreprocess } from '@sveltejs/vite-plugin-svelte';
+
+/** @type {import('@sveltejs/kit').Config} */
+const config = {
+ preprocess: vitePreprocess(),
+
+ kit: {
+ adapter: adapter({
+ pages: 'build',
+ assets: 'build',
+ fallback: 'index.html'
+ }),
+ env: {
+ dir: '../web'
+ }
+ },
+ compilerOptions: {
+ experimental: {
+ async: true
+ }
+ }
+};
+
+export default config;
diff --git a/apps/desktop/tsconfig.json b/apps/desktop/tsconfig.json
new file mode 100644
index 00000000..feea18bf
--- /dev/null
+++ b/apps/desktop/tsconfig.json
@@ -0,0 +1,15 @@
+{
+ "extends": "./.svelte-kit/tsconfig.json",
+ "compilerOptions": {
+ "rewriteRelativeImportExtensions": true,
+ "allowJs": true,
+ "checkJs": true,
+ "esModuleInterop": true,
+ "forceConsistentCasingInFileNames": true,
+ "resolveJsonModule": true,
+ "skipLibCheck": true,
+ "sourceMap": true,
+ "strict": true,
+ "moduleResolution": "bundler"
+ }
+}
diff --git a/apps/desktop/vite.config.ts b/apps/desktop/vite.config.ts
new file mode 100644
index 00000000..2d35c4f5
--- /dev/null
+++ b/apps/desktop/vite.config.ts
@@ -0,0 +1,7 @@
+import tailwindcss from '@tailwindcss/vite';
+import { sveltekit } from '@sveltejs/kit/vite';
+import { defineConfig } from 'vite';
+
+export default defineConfig({
+ plugins: [tailwindcss(), sveltekit()]
+});
diff --git a/apps/web/README.md b/apps/web/README.md
index 028bf7ba..98c09b80 100644
--- a/apps/web/README.md
+++ b/apps/web/README.md
@@ -1,6 +1,6 @@
# @btca/web
-SvelteKit + Convex web app for btca.
+SvelteKit web app for btca. Convex now lives in `packages/convex`.
## Local development
@@ -17,6 +17,12 @@ From `apps/web` directly:
bun run dev
```
+From `packages/convex` directly:
+
+```sh
+bun run dev
+```
+
## Vercel deployment
1. Import the monorepo into Vercel.
@@ -31,7 +37,7 @@ PUBLIC_POSTHOG_ID=
PUBLIC_ANALYTICS_HOST=
```
-`PUBLIC_CONVEX_URL` is injected during the Vercel build by `convex deploy --cmd-url-env-var-name PUBLIC_CONVEX_URL --cmd 'bun run build'`, so each deployment is built against the matching Convex deployment.
+`PUBLIC_CONVEX_URL` is still injected during the Vercel build by the shared Convex package, so each deployment is built against the matching Convex deployment.
## Env split
diff --git a/apps/web/package.json b/apps/web/package.json
index 3fb896fe..ff777a0e 100644
--- a/apps/web/package.json
+++ b/apps/web/package.json
@@ -4,11 +4,9 @@
"version": "0.0.4",
"type": "module",
"scripts": {
- "dev": "turbo run dev:web dev:convex --parallel",
- "dev:web": "vite dev",
- "dev:convex": "convex dev",
+ "dev": "vite dev",
"build": "vite build",
- "deploy:vercel": "convex deploy --cmd-url-env-var-name PUBLIC_CONVEX_URL --cmd 'bun run build'",
+ "deploy:vercel": "bun run --cwd ../../packages/convex deploy:web",
"preview": "vite preview",
"prepare": "svelte-kit sync || echo ''",
"check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json",
@@ -30,33 +28,25 @@
"svelte": "^5.50.2",
"svelte-check": "^4.4.5",
"tailwindcss": "^4.1.18",
- "turbo": "^2.8.7",
"vite": "^7.3.1"
},
"dependencies": {
- "@ai-sdk/openai-compatible": "^2.0.29",
+ "@btca/convex": "workspace:*",
"@btca/shared": "workspace:*",
- "@clerk/backend": "^2.30.1",
"@clerk/clerk-js": "^5.122.1",
"@clerk/types": "^4.101.14",
- "@convex-dev/migrations": "^0.3.1",
- "@daytonaio/sdk": "^0.149.0",
"@lucide/svelte": "^0.577.0",
"@shikijs/langs": "^3.22.0",
"@shikijs/themes": "^3.22.0",
"@tmcp/adapter-zod": "^0.1.7",
"@tmcp/transport-http": "^0.8.4",
- "ai": "^6.0.81",
- "autumn-js": "^0.1.82",
"better-result": "^2.7.0",
- "btca-sandbox": "workspace:*",
"convex": "^1.31.7",
"convex-svelte": "^0.0.12",
"isomorphic-dompurify": "^2.36.0",
"marked": "^17.0.2",
"nanoid": "^5.1.6",
"posthog-js": "^1.359.1",
- "posthog-node": "^5.24.15",
"shiki": "^3.22.0",
"tmcp": "^1.19.2",
"zod": "^4.3.6"
diff --git a/apps/web/src/convex/_generated/api.d.ts b/apps/web/src/convex/_generated/api.d.ts
deleted file mode 100644
index 3bae7cfd..00000000
--- a/apps/web/src/convex/_generated/api.d.ts
+++ /dev/null
@@ -1,200 +0,0 @@
-/* eslint-disable */
-/**
- * Generated `api` utility.
- *
- * THIS CODE IS AUTOMATICALLY GENERATED.
- *
- * To regenerate, run `npx convex dev`.
- * @module
- */
-
-import type * as analytics from "../analytics.js";
-import type * as analyticsEvents from "../analyticsEvents.js";
-import type * as apiHelpers from "../apiHelpers.js";
-import type * as authHelpers from "../authHelpers.js";
-import type * as clerkApiKeys from "../clerkApiKeys.js";
-import type * as clerkApiKeysQueries from "../clerkApiKeysQueries.js";
-import type * as crons from "../crons.js";
-import type * as githubApp from "../githubApp.js";
-import type * as githubAuth from "../githubAuth.js";
-import type * as githubConnections from "../githubConnections.js";
-import type * as http from "../http.js";
-import type * as instances_actions from "../instances/actions.js";
-import type * as instances_mutations from "../instances/mutations.js";
-import type * as instances_queries from "../instances/queries.js";
-import type * as mcp from "../mcp.js";
-import type * as mcp_resourceContract from "../mcp/resourceContract.js";
-import type * as mcpInternal from "../mcpInternal.js";
-import type * as mcpQuestions from "../mcpQuestions.js";
-import type * as messages from "../messages.js";
-import type * as migrations from "../migrations.js";
-import type * as privateWrappers from "../privateWrappers.js";
-import type * as projects from "../projects.js";
-import type * as resourceActions from "../resourceActions.js";
-import type * as resources from "../resources.js";
-import type * as runtimeConfigLock from "../runtimeConfigLock.js";
-import type * as scheduled_queries from "../scheduled/queries.js";
-import type * as scheduled_updates from "../scheduled/updates.js";
-import type * as scheduled_versionCheck from "../scheduled/versionCheck.js";
-import type * as streamSessions from "../streamSessions.js";
-import type * as threadTitle from "../threadTitle.js";
-import type * as threads from "../threads.js";
-import type * as usage from "../usage.js";
-import type * as users from "../users.js";
-
-import type {
- ApiFromModules,
- FilterApi,
- FunctionReference,
-} from "convex/server";
-
-declare const fullApi: ApiFromModules<{
- analytics: typeof analytics;
- analyticsEvents: typeof analyticsEvents;
- apiHelpers: typeof apiHelpers;
- authHelpers: typeof authHelpers;
- clerkApiKeys: typeof clerkApiKeys;
- clerkApiKeysQueries: typeof clerkApiKeysQueries;
- crons: typeof crons;
- githubApp: typeof githubApp;
- githubAuth: typeof githubAuth;
- githubConnections: typeof githubConnections;
- http: typeof http;
- "instances/actions": typeof instances_actions;
- "instances/mutations": typeof instances_mutations;
- "instances/queries": typeof instances_queries;
- mcp: typeof mcp;
- "mcp/resourceContract": typeof mcp_resourceContract;
- mcpInternal: typeof mcpInternal;
- mcpQuestions: typeof mcpQuestions;
- messages: typeof messages;
- migrations: typeof migrations;
- privateWrappers: typeof privateWrappers;
- projects: typeof projects;
- resourceActions: typeof resourceActions;
- resources: typeof resources;
- runtimeConfigLock: typeof runtimeConfigLock;
- "scheduled/queries": typeof scheduled_queries;
- "scheduled/updates": typeof scheduled_updates;
- "scheduled/versionCheck": typeof scheduled_versionCheck;
- streamSessions: typeof streamSessions;
- threadTitle: typeof threadTitle;
- threads: typeof threads;
- usage: typeof usage;
- users: typeof users;
-}>;
-
-/**
- * A utility for referencing Convex functions in your app's public API.
- *
- * Usage:
- * ```js
- * const myFunctionReference = api.myModule.myFunction;
- * ```
- */
-export declare const api: FilterApi<
- typeof fullApi,
- FunctionReference
->;
-
-/**
- * A utility for referencing Convex functions in your app's internal API.
- *
- * Usage:
- * ```js
- * const myFunctionReference = internal.myModule.myFunction;
- * ```
- */
-export declare const internal: FilterApi<
- typeof fullApi,
- FunctionReference
->;
-
-export declare const components: {
- migrations: {
- lib: {
- cancel: FunctionReference<
- "mutation",
- "internal",
- { name: string },
- {
- batchSize?: number;
- cursor?: string | null;
- error?: string;
- isDone: boolean;
- latestEnd?: number;
- latestStart: number;
- name: string;
- next?: Array;
- processed: number;
- state: "inProgress" | "success" | "failed" | "canceled" | "unknown";
- }
- >;
- cancelAll: FunctionReference<
- "mutation",
- "internal",
- { sinceTs?: number },
- Array<{
- batchSize?: number;
- cursor?: string | null;
- error?: string;
- isDone: boolean;
- latestEnd?: number;
- latestStart: number;
- name: string;
- next?: Array;
- processed: number;
- state: "inProgress" | "success" | "failed" | "canceled" | "unknown";
- }>
- >;
- clearAll: FunctionReference<
- "mutation",
- "internal",
- { before?: number },
- null
- >;
- getStatus: FunctionReference<
- "query",
- "internal",
- { limit?: number; names?: Array },
- Array<{
- batchSize?: number;
- cursor?: string | null;
- error?: string;
- isDone: boolean;
- latestEnd?: number;
- latestStart: number;
- name: string;
- next?: Array;
- processed: number;
- state: "inProgress" | "success" | "failed" | "canceled" | "unknown";
- }>
- >;
- migrate: FunctionReference<
- "mutation",
- "internal",
- {
- batchSize?: number;
- cursor?: string | null;
- dryRun: boolean;
- fnHandle: string;
- name: string;
- next?: Array<{ fnHandle: string; name: string }>;
- oneBatchOnly?: boolean;
- },
- {
- batchSize?: number;
- cursor?: string | null;
- error?: string;
- isDone: boolean;
- latestEnd?: number;
- latestStart: number;
- name: string;
- next?: Array;
- processed: number;
- state: "inProgress" | "success" | "failed" | "canceled" | "unknown";
- }
- >;
- };
- };
-};
diff --git a/apps/web/src/lib/billing/plans.ts b/apps/web/src/lib/billing/plans.ts
index f4c708d8..a9213ac1 100644
--- a/apps/web/src/lib/billing/plans.ts
+++ b/apps/web/src/lib/billing/plans.ts
@@ -1,5 +1,5 @@
-import { PRO_AI_BUDGET_MICROS, PRO_AI_BUDGET_USD } from './aiBudget.ts';
-import { WEB_SANDBOX_MODELS } from '../models/webSandboxModels.ts';
+import { PRO_AI_BUDGET_MICROS, PRO_AI_BUDGET_USD } from '@btca/convex/aiBudget';
+import { WEB_SANDBOX_MODELS } from '@btca/convex/webSandboxModels';
export const BILLING_PLAN = {
id: 'btca_pro',
diff --git a/apps/web/src/lib/components/AddResourceModal.svelte b/apps/web/src/lib/components/AddResourceModal.svelte
index 1ee85e7e..cda66755 100644
--- a/apps/web/src/lib/components/AddResourceModal.svelte
+++ b/apps/web/src/lib/components/AddResourceModal.svelte
@@ -8,7 +8,7 @@
X
} from '@lucide/svelte';
import { useConvexClient } from 'convex-svelte';
- import { api } from '../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
import { getAuthState } from '$lib/stores/auth.svelte';
import { getProjectStore } from '$lib/stores/project.svelte';
diff --git a/apps/web/src/lib/components/InstanceCard.svelte b/apps/web/src/lib/components/InstanceCard.svelte
index e7e36561..4999a11f 100644
--- a/apps/web/src/lib/components/InstanceCard.svelte
+++ b/apps/web/src/lib/components/InstanceCard.svelte
@@ -12,7 +12,7 @@
Square
} from '@lucide/svelte';
import { getInstanceStore } from '$lib/stores/instance.svelte';
- import { INSTANCE_DISK_FULL_MESSAGE } from '$lib/instanceErrors';
+ import { INSTANCE_DISK_FULL_MESSAGE } from '@btca/convex/instanceErrors';
type InstanceAction = 'wake' | 'stop' | 'update' | 'reset';
const instanceStore = getInstanceStore();
diff --git a/apps/web/src/lib/components/InstanceStatus.svelte b/apps/web/src/lib/components/InstanceStatus.svelte
index 01c7b42c..f534894c 100644
--- a/apps/web/src/lib/components/InstanceStatus.svelte
+++ b/apps/web/src/lib/components/InstanceStatus.svelte
@@ -10,7 +10,7 @@
Square
} from '@lucide/svelte';
import { getInstanceStore } from '$lib/stores/instance.svelte';
- import { INSTANCE_DISK_FULL_MESSAGE } from '$lib/instanceErrors';
+ import { INSTANCE_DISK_FULL_MESSAGE } from '@btca/convex/instanceErrors';
type InstanceAction = 'wake' | 'stop' | 'update' | 'reset';
diff --git a/apps/web/src/lib/components/ProjectModelPicker.svelte b/apps/web/src/lib/components/ProjectModelPicker.svelte
index 0cfb9d86..0c41c536 100644
--- a/apps/web/src/lib/components/ProjectModelPicker.svelte
+++ b/apps/web/src/lib/components/ProjectModelPicker.svelte
@@ -2,8 +2,8 @@
import { ChevronDown, Loader2, Check } from '@lucide/svelte';
import { useQuery } from 'convex-svelte';
- import { api } from '../../convex/_generated/api';
- import { WEB_SANDBOX_MODELS, getWebSandboxModel } from '../models/webSandboxModels.ts';
+ import { api } from '@btca/convex/api';
+ import { WEB_SANDBOX_MODELS, getWebSandboxModel } from '@btca/convex/webSandboxModels';
import { getInstanceStore } from '../stores/instance.svelte';
import { getProjectStore } from '../stores/project.svelte';
diff --git a/apps/web/src/lib/components/Sidebar.svelte b/apps/web/src/lib/components/Sidebar.svelte
index 97a3ed7d..2db5047b 100644
--- a/apps/web/src/lib/components/Sidebar.svelte
+++ b/apps/web/src/lib/components/Sidebar.svelte
@@ -19,8 +19,8 @@
import { goto } from '$app/navigation';
import { createEventDispatcher, onDestroy } from 'svelte';
import { useConvexClient } from 'convex-svelte';
- import { api } from '../../convex/_generated/api';
- import type { Id } from '../../convex/_generated/dataModel';
+ import { api } from '@btca/convex/api';
+ import type { Id } from '@btca/convex/dataModel';
import { getAuthState, openSignIn, signOut } from '$lib/stores/auth.svelte';
import { getThemeStore } from '$lib/stores/theme.svelte';
import { getProjectStore } from '$lib/stores/project.svelte';
diff --git a/apps/web/src/lib/result/http.ts b/apps/web/src/lib/result/http.ts
index 78ccfad9..468745c7 100644
--- a/apps/web/src/lib/result/http.ts
+++ b/apps/web/src/lib/result/http.ts
@@ -1,6 +1,6 @@
import { Result } from 'better-result';
-import { toWebError, type WebError } from './errors';
+import { toWebError, type WebError } from '@btca/convex/errors';
type ConvexResultError = WebError | Error | string;
diff --git a/apps/web/src/lib/stores/ShikiStore.svelte.ts b/apps/web/src/lib/stores/ShikiStore.svelte.ts
index c0d36734..430e3eb2 100644
--- a/apps/web/src/lib/stores/ShikiStore.svelte.ts
+++ b/apps/web/src/lib/stores/ShikiStore.svelte.ts
@@ -7,7 +7,7 @@ import json from '@shikijs/langs/json';
import toml from '@shikijs/langs/toml';
import darkPlus from '@shikijs/themes/dark-plus';
import lightPlus from '@shikijs/themes/light-plus';
-import { WebValidationError } from '../result/errors';
+import { WebValidationError } from '@btca/convex/errors';
let highlighterPromise: Promise | null = null;
let highlighterInstance: HighlighterCore | null = null;
diff --git a/apps/web/src/lib/stores/auth.svelte.ts b/apps/web/src/lib/stores/auth.svelte.ts
index 5991b1a2..f1408eb0 100644
--- a/apps/web/src/lib/stores/auth.svelte.ts
+++ b/apps/web/src/lib/stores/auth.svelte.ts
@@ -1,6 +1,6 @@
import type { Clerk } from '@clerk/clerk-js';
import type { OAuthScope, SessionResource, UserResource } from '@clerk/types';
-import type { Id } from '../../convex/_generated/dataModel';
+import type { Id } from '@btca/convex/dataModel';
// Auth state
let clerk = $state(null);
diff --git a/apps/web/src/lib/stores/billing.svelte.ts b/apps/web/src/lib/stores/billing.svelte.ts
index a59e8257..349719f9 100644
--- a/apps/web/src/lib/stores/billing.svelte.ts
+++ b/apps/web/src/lib/stores/billing.svelte.ts
@@ -1,10 +1,10 @@
import { Result } from 'better-result';
import { createContext } from 'svelte';
import { useConvexClient } from 'convex-svelte';
-import { api } from '../../convex/_generated/api';
-import type { Id } from '../../convex/_generated/dataModel';
+import { api } from '@btca/convex/api';
+import type { Id } from '@btca/convex/dataModel';
import type { BillingSummary } from '$lib/billing/types';
-import { WebValidationError } from '../result/errors';
+import { WebValidationError } from '@btca/convex/errors';
class BillingStore {
private _client = useConvexClient();
diff --git a/apps/web/src/lib/stores/instance.svelte.ts b/apps/web/src/lib/stores/instance.svelte.ts
index ac6c340a..17eadbfc 100644
--- a/apps/web/src/lib/stores/instance.svelte.ts
+++ b/apps/web/src/lib/stores/instance.svelte.ts
@@ -1,10 +1,10 @@
import { Result } from 'better-result';
import { createContext } from 'svelte';
import { useQuery, useConvexClient } from 'convex-svelte';
-import { instances } from '../../convex/apiHelpers';
-import type { Doc, Id } from '../../convex/_generated/dataModel';
+import { instances } from '@btca/convex/apiHelpers';
+import type { Doc, Id } from '@btca/convex/dataModel';
import { trackEvent, ClientAnalyticsEvents } from './analytics.svelte';
-import { WebValidationError } from '../result/errors';
+import { WebValidationError } from '@btca/convex/errors';
type InstanceStatus = {
instance: Doc<'instances'>;
diff --git a/apps/web/src/lib/stores/project.svelte.ts b/apps/web/src/lib/stores/project.svelte.ts
index 14bd20c2..cee3b0f0 100644
--- a/apps/web/src/lib/stores/project.svelte.ts
+++ b/apps/web/src/lib/stores/project.svelte.ts
@@ -2,9 +2,9 @@ import { Result } from 'better-result';
import { getContext, setContext } from 'svelte';
import { useQuery, useConvexClient } from 'convex-svelte';
import { goto } from '$app/navigation';
-import { api } from '../../convex/_generated/api';
-import type { Id } from '../../convex/_generated/dataModel';
-import { WebValidationError } from '../result/errors';
+import { api } from '@btca/convex/api';
+import type { Id } from '@btca/convex/dataModel';
+import { WebValidationError } from '@btca/convex/errors';
type Project = {
_id: Id<'projects'>;
diff --git a/apps/web/src/lib/stores/theme.svelte.ts b/apps/web/src/lib/stores/theme.svelte.ts
index 3e428d80..4ecc33fa 100644
--- a/apps/web/src/lib/stores/theme.svelte.ts
+++ b/apps/web/src/lib/stores/theme.svelte.ts
@@ -1,7 +1,7 @@
import { Result } from 'better-result';
import { getContext, setContext } from 'svelte';
import { browser } from '$app/environment';
-import { WebValidationError } from '../result/errors';
+import { WebValidationError } from '@btca/convex/errors';
const THEME_KEY = Symbol('theme');
diff --git a/apps/web/src/lib/stores/threadPreload.svelte.ts b/apps/web/src/lib/stores/threadPreload.svelte.ts
index c0dbb59e..58baea7d 100644
--- a/apps/web/src/lib/stores/threadPreload.svelte.ts
+++ b/apps/web/src/lib/stores/threadPreload.svelte.ts
@@ -1,4 +1,4 @@
-import type { Doc, Id } from '../../convex/_generated/dataModel';
+import type { Doc, Id } from '@btca/convex/dataModel';
export type PreloadedThreadWithMessages =
| ((Doc<'threads'> & {
diff --git a/apps/web/src/routes/api/mcp/+server.ts b/apps/web/src/routes/api/mcp/+server.ts
index 9ad5c5c1..d90e3c46 100644
--- a/apps/web/src/routes/api/mcp/+server.ts
+++ b/apps/web/src/routes/api/mcp/+server.ts
@@ -4,7 +4,7 @@ import { ZodJsonSchemaAdapter } from '@tmcp/adapter-zod';
import { ConvexHttpClient } from 'convex/browser';
import { z } from 'zod';
import { PUBLIC_CONVEX_URL } from '$env/static/public';
-import { api } from '../../../convex/_generated/api';
+import { api } from '@btca/convex/api';
import type { RequestHandler } from './$types';
import { extractApiKey, jsonError } from '../../../lib/result/http';
diff --git a/apps/web/src/routes/app/+layout.svelte b/apps/web/src/routes/app/+layout.svelte
index 502201fa..a4a1dcaf 100644
--- a/apps/web/src/routes/app/+layout.svelte
+++ b/apps/web/src/routes/app/+layout.svelte
@@ -15,7 +15,7 @@
trackEvent,
ClientAnalyticsEvents
} from '$lib/stores/analytics.svelte';
- import { api } from '../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
import Sidebar from '$lib/components/Sidebar.svelte';
import ProvisioningModal from '$lib/components/ProvisioningModal.svelte';
import CreateProjectModal from '$lib/components/CreateProjectModal.svelte';
diff --git a/apps/web/src/routes/app/chat/[id]/+page.svelte b/apps/web/src/routes/app/chat/[id]/+page.svelte
index 9aa3b782..0e0e2dd0 100644
--- a/apps/web/src/routes/app/chat/[id]/+page.svelte
+++ b/apps/web/src/routes/app/chat/[id]/+page.svelte
@@ -14,10 +14,10 @@
import { threadPreloadStore } from '$lib/stores/threadPreload.svelte';
import { trackEvent, ClientAnalyticsEvents } from '$lib/stores/analytics.svelte';
import { SUPPORT_URL } from '$lib/billing/plans';
- import { INSTANCE_DISK_FULL_MESSAGE } from '$lib/instanceErrors';
+ import { INSTANCE_DISK_FULL_MESSAGE } from '@btca/convex/instanceErrors';
import type { BtcaChunk, CancelState, MessageStats } from '$lib/types';
- import { api } from '../../../../convex/_generated/api';
- import type { Id } from '../../../../convex/_generated/dataModel';
+ import { api } from '@btca/convex/api';
+ import type { Id } from '@btca/convex/dataModel';
type ThreadMessageRecord = {
_id: Id<'messages'>;
diff --git a/apps/web/src/routes/app/settings/+page.svelte b/apps/web/src/routes/app/settings/+page.svelte
index ab098ccf..5506304d 100644
--- a/apps/web/src/routes/app/settings/+page.svelte
+++ b/apps/web/src/routes/app/settings/+page.svelte
@@ -13,7 +13,7 @@
} from '@lucide/svelte';
import { goto } from '$app/navigation';
import { page } from '$app/state';
- import { api } from '../../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
import { getAuthState, openUserProfile } from '$lib/stores/auth.svelte';
import { getBillingStore } from '$lib/stores/billing.svelte';
import { getProjectStore } from '$lib/stores/project.svelte';
diff --git a/apps/web/src/routes/app/settings/billing/+page.svelte b/apps/web/src/routes/app/settings/billing/+page.svelte
index 5bcabdb8..24fe17ec 100644
--- a/apps/web/src/routes/app/settings/billing/+page.svelte
+++ b/apps/web/src/routes/app/settings/billing/+page.svelte
@@ -7,7 +7,7 @@
import { getBillingStore } from '$lib/stores/billing.svelte';
import { BILLING_PLAN } from '$lib/billing/plans';
import PricingPlans from '$lib/components/pricing/PricingPlans.svelte';
- import { api } from '../../../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
const auth = getAuthState();
const billingStore = getBillingStore();
diff --git a/apps/web/src/routes/app/settings/questions/+page.svelte b/apps/web/src/routes/app/settings/questions/+page.svelte
index a821a8bd..8ed5e444 100644
--- a/apps/web/src/routes/app/settings/questions/+page.svelte
+++ b/apps/web/src/routes/app/settings/questions/+page.svelte
@@ -19,7 +19,7 @@
import { marked } from 'marked';
import DOMPurify from 'isomorphic-dompurify';
import { goto } from '$app/navigation';
- import { api } from '../../../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
import { getProjectStore } from '$lib/stores/project.svelte';
import Dropdown from '$lib/components/Dropdown.svelte';
diff --git a/apps/web/src/routes/app/settings/resources/+page.svelte b/apps/web/src/routes/app/settings/resources/+page.svelte
index b42cb7ca..f6ebb30a 100644
--- a/apps/web/src/routes/app/settings/resources/+page.svelte
+++ b/apps/web/src/routes/app/settings/resources/+page.svelte
@@ -22,7 +22,7 @@
import ResourceLogo from '$lib/components/ResourceLogo.svelte';
import { getAuthState } from '$lib/stores/auth.svelte';
import { getProjectStore } from '$lib/stores/project.svelte';
- import { api } from '../../../../convex/_generated/api';
+ import { api } from '@btca/convex/api';
const auth = getAuthState();
const client = useConvexClient();
diff --git a/apps/web/turbo.json b/apps/web/turbo.json
deleted file mode 100644
index 4a618a35..00000000
--- a/apps/web/turbo.json
+++ /dev/null
@@ -1,14 +0,0 @@
-{
- "$schema": "https://turbo.build/schema.json",
- "extends": ["//"],
- "tasks": {
- "dev:web": {
- "cache": false,
- "persistent": true
- },
- "dev:convex": {
- "cache": false,
- "persistent": true
- }
- }
-}
diff --git a/bun.lock b/bun.lock
index 2627903a..03521194 100644
--- a/bun.lock
+++ b/bun.lock
@@ -49,6 +49,44 @@
"zod": "^4.3.6",
},
},
+ "apps/desktop": {
+ "name": "@btca/desktop",
+ "version": "0.0.1",
+ "dependencies": {
+ "@btca/convex": "workspace:*",
+ "@btca/shared": "workspace:*",
+ "@clerk/clerk-js": "^5.122.1",
+ "@clerk/types": "^4.101.14",
+ "@lucide/svelte": "^0.562.0",
+ "@shikijs/langs": "^3.22.0",
+ "@shikijs/themes": "^3.22.0",
+ "better-result": "^2.7.0",
+ "convex": "^1.31.7",
+ "convex-svelte": "^0.0.12",
+ "isomorphic-dompurify": "^2.36.0",
+ "marked": "^17.0.2",
+ "nanoid": "^5.1.6",
+ "posthog-js": "^1.345.5",
+ "shiki": "^3.22.0",
+ "zod": "^4.3.6",
+ },
+ "devDependencies": {
+ "@sveltejs/adapter-static": "^3.0.8",
+ "@sveltejs/kit": "^2.50.2",
+ "@sveltejs/vite-plugin-svelte": "^6.2.4",
+ "@tailwindcss/typography": "^0.5.19",
+ "@tailwindcss/vite": "^4.1.18",
+ "@tauri-apps/cli": "^2.10.1",
+ "prettier": "^3.8.1",
+ "prettier-plugin-svelte": "^3.4.1",
+ "prettier-plugin-tailwindcss": "^0.7.2",
+ "svelte": "^5.50.2",
+ "svelte-check": "^4.3.6",
+ "tailwindcss": "^4.1.18",
+ "typescript": "^5.9.3",
+ "vite": "^7.3.1",
+ },
+ },
"apps/docs": {
"name": "@btca/docs",
"devDependencies": {
@@ -94,29 +132,22 @@
"name": "@btca/web",
"version": "0.0.4",
"dependencies": {
- "@ai-sdk/openai-compatible": "^2.0.29",
+ "@btca/convex": "workspace:*",
"@btca/shared": "workspace:*",
- "@clerk/backend": "^2.30.1",
"@clerk/clerk-js": "^5.122.1",
"@clerk/types": "^4.101.14",
- "@convex-dev/migrations": "^0.3.1",
- "@daytonaio/sdk": "^0.149.0",
"@lucide/svelte": "^0.577.0",
"@shikijs/langs": "^3.22.0",
"@shikijs/themes": "^3.22.0",
"@tmcp/adapter-zod": "^0.1.7",
"@tmcp/transport-http": "^0.8.4",
- "ai": "^6.0.81",
- "autumn-js": "^0.1.82",
"better-result": "^2.7.0",
- "btca-sandbox": "workspace:*",
"convex": "^1.31.7",
"convex-svelte": "^0.0.12",
"isomorphic-dompurify": "^2.36.0",
"marked": "^17.0.2",
"nanoid": "^5.1.6",
"posthog-js": "^1.359.1",
- "posthog-node": "^5.24.15",
"shiki": "^3.22.0",
"tmcp": "^1.19.2",
"zod": "^4.3.6",
@@ -135,10 +166,33 @@
"svelte": "^5.50.2",
"svelte-check": "^4.4.5",
"tailwindcss": "^4.1.18",
- "turbo": "^2.8.7",
"vite": "^7.3.1",
},
},
+ "packages/convex": {
+ "name": "@btca/convex",
+ "version": "0.0.1",
+ "dependencies": {
+ "@ai-sdk/openai-compatible": "^2.0.29",
+ "@btca/shared": "workspace:*",
+ "@clerk/backend": "^2.30.1",
+ "@convex-dev/migrations": "^0.3.1",
+ "@daytonaio/sdk": "^0.149.0",
+ "ai": "^6.0.81",
+ "autumn-js": "^0.1.82",
+ "better-result": "^2.7.0",
+ "btca-sandbox": "workspace:*",
+ "convex": "^1.31.7",
+ "nanoid": "^5.1.6",
+ "posthog-node": "^5.24.15",
+ "zod": "^4.3.6",
+ },
+ "devDependencies": {
+ "@types/bun": "^1.3.10",
+ "prettier": "^3.8.1",
+ "typescript": "^5.9.3",
+ },
+ },
"packages/shared": {
"name": "@btca/shared",
"version": "0.0.1",
@@ -190,71 +244,71 @@
"@aws-crypto/util": ["@aws-crypto/util@5.2.0", "", { "dependencies": { "@aws-sdk/types": "^3.222.0", "@smithy/util-utf8": "^2.0.0", "tslib": "^2.6.2" } }, "sha512-4RkU9EsI6ZpBve5fseQlGNUWKMa1RLPQ1dnjnQoe07ldfIzcsGb5hC5W0Dm7u423KWzawlrpbjXBrXCEv9zazQ=="],
- "@aws-sdk/client-s3": ["@aws-sdk/client-s3@3.1002.0", "", { "dependencies": { "@aws-crypto/sha1-browser": "5.2.0", "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.973.17", "@aws-sdk/credential-provider-node": "^3.972.16", "@aws-sdk/middleware-bucket-endpoint": "^3.972.6", "@aws-sdk/middleware-expect-continue": "^3.972.6", "@aws-sdk/middleware-flexible-checksums": "^3.973.3", "@aws-sdk/middleware-host-header": "^3.972.6", "@aws-sdk/middleware-location-constraint": "^3.972.6", "@aws-sdk/middleware-logger": "^3.972.6", "@aws-sdk/middleware-recursion-detection": "^3.972.6", "@aws-sdk/middleware-sdk-s3": "^3.972.17", "@aws-sdk/middleware-ssec": "^3.972.6", "@aws-sdk/middleware-user-agent": "^3.972.17", "@aws-sdk/region-config-resolver": "^3.972.6", "@aws-sdk/signature-v4-multi-region": "^3.996.5", "@aws-sdk/types": "^3.973.4", "@aws-sdk/util-endpoints": "^3.996.3", "@aws-sdk/util-user-agent-browser": "^3.972.6", "@aws-sdk/util-user-agent-node": "^3.973.2", "@smithy/config-resolver": "^4.4.9", "@smithy/core": "^3.23.7", "@smithy/eventstream-serde-browser": "^4.2.10", "@smithy/eventstream-serde-config-resolver": "^4.3.10", "@smithy/eventstream-serde-node": "^4.2.10", "@smithy/fetch-http-handler": "^5.3.12", "@smithy/hash-blob-browser": "^4.2.11", "@smithy/hash-node": "^4.2.10", "@smithy/hash-stream-node": "^4.2.10", "@smithy/invalid-dependency": "^4.2.10", "@smithy/md5-js": "^4.2.10", "@smithy/middleware-content-length": "^4.2.10", "@smithy/middleware-endpoint": "^4.4.21", "@smithy/middleware-retry": "^4.4.38", "@smithy/middleware-serde": "^4.2.11", "@smithy/middleware-stack": "^4.2.10", "@smithy/node-config-provider": "^4.3.10", "@smithy/node-http-handler": "^4.4.13", "@smithy/protocol-http": "^5.3.10", "@smithy/smithy-client": "^4.12.1", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.10", "@smithy/util-base64": "^4.3.1", "@smithy/util-body-length-browser": "^4.2.1", "@smithy/util-body-length-node": "^4.2.2", "@smithy/util-defaults-mode-browser": "^4.3.37", "@smithy/util-defaults-mode-node": "^4.2.40", "@smithy/util-endpoints": "^3.3.1", "@smithy/util-middleware": "^4.2.10", "@smithy/util-retry": "^4.2.10", "@smithy/util-stream": "^4.5.16", "@smithy/util-utf8": "^4.2.1", "@smithy/util-waiter": "^4.2.10", "tslib": "^2.6.2" } }, "sha512-tc+vZgvjcm+1Ot+YhQjXZxVELKGGGO3D5cuR4p5xaeitXYX2+RRiz4/WdSak9slumIClnlXsdqhJ0OHognUT+w=="],
+ "@aws-sdk/client-s3": ["@aws-sdk/client-s3@3.1004.0", "", { "dependencies": { "@aws-crypto/sha1-browser": "5.2.0", "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.973.18", "@aws-sdk/credential-provider-node": "^3.972.18", "@aws-sdk/middleware-bucket-endpoint": "^3.972.7", "@aws-sdk/middleware-expect-continue": "^3.972.7", "@aws-sdk/middleware-flexible-checksums": "^3.973.4", "@aws-sdk/middleware-host-header": "^3.972.7", "@aws-sdk/middleware-location-constraint": "^3.972.7", "@aws-sdk/middleware-logger": "^3.972.7", "@aws-sdk/middleware-recursion-detection": "^3.972.7", "@aws-sdk/middleware-sdk-s3": "^3.972.18", "@aws-sdk/middleware-ssec": "^3.972.7", "@aws-sdk/middleware-user-agent": "^3.972.19", "@aws-sdk/region-config-resolver": "^3.972.7", "@aws-sdk/signature-v4-multi-region": "^3.996.6", "@aws-sdk/types": "^3.973.5", "@aws-sdk/util-endpoints": "^3.996.4", "@aws-sdk/util-user-agent-browser": "^3.972.7", "@aws-sdk/util-user-agent-node": "^3.973.4", "@smithy/config-resolver": "^4.4.10", "@smithy/core": "^3.23.8", "@smithy/eventstream-serde-browser": "^4.2.11", "@smithy/eventstream-serde-config-resolver": "^4.3.11", "@smithy/eventstream-serde-node": "^4.2.11", "@smithy/fetch-http-handler": "^5.3.13", "@smithy/hash-blob-browser": "^4.2.12", "@smithy/hash-node": "^4.2.11", "@smithy/hash-stream-node": "^4.2.11", "@smithy/invalid-dependency": "^4.2.11", "@smithy/md5-js": "^4.2.11", "@smithy/middleware-content-length": "^4.2.11", "@smithy/middleware-endpoint": "^4.4.22", "@smithy/middleware-retry": "^4.4.39", "@smithy/middleware-serde": "^4.2.12", "@smithy/middleware-stack": "^4.2.11", "@smithy/node-config-provider": "^4.3.11", "@smithy/node-http-handler": "^4.4.14", "@smithy/protocol-http": "^5.3.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-body-length-node": "^4.2.3", "@smithy/util-defaults-mode-browser": "^4.3.38", "@smithy/util-defaults-mode-node": "^4.2.41", "@smithy/util-endpoints": "^3.3.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-retry": "^4.2.11", "@smithy/util-stream": "^4.5.17", "@smithy/util-utf8": "^4.2.2", "@smithy/util-waiter": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-m0zNfpsona9jQdX1cHtHArOiuvSGZPsgp/KRZS2YjJhKah96G2UN3UNGZQ6aVjXIQjCY6UanCJo0uW9Xf2U41w=="],
- "@aws-sdk/core": ["@aws-sdk/core@3.973.17", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@aws-sdk/xml-builder": "^3.972.9", "@smithy/core": "^3.23.7", "@smithy/node-config-provider": "^4.3.10", "@smithy/property-provider": "^4.2.10", "@smithy/protocol-http": "^5.3.10", "@smithy/signature-v4": "^5.3.10", "@smithy/smithy-client": "^4.12.1", "@smithy/types": "^4.13.0", "@smithy/util-base64": "^4.3.1", "@smithy/util-middleware": "^4.2.10", "@smithy/util-utf8": "^4.2.1", "tslib": "^2.6.2" } }, "sha512-VtgGP0TjbCeyp6DQpiBqJKbemTSIaN2bZc3UbeTDCani3lBCyxn75ouJYD6koSSp0bh7rKLEbUpiFsNCI7tr0w=="],
+ "@aws-sdk/core": ["@aws-sdk/core@3.973.18", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@aws-sdk/xml-builder": "^3.972.10", "@smithy/core": "^3.23.8", "@smithy/node-config-provider": "^4.3.11", "@smithy/property-provider": "^4.2.11", "@smithy/protocol-http": "^5.3.11", "@smithy/signature-v4": "^5.3.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/util-base64": "^4.3.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-GUIlegfcK2LO1J2Y98sCJy63rQSiLiDOgVw7HiHPRqfI2vb3XozTVqemwO0VSGXp54ngCnAQz0Lf0YPCBINNxA=="],
- "@aws-sdk/crc64-nvme": ["@aws-sdk/crc64-nvme@3.972.3", "", { "dependencies": { "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-UExeK+EFiq5LAcbHm96CQLSia+5pvpUVSAsVApscBzayb7/6dJBJKwV4/onsk4VbWSmqxDMcfuTD+pC4RxgZHg=="],
+ "@aws-sdk/crc64-nvme": ["@aws-sdk/crc64-nvme@3.972.4", "", { "dependencies": { "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-HKZIZLbRyvzo/bXZU7Zmk6XqU+1C9DjI56xd02vwuDIxedxBEqP17t9ExhbP9QFeNq/a3l9GOcyirFXxmbDhmw=="],
- "@aws-sdk/credential-provider-env": ["@aws-sdk/credential-provider-env@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-RhHQG1lhkWHL4tK1C/KDjaOeis+9U0tAMnWDiwiSVQZMC7CsST9Xin+sK89XywJ5g/tyABtb7TvFePJ4Te5XSQ=="],
+ "@aws-sdk/credential-provider-env": ["@aws-sdk/credential-provider-env@3.972.16", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-HrdtnadvTGAQUr18sPzGlE5El3ICphnH6SU7UQOMOWFgRKbTRNN8msTxM4emzguUso9CzaHU2xy5ctSrmK5YNA=="],
- "@aws-sdk/credential-provider-http": ["@aws-sdk/credential-provider-http@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/types": "^3.973.4", "@smithy/fetch-http-handler": "^5.3.12", "@smithy/node-http-handler": "^4.4.13", "@smithy/property-provider": "^4.2.10", "@smithy/protocol-http": "^5.3.10", "@smithy/smithy-client": "^4.12.1", "@smithy/types": "^4.13.0", "@smithy/util-stream": "^4.5.16", "tslib": "^2.6.2" } }, "sha512-b/bDL76p51+yQ+0O9ZDH5nw/ioE0sRYkjwjOwFWAWZXo6it2kQZUOXhVpjohx3ldKyUxt/SwAivjUu1Nr/PWlQ=="],
+ "@aws-sdk/credential-provider-http": ["@aws-sdk/credential-provider-http@3.972.18", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/types": "^3.973.5", "@smithy/fetch-http-handler": "^5.3.13", "@smithy/node-http-handler": "^4.4.14", "@smithy/property-provider": "^4.2.11", "@smithy/protocol-http": "^5.3.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/util-stream": "^4.5.17", "tslib": "^2.6.2" } }, "sha512-NyB6smuZAixND5jZumkpkunQ0voc4Mwgkd+SZ6cvAzIB7gK8HV8Zd4rS8Kn5MmoGgusyNfVGG+RLoYc4yFiw+A=="],
- "@aws-sdk/credential-provider-ini": ["@aws-sdk/credential-provider-ini@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/credential-provider-env": "^3.972.15", "@aws-sdk/credential-provider-http": "^3.972.17", "@aws-sdk/credential-provider-login": "^3.972.15", "@aws-sdk/credential-provider-process": "^3.972.15", "@aws-sdk/credential-provider-sso": "^3.972.15", "@aws-sdk/credential-provider-web-identity": "^3.972.15", "@aws-sdk/nested-clients": "^3.996.5", "@aws-sdk/types": "^3.973.4", "@smithy/credential-provider-imds": "^4.2.10", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-qWnM+wB8MmU2kKY7f4KowKjOjkwRosaFxrtseEEIefwoXn1SjN+CbHzXBVdTAQxxkbBiqhPgJ/WHiPtES4grRQ=="],
+ "@aws-sdk/credential-provider-ini": ["@aws-sdk/credential-provider-ini@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/credential-provider-env": "^3.972.16", "@aws-sdk/credential-provider-http": "^3.972.18", "@aws-sdk/credential-provider-login": "^3.972.17", "@aws-sdk/credential-provider-process": "^3.972.16", "@aws-sdk/credential-provider-sso": "^3.972.17", "@aws-sdk/credential-provider-web-identity": "^3.972.17", "@aws-sdk/nested-clients": "^3.996.7", "@aws-sdk/types": "^3.973.5", "@smithy/credential-provider-imds": "^4.2.11", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-dFqh7nfX43B8dO1aPQHOcjC0SnCJ83H3F+1LoCh3X1P7E7N09I+0/taID0asU6GCddfDExqnEvQtDdkuMe5tKQ=="],
- "@aws-sdk/credential-provider-login": ["@aws-sdk/credential-provider-login@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/nested-clients": "^3.996.5", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/protocol-http": "^5.3.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-x92FJy34/95wgu+qOGD8SHcgh1hZ9Qx2uFtQEGn4m9Ljou8ICIv3Ybq5yxdB7A60S8ZGCQB0mIopmjJwiLbh5g=="],
+ "@aws-sdk/credential-provider-login": ["@aws-sdk/credential-provider-login@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/nested-clients": "^3.996.7", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/protocol-http": "^5.3.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-gf2E5b7LpKb+JX2oQsRIDxdRZjBFZt2olCGlWCdb3vBERbXIPgm2t1R5mEnwd4j0UEO/Tbg5zN2KJbHXttJqwA=="],
- "@aws-sdk/credential-provider-node": ["@aws-sdk/credential-provider-node@3.972.16", "", { "dependencies": { "@aws-sdk/credential-provider-env": "^3.972.15", "@aws-sdk/credential-provider-http": "^3.972.17", "@aws-sdk/credential-provider-ini": "^3.972.15", "@aws-sdk/credential-provider-process": "^3.972.15", "@aws-sdk/credential-provider-sso": "^3.972.15", "@aws-sdk/credential-provider-web-identity": "^3.972.15", "@aws-sdk/types": "^3.973.4", "@smithy/credential-provider-imds": "^4.2.10", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-7mlt14Ee4rPFAFUVgpWE7+0CBhetJJyzVFqfIsMp7sgyOSm9Y/+qHZOWAuK5I4JNc+Y5PltvJ9kssTzRo92iXQ=="],
+ "@aws-sdk/credential-provider-node": ["@aws-sdk/credential-provider-node@3.972.18", "", { "dependencies": { "@aws-sdk/credential-provider-env": "^3.972.16", "@aws-sdk/credential-provider-http": "^3.972.18", "@aws-sdk/credential-provider-ini": "^3.972.17", "@aws-sdk/credential-provider-process": "^3.972.16", "@aws-sdk/credential-provider-sso": "^3.972.17", "@aws-sdk/credential-provider-web-identity": "^3.972.17", "@aws-sdk/types": "^3.973.5", "@smithy/credential-provider-imds": "^4.2.11", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-ZDJa2gd1xiPg/nBDGhUlat02O8obaDEnICBAVS8qieZ0+nDfaB0Z3ec6gjZj27OqFTjnB/Q5a0GwQwb7rMVViw=="],
- "@aws-sdk/credential-provider-process": ["@aws-sdk/credential-provider-process@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-PrH3iTeD18y/8uJvQD2s/T87BTGhsdS/1KZU7ReWHXsplBwvCqi7AbnnNbML1pFlQwRWCE2RdSZFWDVId3CvkA=="],
+ "@aws-sdk/credential-provider-process": ["@aws-sdk/credential-provider-process@3.972.16", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-n89ibATwnLEg0ZdZmUds5bq8AfBAdoYEDpqP3uzPLaRuGelsKlIvCYSNNvfgGLi8NaHPNNhs1HjJZYbqkW9b+g=="],
- "@aws-sdk/credential-provider-sso": ["@aws-sdk/credential-provider-sso@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/nested-clients": "^3.996.5", "@aws-sdk/token-providers": "3.1002.0", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-M/+LBHTPKZxxXckM6m4dnJeR+jlm9NynH9b2YDswN4Zj2St05SK/crdL3Wy3WfJTZootnnhm3oTh87Usl7PS7w=="],
+ "@aws-sdk/credential-provider-sso": ["@aws-sdk/credential-provider-sso@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/nested-clients": "^3.996.7", "@aws-sdk/token-providers": "3.1004.0", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-wGtte+48xnhnhHMl/MsxzacBPs5A+7JJedjiP452IkHY7vsbYKcvQBqFye8LwdTJVeHtBHv+JFeTscnwepoWGg=="],
- "@aws-sdk/credential-provider-web-identity": ["@aws-sdk/credential-provider-web-identity@3.972.15", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/nested-clients": "^3.996.5", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-QTH6k93v+UOfFam/ado8zc71tH+enTVyuvLy9uEWXX1x894dN5ovtf/MdBDgFwq3g6c9mbtgVJ4B+yBqDtXvdA=="],
+ "@aws-sdk/credential-provider-web-identity": ["@aws-sdk/credential-provider-web-identity@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/nested-clients": "^3.996.7", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-8aiVJh6fTdl8gcyL+sVNcNwTtWpmoFa1Sh7xlj6Z7L/cZ/tYMEBHq44wTYG8Kt0z/PpGNopD89nbj3FHl9QmTA=="],
- "@aws-sdk/lib-storage": ["@aws-sdk/lib-storage@3.1002.0", "", { "dependencies": { "@smithy/abort-controller": "^4.2.10", "@smithy/middleware-endpoint": "^4.4.21", "@smithy/smithy-client": "^4.12.1", "buffer": "5.6.0", "events": "3.3.0", "stream-browserify": "3.0.0", "tslib": "^2.6.2" }, "peerDependencies": { "@aws-sdk/client-s3": "^3.1002.0" } }, "sha512-vOzNq63BLXUTawI4/D2kU/92wZcHsng6yuIAZuW1FofCvsWORdwdRxJYame506lKZTPCNw49LIAoBT2YJ2+pGw=="],
+ "@aws-sdk/lib-storage": ["@aws-sdk/lib-storage@3.1004.0", "", { "dependencies": { "@smithy/abort-controller": "^4.2.11", "@smithy/middleware-endpoint": "^4.4.22", "@smithy/smithy-client": "^4.12.2", "buffer": "5.6.0", "events": "3.3.0", "stream-browserify": "3.0.0", "tslib": "^2.6.2" }, "peerDependencies": { "@aws-sdk/client-s3": "^3.1004.0" } }, "sha512-4W6UkeLVd/1FyXFvD9PHMw5FSOY7tsf6+I52jmgdZwDZ9gJcJBx6wF9IhaVp1AXhScZGY9HqHiqYt0qlrSHrGw=="],
- "@aws-sdk/middleware-bucket-endpoint": ["@aws-sdk/middleware-bucket-endpoint@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@aws-sdk/util-arn-parser": "^3.972.2", "@smithy/node-config-provider": "^4.3.10", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "@smithy/util-config-provider": "^4.2.1", "tslib": "^2.6.2" } }, "sha512-3H2bhvb7Cb/S6WFsBy/Dy9q2aegC9JmGH1inO8Lb2sWirSqpLJlZmvQHPE29h2tIxzv6el/14X/tLCQ8BQU6ZQ=="],
+ "@aws-sdk/middleware-bucket-endpoint": ["@aws-sdk/middleware-bucket-endpoint@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@aws-sdk/util-arn-parser": "^3.972.3", "@smithy/node-config-provider": "^4.3.11", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-config-provider": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-goX+axlJ6PQlRnzE2bQisZ8wVrlm6dXJfBzMJhd8LhAIBan/w1Kl73fJnalM/S+18VnpzIHumyV6DtgmvqG5IA=="],
- "@aws-sdk/middleware-expect-continue": ["@aws-sdk/middleware-expect-continue@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-QMdffpU+GkSGC+bz6WdqlclqIeCsOfgX8JFZ5xvwDtX+UTj4mIXm3uXu7Ko6dBseRcJz1FA6T9OmlAAY6JgJUg=="],
+ "@aws-sdk/middleware-expect-continue": ["@aws-sdk/middleware-expect-continue@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-mvWqvm61bmZUKmmrtl2uWbokqpenY3Mc3Jf4nXB/Hse6gWxLPaCQThmhPBDzsPSV8/Odn8V6ovWt3pZ7vy4BFQ=="],
- "@aws-sdk/middleware-flexible-checksums": ["@aws-sdk/middleware-flexible-checksums@3.973.3", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@aws-crypto/crc32c": "5.2.0", "@aws-crypto/util": "5.2.0", "@aws-sdk/core": "^3.973.17", "@aws-sdk/crc64-nvme": "^3.972.3", "@aws-sdk/types": "^3.973.4", "@smithy/is-array-buffer": "^4.2.1", "@smithy/node-config-provider": "^4.3.10", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "@smithy/util-middleware": "^4.2.10", "@smithy/util-stream": "^4.5.16", "@smithy/util-utf8": "^4.2.1", "tslib": "^2.6.2" } }, "sha512-C9Mu9pXMpQh7jBydx0MrfQxNIKwJvKbVbJJ0GZthM+cQ+KTChXA01MwttRsMq0ZRb4pBJZQtIKDUxXusDr5OKg=="],
+ "@aws-sdk/middleware-flexible-checksums": ["@aws-sdk/middleware-flexible-checksums@3.973.4", "", { "dependencies": { "@aws-crypto/crc32": "5.2.0", "@aws-crypto/crc32c": "5.2.0", "@aws-crypto/util": "5.2.0", "@aws-sdk/core": "^3.973.18", "@aws-sdk/crc64-nvme": "^3.972.4", "@aws-sdk/types": "^3.973.5", "@smithy/is-array-buffer": "^4.2.2", "@smithy/node-config-provider": "^4.3.11", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-middleware": "^4.2.11", "@smithy/util-stream": "^4.5.17", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-7CH2jcGmkvkHc5Buz9IGbdjq1729AAlgYJiAvGq7qhCHqYleCsriWdSnmsqWTwdAfXHMT+pkxX3w6v5tJNcSug=="],
- "@aws-sdk/middleware-host-header": ["@aws-sdk/middleware-host-header@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-5XHwjPH1lHB+1q4bfC7T8Z5zZrZXfaLcjSMwTd1HPSPrCmPFMbg3UQ5vgNWcVj0xoX4HWqTGkSf2byrjlnRg5w=="],
+ "@aws-sdk/middleware-host-header": ["@aws-sdk/middleware-host-header@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-aHQZgztBFEpDU1BB00VWCIIm85JjGjQW1OG9+98BdmaOpguJvzmXBGbnAiYcciCd+IS4e9BEq664lhzGnWJHgQ=="],
- "@aws-sdk/middleware-location-constraint": ["@aws-sdk/middleware-location-constraint@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-XdZ2TLwyj3Am6kvUc67vquQvs6+D8npXvXgyEUJAdkUDx5oMFJKOqpK+UpJhVDsEL068WAJl2NEGzbSik7dGJQ=="],
+ "@aws-sdk/middleware-location-constraint": ["@aws-sdk/middleware-location-constraint@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-vdK1LJfffBp87Lj0Bw3WdK1rJk9OLDYdQpqoKgmpIZPe+4+HawZ6THTbvjhJt4C4MNnRrHTKHQjkwBiIpDBoig=="],
- "@aws-sdk/middleware-logger": ["@aws-sdk/middleware-logger@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-iFnaMFMQdljAPrvsCVKYltPt2j40LQqukAbXvW7v0aL5I+1GO7bZ/W8m12WxW3gwyK5p5u1WlHg8TSAizC5cZw=="],
+ "@aws-sdk/middleware-logger": ["@aws-sdk/middleware-logger@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-LXhiWlWb26txCU1vcI9PneESSeRp/RYY/McuM4SpdrimQR5NgwaPb4VJCadVeuGWgh6QmqZ6rAKSoL1ob16W6w=="],
- "@aws-sdk/middleware-recursion-detection": ["@aws-sdk/middleware-recursion-detection@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@aws/lambda-invoke-store": "^0.2.2", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-dY4v3of5EEMvik6+UDwQ96KfUFDk8m1oZDdkSc5lwi4o7rFrjnv0A+yTV+gu230iybQZnKgDLg/rt2P3H+Vscw=="],
+ "@aws-sdk/middleware-recursion-detection": ["@aws-sdk/middleware-recursion-detection@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@aws/lambda-invoke-store": "^0.2.2", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-l2VQdcBcYLzIzykCHtXlbpiVCZ94/xniLIkAj0jpnpjY4xlgZx7f56Ypn+uV1y3gG0tNVytJqo3K9bfMFee7SQ=="],
- "@aws-sdk/middleware-sdk-s3": ["@aws-sdk/middleware-sdk-s3@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/types": "^3.973.4", "@aws-sdk/util-arn-parser": "^3.972.2", "@smithy/core": "^3.23.7", "@smithy/node-config-provider": "^4.3.10", "@smithy/protocol-http": "^5.3.10", "@smithy/signature-v4": "^5.3.10", "@smithy/smithy-client": "^4.12.1", "@smithy/types": "^4.13.0", "@smithy/util-config-provider": "^4.2.1", "@smithy/util-middleware": "^4.2.10", "@smithy/util-stream": "^4.5.16", "@smithy/util-utf8": "^4.2.1", "tslib": "^2.6.2" } }, "sha512-uSyOGoVFMP44pTt29MIMfsOjegqE/7lT0K3HG0GWPiH2lD4rqZC/TRi/kH4zrGiOQdsaLc+dkfd7Sb2q8vh+gA=="],
+ "@aws-sdk/middleware-sdk-s3": ["@aws-sdk/middleware-sdk-s3@3.972.18", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/types": "^3.973.5", "@aws-sdk/util-arn-parser": "^3.972.3", "@smithy/core": "^3.23.8", "@smithy/node-config-provider": "^4.3.11", "@smithy/protocol-http": "^5.3.11", "@smithy/signature-v4": "^5.3.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/util-config-provider": "^4.2.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-stream": "^4.5.17", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-5E3XxaElrdyk6ZJ0TjH7Qm6ios4b/qQCiLr6oQ8NK7e4Kn6JBTJCaYioQCQ65BpZ1+l1mK5wTAac2+pEz0Smpw=="],
- "@aws-sdk/middleware-ssec": ["@aws-sdk/middleware-ssec@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-acvMUX9jF4I2Ew+Z/EA6gfaFaz9ehci5wxBmXCZeulLuv8m+iGf6pY9uKz8TPjg39bdAz3hxoE0eLP8Qz+IYlA=="],
+ "@aws-sdk/middleware-ssec": ["@aws-sdk/middleware-ssec@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-G9clGVuAml7d8DYzY6DnRi7TIIDRvZ3YpqJPz/8wnWS5fYx/FNWNmkO6iJVlVkQg9BfeMzd+bVPtPJOvC4B+nQ=="],
- "@aws-sdk/middleware-user-agent": ["@aws-sdk/middleware-user-agent@3.972.17", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/types": "^3.973.4", "@aws-sdk/util-endpoints": "^3.996.3", "@smithy/core": "^3.23.7", "@smithy/protocol-http": "^5.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-HHArkgWzomuwufXwheQqkddu763PWCpoNTq1dGjqXzJT/lojX3VlOqjNSR2Xvb6/T9ISfwYcMOcbFgUp4EWxXA=="],
+ "@aws-sdk/middleware-user-agent": ["@aws-sdk/middleware-user-agent@3.972.19", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/types": "^3.973.5", "@aws-sdk/util-endpoints": "^3.996.4", "@smithy/core": "^3.23.8", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-retry": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-Km90fcXt3W/iqujHzuM6IaDkYCj73gsYufcuWXApWdzoTy6KGk8fnchAjePMARU0xegIR3K4N3yIo1vy7OVe8A=="],
- "@aws-sdk/nested-clients": ["@aws-sdk/nested-clients@3.996.5", "", { "dependencies": { "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.973.17", "@aws-sdk/middleware-host-header": "^3.972.6", "@aws-sdk/middleware-logger": "^3.972.6", "@aws-sdk/middleware-recursion-detection": "^3.972.6", "@aws-sdk/middleware-user-agent": "^3.972.17", "@aws-sdk/region-config-resolver": "^3.972.6", "@aws-sdk/types": "^3.973.4", "@aws-sdk/util-endpoints": "^3.996.3", "@aws-sdk/util-user-agent-browser": "^3.972.6", "@aws-sdk/util-user-agent-node": "^3.973.2", "@smithy/config-resolver": "^4.4.9", "@smithy/core": "^3.23.7", "@smithy/fetch-http-handler": "^5.3.12", "@smithy/hash-node": "^4.2.10", "@smithy/invalid-dependency": "^4.2.10", "@smithy/middleware-content-length": "^4.2.10", "@smithy/middleware-endpoint": "^4.4.21", "@smithy/middleware-retry": "^4.4.38", "@smithy/middleware-serde": "^4.2.11", "@smithy/middleware-stack": "^4.2.10", "@smithy/node-config-provider": "^4.3.10", "@smithy/node-http-handler": "^4.4.13", "@smithy/protocol-http": "^5.3.10", "@smithy/smithy-client": "^4.12.1", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.10", "@smithy/util-base64": "^4.3.1", "@smithy/util-body-length-browser": "^4.2.1", "@smithy/util-body-length-node": "^4.2.2", "@smithy/util-defaults-mode-browser": "^4.3.37", "@smithy/util-defaults-mode-node": "^4.2.40", "@smithy/util-endpoints": "^3.3.1", "@smithy/util-middleware": "^4.2.10", "@smithy/util-retry": "^4.2.10", "@smithy/util-utf8": "^4.2.1", "tslib": "^2.6.2" } }, "sha512-zn0WApcULn7Rtl6T+KP2CQTZo/7wOa2YV1yHQnbijTQoi4YXQHM8s21JcJzt33/mqPh8AdvWX1f+83KvKuxlZw=="],
+ "@aws-sdk/nested-clients": ["@aws-sdk/nested-clients@3.996.7", "", { "dependencies": { "@aws-crypto/sha256-browser": "5.2.0", "@aws-crypto/sha256-js": "5.2.0", "@aws-sdk/core": "^3.973.18", "@aws-sdk/middleware-host-header": "^3.972.7", "@aws-sdk/middleware-logger": "^3.972.7", "@aws-sdk/middleware-recursion-detection": "^3.972.7", "@aws-sdk/middleware-user-agent": "^3.972.19", "@aws-sdk/region-config-resolver": "^3.972.7", "@aws-sdk/types": "^3.973.5", "@aws-sdk/util-endpoints": "^3.996.4", "@aws-sdk/util-user-agent-browser": "^3.972.7", "@aws-sdk/util-user-agent-node": "^3.973.4", "@smithy/config-resolver": "^4.4.10", "@smithy/core": "^3.23.8", "@smithy/fetch-http-handler": "^5.3.13", "@smithy/hash-node": "^4.2.11", "@smithy/invalid-dependency": "^4.2.11", "@smithy/middleware-content-length": "^4.2.11", "@smithy/middleware-endpoint": "^4.4.22", "@smithy/middleware-retry": "^4.4.39", "@smithy/middleware-serde": "^4.2.12", "@smithy/middleware-stack": "^4.2.11", "@smithy/node-config-provider": "^4.3.11", "@smithy/node-http-handler": "^4.4.14", "@smithy/protocol-http": "^5.3.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-body-length-node": "^4.2.3", "@smithy/util-defaults-mode-browser": "^4.3.38", "@smithy/util-defaults-mode-node": "^4.2.41", "@smithy/util-endpoints": "^3.3.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-retry": "^4.2.11", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-MlGWA8uPaOs5AiTZ5JLM4uuWDm9EEAnm9cqwvqQIc6kEgel/8s1BaOWm9QgUcfc9K8qd7KkC3n43yDbeXOA2tg=="],
- "@aws-sdk/region-config-resolver": ["@aws-sdk/region-config-resolver@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/config-resolver": "^4.4.9", "@smithy/node-config-provider": "^4.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-Aa5PusHLXAqLTX1UKDvI3pHQJtIsF7Q+3turCHqfz/1F61/zDMWfbTC8evjhrrYVAtz9Vsv3SJ/waSUeu7B6gw=="],
+ "@aws-sdk/region-config-resolver": ["@aws-sdk/region-config-resolver@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/config-resolver": "^4.4.10", "@smithy/node-config-provider": "^4.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-/Ev/6AI8bvt4HAAptzSjThGUMjcWaX3GX8oERkB0F0F9x2dLSBdgFDiyrRz3i0u0ZFZFQ1b28is4QhyqXTUsVA=="],
- "@aws-sdk/signature-v4-multi-region": ["@aws-sdk/signature-v4-multi-region@3.996.5", "", { "dependencies": { "@aws-sdk/middleware-sdk-s3": "^3.972.17", "@aws-sdk/types": "^3.973.4", "@smithy/protocol-http": "^5.3.10", "@smithy/signature-v4": "^5.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-AVIhf74wRMzU1WBPVzcGPjlADF5VxZ8m8Ctm1v7eO4/reWMhZnEBn4tlR4vM4pOYFkdrYp3MTzYVZIikCO+53Q=="],
+ "@aws-sdk/signature-v4-multi-region": ["@aws-sdk/signature-v4-multi-region@3.996.6", "", { "dependencies": { "@aws-sdk/middleware-sdk-s3": "^3.972.18", "@aws-sdk/types": "^3.973.5", "@smithy/protocol-http": "^5.3.11", "@smithy/signature-v4": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-NnsOQsVmJXy4+IdPFUjRCWPn9qNH1TzS/f7MiWgXeoHs903tJpAWQWQtoFvLccyPoBgomKP9L89RRr2YsT/L0g=="],
- "@aws-sdk/token-providers": ["@aws-sdk/token-providers@3.1002.0", "", { "dependencies": { "@aws-sdk/core": "^3.973.17", "@aws-sdk/nested-clients": "^3.996.5", "@aws-sdk/types": "^3.973.4", "@smithy/property-provider": "^4.2.10", "@smithy/shared-ini-file-loader": "^4.4.5", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-x972uKOydFn4Rb0PZJzLdNW59rH0KWC78Q2JbQzZpGlGt0DxjYdDRwBG6F42B1MyaEwHGqO/tkGc4r3/PRFfMw=="],
+ "@aws-sdk/token-providers": ["@aws-sdk/token-providers@3.1004.0", "", { "dependencies": { "@aws-sdk/core": "^3.973.18", "@aws-sdk/nested-clients": "^3.996.7", "@aws-sdk/types": "^3.973.5", "@smithy/property-provider": "^4.2.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-j9BwZZId9sFp+4GPhf6KrwO8Tben2sXibZA8D1vv2I1zBdvkUHcBA2g4pkqIpTRalMTLC0NPkBPX0gERxfy/iA=="],
- "@aws-sdk/types": ["@aws-sdk/types@3.973.4", "", { "dependencies": { "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-RW60aH26Bsc016Y9B98hC0Plx6fK5P2v/iQYwMzrSjiDh1qRMUCP6KrXHYEHe3uFvKiOC93Z9zk4BJsUi6Tj1Q=="],
+ "@aws-sdk/types": ["@aws-sdk/types@3.973.5", "", { "dependencies": { "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-hl7BGwDCWsjH8NkZfx+HgS7H2LyM2lTMAI7ba9c8O0KqdBLTdNJivsHpqjg9rNlAlPyREb6DeDRXUl0s8uFdmQ=="],
- "@aws-sdk/util-arn-parser": ["@aws-sdk/util-arn-parser@3.972.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-VkykWbqMjlSgBFDyrY3nOSqupMc6ivXuGmvci6Q3NnLq5kC+mKQe2QBZ4nrWRE/jqOxeFP2uYzLtwncYYcvQDg=="],
+ "@aws-sdk/util-arn-parser": ["@aws-sdk/util-arn-parser@3.972.3", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-HzSD8PMFrvgi2Kserxuff5VitNq2sgf3w9qxmskKDiDTThWfVteJxuCS9JXiPIPtmCrp+7N9asfIaVhBFORllA=="],
- "@aws-sdk/util-endpoints": ["@aws-sdk/util-endpoints@3.996.3", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.10", "@smithy/util-endpoints": "^3.3.1", "tslib": "^2.6.2" } }, "sha512-yWIQSNiCjykLL+ezN5A+DfBb1gfXTytBxm57e64lYmwxDHNmInYHRJYYRAGWG1o77vKEiWaw4ui28e3yb1k5aQ=="],
+ "@aws-sdk/util-endpoints": ["@aws-sdk/util-endpoints@3.996.4", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "@smithy/util-endpoints": "^3.3.2", "tslib": "^2.6.2" } }, "sha512-Hek90FBmd4joCFj+Vc98KLJh73Zqj3s2W56gjAcTkrNLMDI5nIFkG9YpfcJiVI1YlE2Ne1uOQNe+IgQ/Vz2XRA=="],
- "@aws-sdk/util-locate-window": ["@aws-sdk/util-locate-window@3.965.4", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-H1onv5SkgPBK2P6JR2MjGgbOnttoNzSPIRoeZTNPZYyaplwGg50zS3amXvXqF0/qfXpWEC9rLWU564QTB9bSog=="],
+ "@aws-sdk/util-locate-window": ["@aws-sdk/util-locate-window@3.965.5", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-WhlJNNINQB+9qtLtZJcpQdgZw3SCDCpXdUJP7cToGwHbCWCnRckGlc6Bx/OhWwIYFNAn+FIydY8SZ0QmVu3xTQ=="],
- "@aws-sdk/util-user-agent-browser": ["@aws-sdk/util-user-agent-browser@3.972.6", "", { "dependencies": { "@aws-sdk/types": "^3.973.4", "@smithy/types": "^4.13.0", "bowser": "^2.11.0", "tslib": "^2.6.2" } }, "sha512-Fwr/llD6GOrFgQnKaI2glhohdGuBDfHfora6iG9qsBBBR8xv1SdCSwbtf5CWlUdCw5X7g76G/9Hf0Inh0EmoxA=="],
+ "@aws-sdk/util-user-agent-browser": ["@aws-sdk/util-user-agent-browser@3.972.7", "", { "dependencies": { "@aws-sdk/types": "^3.973.5", "@smithy/types": "^4.13.0", "bowser": "^2.11.0", "tslib": "^2.6.2" } }, "sha512-7SJVuvhKhMF/BkNS1n0QAJYgvEwYbK2QLKBrzDiwQGiTRU6Yf1f3nehTzm/l21xdAOtWSfp2uWSddPnP2ZtsVw=="],
- "@aws-sdk/util-user-agent-node": ["@aws-sdk/util-user-agent-node@3.973.2", "", { "dependencies": { "@aws-sdk/middleware-user-agent": "^3.972.17", "@aws-sdk/types": "^3.973.4", "@smithy/node-config-provider": "^4.3.10", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" }, "peerDependencies": { "aws-crt": ">=1.0.0" }, "optionalPeers": ["aws-crt"] }, "sha512-lpaIuekdkpw7VRiik0IZmd6TyvEUcuLgKZ5fKRGpCA3I4PjrD/XH15sSwW+OptxQjNU4DEzSxag70spC9SluvA=="],
+ "@aws-sdk/util-user-agent-node": ["@aws-sdk/util-user-agent-node@3.973.4", "", { "dependencies": { "@aws-sdk/middleware-user-agent": "^3.972.19", "@aws-sdk/types": "^3.973.5", "@smithy/node-config-provider": "^4.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" }, "peerDependencies": { "aws-crt": ">=1.0.0" }, "optionalPeers": ["aws-crt"] }, "sha512-uqKeLqZ9D3nQjH7HGIERNXK9qnSpUK08l4MlJ5/NZqSSdeJsVANYp437EM9sEzwU28c2xfj2V6qlkqzsgtKs6Q=="],
- "@aws-sdk/xml-builder": ["@aws-sdk/xml-builder@3.972.9", "", { "dependencies": { "@smithy/types": "^4.13.0", "fast-xml-parser": "5.4.1", "tslib": "^2.6.2" } }, "sha512-ItnlMgSqkPrUfJs7EsvU/01zw5UeIb2tNPhD09LBLHbg+g+HDiKibSLwpkuz/ZIlz4F2IMn+5XgE4AK/pfPuog=="],
+ "@aws-sdk/xml-builder": ["@aws-sdk/xml-builder@3.972.10", "", { "dependencies": { "@smithy/types": "^4.13.0", "fast-xml-parser": "5.4.1", "tslib": "^2.6.2" } }, "sha512-OnejAIVD+CxzyAUrVic7lG+3QRltyja9LoNqCE/1YVs8ichoTbJlVSaZ9iSMcnHLyzrSNtvaOGjSDRP+d/ouFA=="],
"@aws/lambda-invoke-store": ["@aws/lambda-invoke-store@0.2.3", "", {}, "sha512-oLvsaPMTBejkkmHhjf09xTgk71mOqyr/409NKhRIL08If7AhVfUsJhVsx386uJaqNd42v9kWamQ9lFbkoC2dYw=="],
@@ -364,6 +418,10 @@
"@btca/analytics-proxy": ["@btca/analytics-proxy@workspace:apps/analytics-proxy"],
+ "@btca/convex": ["@btca/convex@workspace:packages/convex"],
+
+ "@btca/desktop": ["@btca/desktop@workspace:apps/desktop"],
+
"@btca/docs": ["@btca/docs@workspace:apps/docs"],
"@btca/shared": ["@btca/shared@workspace:packages/shared"],
@@ -410,7 +468,7 @@
"@effect/platform-bun": ["@effect/platform-bun@4.0.0-beta.20", "", { "dependencies": { "@effect/platform-node-shared": "^4.0.0-beta.20" }, "peerDependencies": { "effect": "^4.0.0-beta.20" } }, "sha512-6qgk4k7Z1+pgZihtqHcAUpnDxtsMogHJkJKgknXE36smCIoctx3mK5d5A2rrKmK91zsWo3OvATZhhR6LzFXEPA=="],
- "@effect/platform-node-shared": ["@effect/platform-node-shared@4.0.0-beta.27", "", { "dependencies": { "@types/ws": "^8.18.1", "ws": "^8.19.0" }, "peerDependencies": { "effect": "^4.0.0-beta.27" } }, "sha512-prRa5MIWGAvq2g6FgVP5RFfPwI+DvcXCaEZetvPMK70wdkTR1+jKQqJ9QDKWNIP+2d4llE/cF21SvcpBJWq0gw=="],
+ "@effect/platform-node-shared": ["@effect/platform-node-shared@4.0.0-beta.29", "", { "dependencies": { "@types/ws": "^8.18.1", "ws": "^8.19.0" }, "peerDependencies": { "effect": "^4.0.0-beta.29" } }, "sha512-6nV/yI1LgYJ1PzTjb20Ar4a+Z9dBltd2QCNzLj9NmdBQusJO9ve7qk0GBa3IFkmAVAqUfCVXEwhYz3TjnP+lrA=="],
"@emotion/babel-plugin": ["@emotion/babel-plugin@11.13.5", "", { "dependencies": { "@babel/helper-module-imports": "^7.16.7", "@babel/runtime": "^7.18.3", "@emotion/hash": "^0.9.2", "@emotion/memoize": "^0.9.0", "@emotion/serialize": "^1.3.3", "babel-plugin-macros": "^3.1.0", "convert-source-map": "^1.5.0", "escape-string-regexp": "^4.0.0", "find-root": "^1.1.0", "source-map": "^0.5.7", "stylis": "4.2.0" } }, "sha512-pxHCpT2ex+0q+HH91/zsdHkw/lXd468DIN2zvfvLtPKLLMo6gQj7oLObq8PhkrxOZb/gGCq03S3Z7PDhS8pduQ=="],
@@ -434,57 +492,57 @@
"@emotion/weak-memoize": ["@emotion/weak-memoize@0.3.1", "", {}, "sha512-EsBwpc7hBUJWAsNPBmJy4hxWx12v6bshQsldrVmjxJoc3isbxhOrF2IcCpaXxfvq03NwkI7sbsOLXbYuqF/8Ww=="],
- "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.25.12", "", { "os": "aix", "cpu": "ppc64" }, "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA=="],
+ "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.27.0", "", { "os": "aix", "cpu": "ppc64" }, "sha512-KuZrd2hRjz01y5JK9mEBSD3Vj3mbCvemhT466rSuJYeE/hjuBrHfjjcjMdTm/sz7au+++sdbJZJmuBwQLuw68A=="],
- "@esbuild/android-arm": ["@esbuild/android-arm@0.25.12", "", { "os": "android", "cpu": "arm" }, "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg=="],
+ "@esbuild/android-arm": ["@esbuild/android-arm@0.27.0", "", { "os": "android", "cpu": "arm" }, "sha512-j67aezrPNYWJEOHUNLPj9maeJte7uSMM6gMoxfPC9hOg8N02JuQi/T7ewumf4tNvJadFkvLZMlAq73b9uwdMyQ=="],
- "@esbuild/android-arm64": ["@esbuild/android-arm64@0.25.12", "", { "os": "android", "cpu": "arm64" }, "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg=="],
+ "@esbuild/android-arm64": ["@esbuild/android-arm64@0.27.0", "", { "os": "android", "cpu": "arm64" }, "sha512-CC3vt4+1xZrs97/PKDkl0yN7w8edvU2vZvAFGD16n9F0Cvniy5qvzRXjfO1l94efczkkQE6g1x0i73Qf5uthOQ=="],
- "@esbuild/android-x64": ["@esbuild/android-x64@0.25.12", "", { "os": "android", "cpu": "x64" }, "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg=="],
+ "@esbuild/android-x64": ["@esbuild/android-x64@0.27.0", "", { "os": "android", "cpu": "x64" }, "sha512-wurMkF1nmQajBO1+0CJmcN17U4BP6GqNSROP8t0X/Jiw2ltYGLHpEksp9MpoBqkrFR3kv2/te6Sha26k3+yZ9Q=="],
- "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.25.12", "", { "os": "darwin", "cpu": "arm64" }, "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg=="],
+ "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.27.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-uJOQKYCcHhg07DL7i8MzjvS2LaP7W7Pn/7uA0B5S1EnqAirJtbyw4yC5jQ5qcFjHK9l6o/MX9QisBg12kNkdHg=="],
- "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.25.12", "", { "os": "darwin", "cpu": "x64" }, "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA=="],
+ "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.27.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-8mG6arH3yB/4ZXiEnXof5MK72dE6zM9cDvUcPtxhUZsDjESl9JipZYW60C3JGreKCEP+p8P/72r69m4AZGJd5g=="],
- "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.25.12", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg=="],
+ "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.27.0", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-9FHtyO988CwNMMOE3YIeci+UV+x5Zy8fI2qHNpsEtSF83YPBmE8UWmfYAQg6Ux7Gsmd4FejZqnEUZCMGaNQHQw=="],
- "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.25.12", "", { "os": "freebsd", "cpu": "x64" }, "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ=="],
+ "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.27.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-zCMeMXI4HS/tXvJz8vWGexpZj2YVtRAihHLk1imZj4efx1BQzN76YFeKqlDr3bUWI26wHwLWPd3rwh6pe4EV7g=="],
- "@esbuild/linux-arm": ["@esbuild/linux-arm@0.25.12", "", { "os": "linux", "cpu": "arm" }, "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw=="],
+ "@esbuild/linux-arm": ["@esbuild/linux-arm@0.27.0", "", { "os": "linux", "cpu": "arm" }, "sha512-t76XLQDpxgmq2cNXKTVEB7O7YMb42atj2Re2Haf45HkaUpjM2J0UuJZDuaGbPbamzZ7bawyGFUkodL+zcE+jvQ=="],
- "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.25.12", "", { "os": "linux", "cpu": "arm64" }, "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ=="],
+ "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.27.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-AS18v0V+vZiLJyi/4LphvBE+OIX682Pu7ZYNsdUHyUKSoRwdnOsMf6FDekwoAFKej14WAkOef3zAORJgAtXnlQ=="],
- "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.25.12", "", { "os": "linux", "cpu": "ia32" }, "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA=="],
+ "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.27.0", "", { "os": "linux", "cpu": "ia32" }, "sha512-Mz1jxqm/kfgKkc/KLHC5qIujMvnnarD9ra1cEcrs7qshTUSksPihGrWHVG5+osAIQ68577Zpww7SGapmzSt4Nw=="],
- "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng=="],
+ "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-QbEREjdJeIreIAbdG2hLU1yXm1uu+LTdzoq1KCo4G4pFOLlvIspBm36QrQOar9LFduavoWX2msNFAAAY9j4BDg=="],
- "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw=="],
+ "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-sJz3zRNe4tO2wxvDpH/HYJilb6+2YJxo/ZNbVdtFiKDufzWq4JmKAiHy9iGoLjAV7r/W32VgaHGkk35cUXlNOg=="],
- "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.25.12", "", { "os": "linux", "cpu": "ppc64" }, "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA=="],
+ "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.27.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-z9N10FBD0DCS2dmSABDBb5TLAyF1/ydVb+N4pi88T45efQ/w4ohr/F/QYCkxDPnkhkp6AIpIcQKQ8F0ANoA2JA=="],
- "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w=="],
+ "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-pQdyAIZ0BWIC5GyvVFn5awDiO14TkT/19FTmFcPdDec94KJ1uZcmFs21Fo8auMXzD4Tt+diXu1LW1gHus9fhFQ=="],
- "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.25.12", "", { "os": "linux", "cpu": "s390x" }, "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg=="],
+ "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.27.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-hPlRWR4eIDDEci953RI1BLZitgi5uqcsjKMxwYfmi4LcwyWo2IcRP+lThVnKjNtk90pLS8nKdroXYOqW+QQH+w=="],
- "@esbuild/linux-x64": ["@esbuild/linux-x64@0.25.12", "", { "os": "linux", "cpu": "x64" }, "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw=="],
+ "@esbuild/linux-x64": ["@esbuild/linux-x64@0.27.0", "", { "os": "linux", "cpu": "x64" }, "sha512-1hBWx4OUJE2cab++aVZ7pObD6s+DK4mPGpemtnAORBvb5l/g5xFGk0vc0PjSkrDs0XaXj9yyob3d14XqvnQ4gw=="],
- "@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg=="],
+ "@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-6m0sfQfxfQfy1qRuecMkJlf1cIzTOgyaeXaiVaaki8/v+WB+U4hc6ik15ZW6TAllRlg/WuQXxWj1jx6C+dfy3w=="],
- "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.25.12", "", { "os": "none", "cpu": "x64" }, "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ=="],
+ "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.27.0", "", { "os": "none", "cpu": "x64" }, "sha512-xbbOdfn06FtcJ9d0ShxxvSn2iUsGd/lgPIO2V3VZIPDbEaIj1/3nBBe1AwuEZKXVXkMmpr6LUAgMkLD/4D2PPA=="],
- "@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.25.12", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A=="],
+ "@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.27.0", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fWgqR8uNbCQ/GGv0yhzttj6sU/9Z5/Sv/VGU3F5OuXK6J6SlriONKrQ7tNlwBrJZXRYk5jUhuWvF7GYzGguBZQ=="],
- "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.25.12", "", { "os": "openbsd", "cpu": "x64" }, "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw=="],
+ "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.27.0", "", { "os": "openbsd", "cpu": "x64" }, "sha512-aCwlRdSNMNxkGGqQajMUza6uXzR/U0dIl1QmLjPtRbLOx3Gy3otfFu/VjATy4yQzo9yFDGTxYDo1FfAD9oRD2A=="],
- "@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg=="],
+ "@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-nyvsBccxNAsNYz2jVFYwEGuRRomqZ149A39SHWk4hV0jWxKM0hjBPm3AmdxcbHiFLbBSwG6SbpIcUbXjgyECfA=="],
- "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.25.12", "", { "os": "sunos", "cpu": "x64" }, "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w=="],
+ "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.27.0", "", { "os": "sunos", "cpu": "x64" }, "sha512-Q1KY1iJafM+UX6CFEL+F4HRTgygmEW568YMqDA5UV97AuZSm21b7SXIrRJDwXWPzr8MGr75fUZPV67FdtMHlHA=="],
- "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.25.12", "", { "os": "win32", "cpu": "arm64" }, "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg=="],
+ "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.27.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-W1eyGNi6d+8kOmZIwi/EDjrL9nxQIQ0MiGqe/AWc6+IaHloxHSGoeRgDRKHFISThLmsewZ5nHFvGFWdBYlgKPg=="],
- "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.25.12", "", { "os": "win32", "cpu": "ia32" }, "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ=="],
+ "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.27.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-30z1aKL9h22kQhilnYkORFYt+3wp7yZsHWus+wSKAJR8JtdfI76LJ4SBdMsCopTR3z/ORqVu5L1vtnHZWVj4cQ=="],
- "@esbuild/win32-x64": ["@esbuild/win32-x64@0.25.12", "", { "os": "win32", "cpu": "x64" }, "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA=="],
+ "@esbuild/win32-x64": ["@esbuild/win32-x64@0.27.0", "", { "os": "win32", "cpu": "x64" }, "sha512-aIitBcjQeyOhMTImhLZmtxfdOcuNRpwlPNmlFKPcHQYPhEssw75Cl1TSXJXpMkzaua9FUetx/4OQKq7eJul5Cg=="],
"@exodus/bytes": ["@exodus/bytes@1.15.0", "", { "peerDependencies": { "@noble/hashes": "^1.8.0 || ^2.0.0" }, "optionalPeers": ["@noble/hashes"] }, "sha512-UY0nlA+feH81UGSHv92sLEPLCeZFjXOuHhrIo0HQydScuQc8s0A7kL/UdgwgDq8g8ilksmuoF35YVTNphV2aBQ=="],
@@ -630,7 +688,7 @@
"@js-sdsl/ordered-map": ["@js-sdsl/ordered-map@4.4.2", "", {}, "sha512-iUKgm52T8HOE/makSxjqoWhe95ZJA1/G1sYsGev2JDKUSS14KAgg1LHb+Ba+IPow0xflbnSkOsZcO08C7w1gYw=="],
- "@lucide/svelte": ["@lucide/svelte@0.577.0", "", { "peerDependencies": { "svelte": "^5" } }, "sha512-0P6mkySd2MapIEgq08tADPmcN4DHndC/02PWwaLkOerXlx5Sv9aT4BxyXLIY+eccr0g/nEyCYiJesqS61YdBZQ=="],
+ "@lucide/svelte": ["@lucide/svelte@0.562.0", "", { "peerDependencies": { "svelte": "^5" } }, "sha512-wDMULwtTFN2Sc/TFBm6gfuVCNb4Y5P9LDrwxNnUbV52+IEU7NXZmvxwXoz+vrrpad6Xupq+Hw5eUlqIHEGhouw=="],
"@mapbox/node-pre-gyp": ["@mapbox/node-pre-gyp@2.0.3", "", { "dependencies": { "consola": "^3.2.3", "detect-libc": "^2.0.0", "https-proxy-agent": "^7.0.5", "node-fetch": "^2.6.7", "nopt": "^8.0.0", "semver": "^7.5.3", "tar": "^7.4.0" }, "bin": { "node-pre-gyp": "bin/node-pre-gyp" } }, "sha512-uwPAhccfFJlsfCxMYTwOdVfOz3xqyj8xYL3zJj8f0pb30tLohnnFPhLuqp4/qoEz8sNxe4SESZedcBojRefIzg=="],
@@ -716,7 +774,7 @@
"@opentelemetry/sdk-node": ["@opentelemetry/sdk-node@0.207.0", "", { "dependencies": { "@opentelemetry/api-logs": "0.207.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/exporter-logs-otlp-grpc": "0.207.0", "@opentelemetry/exporter-logs-otlp-http": "0.207.0", "@opentelemetry/exporter-logs-otlp-proto": "0.207.0", "@opentelemetry/exporter-metrics-otlp-grpc": "0.207.0", "@opentelemetry/exporter-metrics-otlp-http": "0.207.0", "@opentelemetry/exporter-metrics-otlp-proto": "0.207.0", "@opentelemetry/exporter-prometheus": "0.207.0", "@opentelemetry/exporter-trace-otlp-grpc": "0.207.0", "@opentelemetry/exporter-trace-otlp-http": "0.207.0", "@opentelemetry/exporter-trace-otlp-proto": "0.207.0", "@opentelemetry/exporter-zipkin": "2.2.0", "@opentelemetry/instrumentation": "0.207.0", "@opentelemetry/propagator-b3": "2.2.0", "@opentelemetry/propagator-jaeger": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/sdk-logs": "0.207.0", "@opentelemetry/sdk-metrics": "2.2.0", "@opentelemetry/sdk-trace-base": "2.2.0", "@opentelemetry/sdk-trace-node": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-hnRsX/M8uj0WaXOBvFenQ8XsE8FLVh2uSnn1rkWu4mx+qu7EKGUZvZng6y/95cyzsqOfiaDDr08Ek4jppkIDNg=="],
- "@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+ "@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.6.0", "", { "dependencies": { "@opentelemetry/core": "2.6.0", "@opentelemetry/resources": "2.6.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-g/OZVkqlxllgFM7qMKqbPV9c1DUPhQ7d4n3pgZFcrnrNft9eJXZM2TNHTPYREJBrtNdRytYyvwjgL5geDKl3EQ=="],
"@opentelemetry/sdk-trace-node": ["@opentelemetry/sdk-trace-node@2.2.0", "", { "dependencies": { "@opentelemetry/context-async-hooks": "2.2.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/sdk-trace-base": "2.2.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.0.0 <1.10.0" } }, "sha512-+OaRja3f0IqGG2kptVeYsrZQK9nKRSpfFrKtRBq4uh6nIB8bTBgaGvYQrQoRrQWQMA5dK5yLhDMDc0dvYvCOIQ=="],
@@ -886,7 +944,7 @@
"@smithy/config-resolver": ["@smithy/config-resolver@4.4.10", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.11", "@smithy/types": "^4.13.0", "@smithy/util-config-provider": "^4.2.2", "@smithy/util-endpoints": "^3.3.2", "@smithy/util-middleware": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-IRTkd6ps0ru+lTWnfnsbXzW80A8Od8p3pYiZnW98K2Hb20rqfsX7VTlfUwhrcOeSSy68Gn9WBofwPuw3e5CCsg=="],
- "@smithy/core": ["@smithy/core@3.23.8", "", { "dependencies": { "@smithy/middleware-serde": "^4.2.12", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-stream": "^4.5.17", "@smithy/util-utf8": "^4.2.2", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-f7uPeBi7ehmLT4YF2u9j3qx6lSnurG1DLXOsTtJrIRNDF7VXio4BGHQ+SQteN/BrUVudbkuL4v7oOsRCzq4BqA=="],
+ "@smithy/core": ["@smithy/core@3.23.9", "", { "dependencies": { "@smithy/middleware-serde": "^4.2.12", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-base64": "^4.3.2", "@smithy/util-body-length-browser": "^4.2.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-stream": "^4.5.17", "@smithy/util-utf8": "^4.2.2", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-1Vcut4LEL9HZsdpI0vFiRYIsaoPwZLjAxnVQDUMQK8beMS+EYPLDQCXtbzfxmM5GzSgjfe2Q9M7WaXwIMQllyQ=="],
"@smithy/credential-provider-imds": ["@smithy/credential-provider-imds@4.2.11", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.11", "@smithy/property-provider": "^4.2.11", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-lBXrS6ku0kTj3xLmsJW0WwqWbGQ6ueooYyp/1L9lkyT0M02C+DWwYwc5aTyXFbRaK38ojALxNixg+LxKSHZc0g=="],
@@ -916,9 +974,9 @@
"@smithy/middleware-content-length": ["@smithy/middleware-content-length@4.2.11", "", { "dependencies": { "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-UvIfKYAKhCzr4p6jFevPlKhQwyQwlJ6IeKLDhmV1PlYfcW3RL4ROjNEDtSik4NYMi9kDkH7eSwyTP3vNJ/u/Dw=="],
- "@smithy/middleware-endpoint": ["@smithy/middleware-endpoint@4.4.22", "", { "dependencies": { "@smithy/core": "^3.23.8", "@smithy/middleware-serde": "^4.2.12", "@smithy/node-config-provider": "^4.3.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "@smithy/util-middleware": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-sc81w1o4Jy+/MAQlY3sQ8C7CmSpcvIi3TAzXblUv2hjG11BBSJi/Cw8vDx5BxMxapuH2I+Gc+45vWsgU07WZRQ=="],
+ "@smithy/middleware-endpoint": ["@smithy/middleware-endpoint@4.4.23", "", { "dependencies": { "@smithy/core": "^3.23.9", "@smithy/middleware-serde": "^4.2.12", "@smithy/node-config-provider": "^4.3.11", "@smithy/shared-ini-file-loader": "^4.4.6", "@smithy/types": "^4.13.0", "@smithy/url-parser": "^4.2.11", "@smithy/util-middleware": "^4.2.11", "tslib": "^2.6.2" } }, "sha512-UEFIejZy54T1EJn2aWJ45voB7RP2T+IRzUqocIdM6GFFa5ClZncakYJfcYnoXt3UsQrZZ9ZRauGm77l9UCbBLw=="],
- "@smithy/middleware-retry": ["@smithy/middleware-retry@4.4.39", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.11", "@smithy/protocol-http": "^5.3.11", "@smithy/service-error-classification": "^4.2.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "@smithy/util-middleware": "^4.2.11", "@smithy/util-retry": "^4.2.11", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-MCVCxaCzuZgiHtHGV2Ke44nh6t4+8/tO+rTYOzrr2+G4nMLU/qbzNCWKBX54lyEaVcGQrfOJiG2f8imtiw+nIQ=="],
+ "@smithy/middleware-retry": ["@smithy/middleware-retry@4.4.40", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.11", "@smithy/protocol-http": "^5.3.11", "@smithy/service-error-classification": "^4.2.11", "@smithy/smithy-client": "^4.12.3", "@smithy/types": "^4.13.0", "@smithy/util-middleware": "^4.2.11", "@smithy/util-retry": "^4.2.11", "@smithy/uuid": "^1.1.2", "tslib": "^2.6.2" } }, "sha512-YhEMakG1Ae57FajERdHNZ4ShOPIY7DsgV+ZoAxo/5BT0KIe+f6DDU2rtIymNNFIj22NJfeeI6LWIifrwM0f+rA=="],
"@smithy/middleware-serde": ["@smithy/middleware-serde@4.2.12", "", { "dependencies": { "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-W9g1bOLui7Xn5FABRVS0o3rXL0gfN37d/8I/W7i0N7oxjx9QecUmXEMSUMADTODwdtka9cN43t5BI2CodLJpng=="],
@@ -942,7 +1000,7 @@
"@smithy/signature-v4": ["@smithy/signature-v4@5.3.11", "", { "dependencies": { "@smithy/is-array-buffer": "^4.2.2", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-hex-encoding": "^4.2.2", "@smithy/util-middleware": "^4.2.11", "@smithy/util-uri-escape": "^4.2.2", "@smithy/util-utf8": "^4.2.2", "tslib": "^2.6.2" } }, "sha512-V1L6N9aKOBAN4wEHLyqjLBnAz13mtILU0SeDrjOaIZEeN6IFa6DxwRt1NNpOdmSpQUfkBj0qeD3m6P77uzMhgQ=="],
- "@smithy/smithy-client": ["@smithy/smithy-client@4.12.2", "", { "dependencies": { "@smithy/core": "^3.23.8", "@smithy/middleware-endpoint": "^4.4.22", "@smithy/middleware-stack": "^4.2.11", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-stream": "^4.5.17", "tslib": "^2.6.2" } }, "sha512-HezY3UuG0k4T+4xhFKctLXCA5N2oN+Rtv+mmL8Gt7YmsUY2yhmcLyW75qrSzldfj75IsCW/4UhY3s20KcFnZqA=="],
+ "@smithy/smithy-client": ["@smithy/smithy-client@4.12.3", "", { "dependencies": { "@smithy/core": "^3.23.9", "@smithy/middleware-endpoint": "^4.4.23", "@smithy/middleware-stack": "^4.2.11", "@smithy/protocol-http": "^5.3.11", "@smithy/types": "^4.13.0", "@smithy/util-stream": "^4.5.17", "tslib": "^2.6.2" } }, "sha512-7k4UxjSpHmPN2AxVhvIazRSzFQjWnud3sOsXcFStzagww17j1cFQYqTSiQ8xuYK3vKLR1Ni8FzuT3VlKr3xCNw=="],
"@smithy/types": ["@smithy/types@4.13.0", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-COuLsZILbbQsdrwKQpkkpyep7lCsByxwj7m0Mg5v66/ZTyenlfBc40/QFQ5chO0YN/PNEH1Bi3fGtfXPnYNeDw=="],
@@ -958,9 +1016,9 @@
"@smithy/util-config-provider": ["@smithy/util-config-provider@4.2.2", "", { "dependencies": { "tslib": "^2.6.2" } }, "sha512-dWU03V3XUprJwaUIFVv4iOnS1FC9HnMHDfUrlNDSh4315v0cWyaIErP8KiqGVbf5z+JupoVpNM7ZB3jFiTejvQ=="],
- "@smithy/util-defaults-mode-browser": ["@smithy/util-defaults-mode-browser@4.3.38", "", { "dependencies": { "@smithy/property-provider": "^4.2.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-c8P1mFLNxcsdAMabB8/VUQUbWzFmgujWi4bAXSggcqLYPc8V4U5abqFqOyn+dK4YT+q8UyCVkTO8807t4t2syA=="],
+ "@smithy/util-defaults-mode-browser": ["@smithy/util-defaults-mode-browser@4.3.39", "", { "dependencies": { "@smithy/property-provider": "^4.2.11", "@smithy/smithy-client": "^4.12.3", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-ui7/Ho/+VHqS7Km2wBw4/Ab4RktoiSshgcgpJzC4keFPs6tLJS4IQwbeahxQS3E/w98uq6E1mirCH/id9xIXeQ=="],
- "@smithy/util-defaults-mode-node": ["@smithy/util-defaults-mode-node@4.2.41", "", { "dependencies": { "@smithy/config-resolver": "^4.4.10", "@smithy/credential-provider-imds": "^4.2.11", "@smithy/node-config-provider": "^4.3.11", "@smithy/property-provider": "^4.2.11", "@smithy/smithy-client": "^4.12.2", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-/UG+9MT3UZAR0fLzOtMJMfWGcjjHvgggq924x/CRy8vRbL+yFf3Z6vETlvq8vDH92+31P/1gSOFoo7303wN8WQ=="],
+ "@smithy/util-defaults-mode-node": ["@smithy/util-defaults-mode-node@4.2.42", "", { "dependencies": { "@smithy/config-resolver": "^4.4.10", "@smithy/credential-provider-imds": "^4.2.11", "@smithy/node-config-provider": "^4.3.11", "@smithy/property-provider": "^4.2.11", "@smithy/smithy-client": "^4.12.3", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-QDA84CWNe8Akpj15ofLO+1N3Rfg8qa2K5uX0y6HnOp4AnRYRgWrKx/xzbYNbVF9ZsyJUYOfcoaN3y93wA/QJ2A=="],
"@smithy/util-endpoints": ["@smithy/util-endpoints@3.3.2", "", { "dependencies": { "@smithy/node-config-provider": "^4.3.11", "@smithy/types": "^4.13.0", "tslib": "^2.6.2" } }, "sha512-+4HFLpE5u29AbFlTdlKIT7jfOzZ8PDYZKTb3e+AgLz986OYwqTourQ5H+jg79/66DB69Un1+qKecLnkZdAsYcA=="],
@@ -1028,6 +1086,8 @@
"@sveltejs/acorn-typescript": ["@sveltejs/acorn-typescript@1.0.9", "", { "peerDependencies": { "acorn": "^8.9.0" } }, "sha512-lVJX6qEgs/4DOcRTpo56tmKzVPtoWAaVbL4hfO7t7NVwl9AAXzQR6cihesW1BmNMPl+bK6dreu2sOKBP2Q9CIA=="],
+ "@sveltejs/adapter-static": ["@sveltejs/adapter-static@3.0.10", "", { "peerDependencies": { "@sveltejs/kit": "^2.0.0" } }, "sha512-7D9lYFWJmB7zxZyTE/qxjksvMqzMuYrrsyh1f4AlZqeZeACPRySjbC3aFiY55wb1tWUaKOQG9PVbm74JcN2Iew=="],
+
"@sveltejs/adapter-vercel": ["@sveltejs/adapter-vercel@6.3.3", "", { "dependencies": { "@vercel/nft": "^1.3.2", "esbuild": "^0.25.4" }, "peerDependencies": { "@sveltejs/kit": "^2.4.0" } }, "sha512-jI7jT/XqRyFe9oqKvFcNPQfyNBi3pXqN1iQXa2lmeKT5Vzgr9iSOqJOD3pXf/9Q2Os6SXzqYYm6osRjHYEhkyw=="],
"@sveltejs/kit": ["@sveltejs/kit@2.53.4", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "@sveltejs/acorn-typescript": "^1.0.5", "@types/cookie": "^0.6.0", "acorn": "^8.14.1", "cookie": "^0.6.0", "devalue": "^5.6.3", "esm-env": "^1.2.2", "kleur": "^4.1.5", "magic-string": "^0.30.5", "mrmime": "^2.0.0", "set-cookie-parser": "^3.0.0", "sirv": "^3.0.0" }, "peerDependencies": { "@opentelemetry/api": "^1.0.0", "@sveltejs/vite-plugin-svelte": "^3.0.0 || ^4.0.0-next.1 || ^5.0.0 || ^6.0.0-next.0 || ^7.0.0", "svelte": "^4.0.0 || ^5.0.0-next.0", "typescript": "^5.3.3", "vite": "^5.0.3 || ^6.0.0 || ^7.0.0-beta.0 || ^8.0.0" }, "optionalPeers": ["@opentelemetry/api", "typescript"], "bin": { "svelte-kit": "svelte-kit.js" } }, "sha512-iAIPEahFgDJJyvz8g0jP08KvqnM6JvdW8YfsygZ+pMeMvyM2zssWMltcsotETvjSZ82G3VlitgDtBIvpQSZrTA=="],
@@ -1074,6 +1134,30 @@
"@tanstack/react-query": ["@tanstack/react-query@5.90.21", "", { "dependencies": { "@tanstack/query-core": "5.90.20" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-0Lu6y5t+tvlTJMTO7oh5NSpJfpg/5D41LlThfepTixPYkJ0sE2Jj0m0f6yYqujBwIXlId87e234+MxG3D3g7kg=="],
+ "@tauri-apps/cli": ["@tauri-apps/cli@2.10.1", "", { "optionalDependencies": { "@tauri-apps/cli-darwin-arm64": "2.10.1", "@tauri-apps/cli-darwin-x64": "2.10.1", "@tauri-apps/cli-linux-arm-gnueabihf": "2.10.1", "@tauri-apps/cli-linux-arm64-gnu": "2.10.1", "@tauri-apps/cli-linux-arm64-musl": "2.10.1", "@tauri-apps/cli-linux-riscv64-gnu": "2.10.1", "@tauri-apps/cli-linux-x64-gnu": "2.10.1", "@tauri-apps/cli-linux-x64-musl": "2.10.1", "@tauri-apps/cli-win32-arm64-msvc": "2.10.1", "@tauri-apps/cli-win32-ia32-msvc": "2.10.1", "@tauri-apps/cli-win32-x64-msvc": "2.10.1" }, "bin": { "tauri": "tauri.js" } }, "sha512-jQNGF/5quwORdZSSLtTluyKQ+o6SMa/AUICfhf4egCGFdMHqWssApVgYSbg+jmrZoc8e1DscNvjTnXtlHLS11g=="],
+
+ "@tauri-apps/cli-darwin-arm64": ["@tauri-apps/cli-darwin-arm64@2.10.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-Z2OjCXiZ+fbYZy7PmP3WRnOpM9+Fy+oonKDEmUE6MwN4IGaYqgceTjwHucc/kEEYZos5GICve35f7ZiizgqEnQ=="],
+
+ "@tauri-apps/cli-darwin-x64": ["@tauri-apps/cli-darwin-x64@2.10.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-V/irQVvjPMGOTQqNj55PnQPVuH4VJP8vZCN7ajnj+ZS8Kom1tEM2hR3qbbIRoS3dBKs5mbG8yg1WC+97dq17Pw=="],
+
+ "@tauri-apps/cli-linux-arm-gnueabihf": ["@tauri-apps/cli-linux-arm-gnueabihf@2.10.1", "", { "os": "linux", "cpu": "arm" }, "sha512-Hyzwsb4VnCWKGfTw+wSt15Z2pLw2f0JdFBfq2vHBOBhvg7oi6uhKiF87hmbXOBXUZaGkyRDkCHsdzJcIfoJC2w=="],
+
+ "@tauri-apps/cli-linux-arm64-gnu": ["@tauri-apps/cli-linux-arm64-gnu@2.10.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-OyOYs2t5GkBIvyWjA1+h4CZxTcdz1OZPCWAPz5DYEfB0cnWHERTnQ/SLayQzncrT0kwRoSfSz9KxenkyJoTelA=="],
+
+ "@tauri-apps/cli-linux-arm64-musl": ["@tauri-apps/cli-linux-arm64-musl@2.10.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-MIj78PDDGjkg3NqGptDOGgfXks7SYJwhiMh8SBoZS+vfdz7yP5jN18bNaLnDhsVIPARcAhE1TlsZe/8Yxo2zqg=="],
+
+ "@tauri-apps/cli-linux-riscv64-gnu": ["@tauri-apps/cli-linux-riscv64-gnu@2.10.1", "", { "os": "linux", "cpu": "none" }, "sha512-X0lvOVUg8PCVaoEtEAnpxmnkwlE1gcMDTqfhbefICKDnOTJ5Est3qL0SrWxizDackIOKBcvtpejrSiVpuJI1kw=="],
+
+ "@tauri-apps/cli-linux-x64-gnu": ["@tauri-apps/cli-linux-x64-gnu@2.10.1", "", { "os": "linux", "cpu": "x64" }, "sha512-2/12bEzsJS9fAKybxgicCDFxYD1WEI9kO+tlDwX5znWG2GwMBaiWcmhGlZ8fi+DMe9CXlcVarMTYc0L3REIRxw=="],
+
+ "@tauri-apps/cli-linux-x64-musl": ["@tauri-apps/cli-linux-x64-musl@2.10.1", "", { "os": "linux", "cpu": "x64" }, "sha512-Y8J0ZzswPz50UcGOFuXGEMrxbjwKSPgXftx5qnkuMs2rmwQB5ssvLb6tn54wDSYxe7S6vlLob9vt0VKuNOaCIQ=="],
+
+ "@tauri-apps/cli-win32-arm64-msvc": ["@tauri-apps/cli-win32-arm64-msvc@2.10.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-iSt5B86jHYAPJa/IlYw++SXtFPGnWtFJriHn7X0NFBVunF6zu9+/zOn8OgqIWSl8RgzhLGXQEEtGBdR4wzpVgg=="],
+
+ "@tauri-apps/cli-win32-ia32-msvc": ["@tauri-apps/cli-win32-ia32-msvc@2.10.1", "", { "os": "win32", "cpu": "ia32" }, "sha512-gXyxgEzsFegmnWywYU5pEBURkcFN/Oo45EAwvZrHMh+zUSEAvO5E8TXsgPADYm31d1u7OQU3O3HsYfVBf2moHw=="],
+
+ "@tauri-apps/cli-win32-x64-msvc": ["@tauri-apps/cli-win32-x64-msvc@2.10.1", "", { "os": "win32", "cpu": "x64" }, "sha512-6Cn7YpPFwzChy0ERz6djKEmUehWrYlM+xTaNzGPgZocw3BD7OfwfWHKVWxXzdjEW2KfKkHddfdxK1XXTYqBRLg=="],
+
"@tmcp/adapter-zod": ["@tmcp/adapter-zod@0.1.7", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "@types/json-schema": "^7.0.15", "zod": "^4.0.5" }, "peerDependencies": { "tmcp": "^1.16.1" } }, "sha512-XF7f9D9HFbKJDUwRHYSd1IaxGaTaWuFjTbwAfUUrh/ejucewvHClMXMxxe9mQYRFu3yOoPCQxmC9ZovVzuwMlQ=="],
"@tmcp/session-manager": ["@tmcp/session-manager@0.2.1", "", { "peerDependencies": { "tmcp": "^1.16.3" } }, "sha512-DOGy9LfufXCy1wfpGHZ6qPSDQtRnTVwOb71+41ffovTqzLMZlK3iLK/LIsekHxIiku+iIAUiqEKN+DHbqEm8IA=="],
@@ -1330,7 +1414,7 @@
"camelcase": ["camelcase@5.3.1", "", {}, "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg=="],
- "caniuse-lite": ["caniuse-lite@1.0.30001776", "", {}, "sha512-sg01JDPzZ9jGshqKSckOQthXnYwOEP50jeVFhaSFbZcOy05TiuuaffDOfcwtCisJ9kNQuLBFibYywv2Bgm9osw=="],
+ "caniuse-lite": ["caniuse-lite@1.0.30001777", "", {}, "sha512-tmN+fJxroPndC74efCdp12j+0rk0RHwV5Jwa1zWaFVyw2ZxAuPeG8ZgWC3Wz7uSjT3qMRQ5XHZ4COgQmsCMJAQ=="],
"ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="],
@@ -1384,7 +1468,7 @@
"comma-separated-tokens": ["comma-separated-tokens@2.0.3", "", {}, "sha512-Fu4hJdvzeylCfQPp9SGWidpzrMs7tTrlu6Vb8XGaRGck8QSNZJJp538Wrb60Lax4fPwR64ViY468OIUTbRlGZg=="],
- "commander": ["commander@14.0.1", "", {}, "sha512-2JkV3gUZUVrbNA+1sjBOYLsMZ5cEEl8GTFP2a4AVz5hvasAMCQ1D2l2le/cX+pV4N6ZU17zjUahLpIXRrnWL8A=="],
+ "commander": ["commander@14.0.3", "", {}, "sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw=="],
"compressjs": ["compressjs@1.0.3", "", { "dependencies": { "amdefine": "~1.0.0", "commander": "~2.8.1" }, "bin": { "compressjs": "./bin/compressjs" } }, "sha512-jpKJjBTretQACTGLNuvnozP1JdP2ZLrjdGdBgk/tz1VfXlUcBhhSZW6vEsuThmeot/yjvSrPQKEgfF3X2Lpi8Q=="],
@@ -1520,13 +1604,13 @@
"es6-promisify": ["es6-promisify@5.0.0", "", { "dependencies": { "es6-promise": "^4.0.3" } }, "sha512-C+d6UdsYDk0lMebHNR4S2NybQMMngAOnOwYBQjTOiv0MkoJMP0Myw2mgpDLBcpfCmRLxyFqYhS/CfOENq4SJhQ=="],
- "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=="],
+ "esbuild": ["esbuild@0.27.0", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.27.0", "@esbuild/android-arm": "0.27.0", "@esbuild/android-arm64": "0.27.0", "@esbuild/android-x64": "0.27.0", "@esbuild/darwin-arm64": "0.27.0", "@esbuild/darwin-x64": "0.27.0", "@esbuild/freebsd-arm64": "0.27.0", "@esbuild/freebsd-x64": "0.27.0", "@esbuild/linux-arm": "0.27.0", "@esbuild/linux-arm64": "0.27.0", "@esbuild/linux-ia32": "0.27.0", "@esbuild/linux-loong64": "0.27.0", "@esbuild/linux-mips64el": "0.27.0", "@esbuild/linux-ppc64": "0.27.0", "@esbuild/linux-riscv64": "0.27.0", "@esbuild/linux-s390x": "0.27.0", "@esbuild/linux-x64": "0.27.0", "@esbuild/netbsd-arm64": "0.27.0", "@esbuild/netbsd-x64": "0.27.0", "@esbuild/openbsd-arm64": "0.27.0", "@esbuild/openbsd-x64": "0.27.0", "@esbuild/openharmony-arm64": "0.27.0", "@esbuild/sunos-x64": "0.27.0", "@esbuild/win32-arm64": "0.27.0", "@esbuild/win32-ia32": "0.27.0", "@esbuild/win32-x64": "0.27.0" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-jd0f4NHbD6cALCyGElNpGAOtWxSq46l9X/sWB0Nzd5er4Kz2YTm+Vl0qKFT9KUJvD8+fiO8AvoHhFvEatfVixA=="],
"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@2.0.0", "", {}, "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w=="],
+ "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
"esm-env": ["esm-env@1.2.2", "", {}, "sha512-Epxrv+Nr/CaL4ZcFGPJIYLWFom+YeV1DqMLHJoEd9SYRxNbaFruBwfEX/kkHUJf55j2+TUbmDcmuilbP1TmXHA=="],
@@ -1856,7 +1940,7 @@
"json5": ["json5@2.2.3", "", { "bin": { "json5": "lib/cli.js" } }, "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg=="],
- "just-bash": ["just-bash@2.12.4", "", { "dependencies": { "compressjs": "^1.0.3", "diff": "^8.0.2", "fast-xml-parser": "^5.3.3", "file-type": "^21.2.0", "ini": "^6.0.0", "minimatch": "^10.1.1", "modern-tar": "^0.7.3", "papaparse": "^5.5.3", "re2js": "^1.2.1", "smol-toml": "^1.6.0", "sprintf-js": "^1.1.3", "sql.js": "^1.13.0", "turndown": "^7.2.2", "yaml": "^2.8.2" }, "optionalDependencies": { "@mongodb-js/zstd": "^7.0.0", "node-liblzma": "^2.0.3" }, "bin": { "just-bash": "dist/bin/just-bash.js", "just-bash-shell": "dist/bin/shell/shell.js" } }, "sha512-TDF3pmEnVikwvA0x3w+MfpRmN5xRFhfnDb5rgQw2hbWLTyMXPgbMomfsLFdWew3oseCFmIXEGmc4NW/3tWG8mg=="],
+ "just-bash": ["just-bash@2.12.5", "", { "dependencies": { "compressjs": "^1.0.3", "diff": "^8.0.2", "fast-xml-parser": "^5.3.3", "file-type": "^21.2.0", "ini": "^6.0.0", "minimatch": "^10.1.1", "modern-tar": "^0.7.3", "papaparse": "^5.5.3", "re2js": "^1.2.1", "smol-toml": "^1.6.0", "sprintf-js": "^1.1.3", "sql.js": "^1.13.0", "turndown": "^7.2.2", "yaml": "^2.8.2" }, "optionalDependencies": { "@mongodb-js/zstd": "^7.0.0", "node-liblzma": "^2.0.3" }, "bin": { "just-bash": "dist/bin/just-bash.js", "just-bash-shell": "dist/bin/shell/shell.js" } }, "sha512-oycgMv4vsIjNbjC9xUnhSPkfXNN3gsi0zShdTDob/Sk0YC3L9kosDXZyg4gYQTyTbTg291ibzdU2U5M+qyAMMQ=="],
"kind-of": ["kind-of@3.2.2", "", { "dependencies": { "is-buffer": "^1.1.5" } }, "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ=="],
@@ -2328,7 +2412,7 @@
"smol-toml": ["smol-toml@1.6.0", "", {}, "sha512-4zemZi0HvTnYwLfrpk/CF9LOd9Lt87kAt50GnqhMpyF9U3poDAP2+iukq2bZsO/ufegbYehBkqINbsWxj4l4cw=="],
- "source-map": ["source-map@0.6.1", "", {}, "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g=="],
+ "source-map": ["source-map@0.5.7", "", {}, "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ=="],
"source-map-js": ["source-map-js@1.2.1", "", {}, "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA=="],
@@ -2436,9 +2520,9 @@
"tinyglobby": ["tinyglobby@0.2.15", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.3" } }, "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ=="],
- "tldts": ["tldts@7.0.24", "", { "dependencies": { "tldts-core": "^7.0.24" }, "bin": { "tldts": "bin/cli.js" } }, "sha512-1r6vQTTt1rUiJkI5vX7KG8PR342Ru/5Oh13kEQP2SMbRSZpOey9SrBe27IDxkoWulx8ShWu4K6C0BkctP8Z1bQ=="],
+ "tldts": ["tldts@7.0.25", "", { "dependencies": { "tldts-core": "^7.0.25" }, "bin": { "tldts": "bin/cli.js" } }, "sha512-keinCnPbwXEUG3ilrWQZU+CqcTTzHq9m2HhoUP2l7Xmi8l1LuijAXLpAJ5zRW+ifKTNscs4NdCkfkDCBYm352w=="],
- "tldts-core": ["tldts-core@7.0.24", "", {}, "sha512-pj7yygNMoMRqG7ML2SDQ0xNIOfN3IBDUcPVM2Sg6hP96oFNN2nqnzHreT3z9xLq85IWJyNTvD38O002DdOrPMw=="],
+ "tldts-core": ["tldts-core@7.0.25", "", {}, "sha512-ZjCZK0rppSBu7rjHYDYsEaMOIbbT+nWF57hKkv4IUmZWBNrBWBOjIElc0mKRgLM8bm7x/BBlof6t2gi/Oq/Asw=="],
"tmcp": ["tmcp@1.19.2", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "json-rpc-2.0": "^1.7.1", "sqids": "^0.3.0", "uri-template-matcher": "^1.1.1", "valibot": "^1.1.0" } }, "sha512-/AEG/jlzflGKqCKm7GNdhz50VtFlN+3vcnKd+iQJXcHdIPrpCRMolEj57SfgcXcfE2ouX7J6Q05gPKsS2NZhKg=="],
@@ -2634,6 +2718,8 @@
"@base-org/account/preact": ["preact@10.24.2", "", {}, "sha512-1cSoF0aCC8uaARATfrlz4VCBqE8LwZwRfLgkxJOQwAlQt6ayTmi0D9OF7nXid1POI5SZidFuG9CnlXbDfLqY/Q=="],
+ "@btca/web/@lucide/svelte": ["@lucide/svelte@0.577.0", "", { "peerDependencies": { "svelte": "^5" } }, "sha512-0P6mkySd2MapIEgq08tADPmcN4DHndC/02PWwaLkOerXlx5Sv9aT4BxyXLIY+eccr0g/nEyCYiJesqS61YdBZQ=="],
+
"@clerk/shared/csstype": ["csstype@3.1.3", "", {}, "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="],
"@clerk/shared/swr": ["swr@2.3.4", "", { "dependencies": { "dequal": "^2.0.3", "use-sync-external-store": "^1.4.0" }, "peerDependencies": { "react": "^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-bYd2lrhc+VarcpkgWclcUi92wYCpOgMws9Sd1hG1ntAu0NEy+14CbotuFjshBU2kt9rYj9TSmDcybpxpeTU1fg=="],
@@ -2646,10 +2732,6 @@
"@emotion/babel-plugin/convert-source-map": ["convert-source-map@1.9.0", "", {}, "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A=="],
- "@emotion/babel-plugin/escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
-
- "@emotion/babel-plugin/source-map": ["source-map@0.5.7", "", {}, "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ=="],
-
"@emotion/serialize/@emotion/memoize": ["@emotion/memoize@0.9.0", "", {}, "sha512-30FAj7/EoJ5mwVPOWhAyCX+FPfMDrVecJAM+Iw9NRoSl4BBAQeqj4cApHHUXOVvIPgLVDsCFoz/hGD+5QQD1GQ=="],
"@inquirer/checkbox/@inquirer/ansi": ["@inquirer/ansi@1.0.2", "", {}, "sha512-S8qNSZiYzFd0wAcyG5AXCvUHC5Sr7xpZ9wZ2py9XR88jUz8wooStVx5M6dRzczbBWjic9NP7+rY0Xi7qqK/aMQ=="],
@@ -2702,14 +2784,8 @@
"@inquirer/search/@inquirer/type": ["@inquirer/type@3.0.10", "", { "peerDependencies": { "@types/node": ">=18" }, "optionalPeers": ["@types/node"] }, "sha512-BvziSRxfz5Ov8ch0z/n3oijRSEcEsHnhggm4xFZe93DHcUCTlutlq9Ox4SVENAfcRD22UQq7T/atg9Wr3k09eA=="],
- "@jest/environment/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "@jest/fake-timers/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"@jest/transform/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="],
- "@jest/types/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"@jest/types/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="],
"@jimp/core/file-type": ["file-type@16.5.4", "", { "dependencies": { "readable-web-to-node-stream": "^3.0.0", "strtok3": "^6.2.4", "token-types": "^4.1.1" } }, "sha512-/yFHK0aGjFEgDJjEKP0pWCplsPFPhwyfwevf/pVxiN0tmE4L9LmwWxWukdJSHdoCli4VgQLehjJtwQBnqmsKcw=="],
@@ -2760,21 +2836,39 @@
"@opentelemetry/exporter-logs-otlp-proto/@opentelemetry/sdk-logs": ["@opentelemetry/sdk-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api-logs": "0.207.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.4.0 <1.10.0" } }, "sha512-4MEQmn04y+WFe6cyzdrXf58hZxilvY59lzZj2AccuHW/+BxLn/rGVN/Irsi/F0qfBOpMOrrCLKTExoSL2zoQmg=="],
+ "@opentelemetry/exporter-logs-otlp-proto/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentelemetry/exporter-trace-otlp-grpc/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentelemetry/exporter-trace-otlp-http/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentelemetry/exporter-trace-otlp-proto/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentelemetry/exporter-zipkin/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
"@opentelemetry/instrumentation/@opentelemetry/api-logs": ["@opentelemetry/api-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api": "^1.3.0" } }, "sha512-lAb0jQRVyleQQGiuuvCOTDVspc14nx6XJjP4FspJ1sNARo3Regq4ZZbrc3rN4b1TYSuUCvgH+UXUPug4SLOqEQ=="],
"@opentelemetry/otlp-transformer/@opentelemetry/api-logs": ["@opentelemetry/api-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api": "^1.3.0" } }, "sha512-lAb0jQRVyleQQGiuuvCOTDVspc14nx6XJjP4FspJ1sNARo3Regq4ZZbrc3rN4b1TYSuUCvgH+UXUPug4SLOqEQ=="],
"@opentelemetry/otlp-transformer/@opentelemetry/sdk-logs": ["@opentelemetry/sdk-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api-logs": "0.207.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.4.0 <1.10.0" } }, "sha512-4MEQmn04y+WFe6cyzdrXf58hZxilvY59lzZj2AccuHW/+BxLn/rGVN/Irsi/F0qfBOpMOrrCLKTExoSL2zoQmg=="],
+ "@opentelemetry/otlp-transformer/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
"@opentelemetry/sdk-node/@opentelemetry/api-logs": ["@opentelemetry/api-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api": "^1.3.0" } }, "sha512-lAb0jQRVyleQQGiuuvCOTDVspc14nx6XJjP4FspJ1sNARo3Regq4ZZbrc3rN4b1TYSuUCvgH+UXUPug4SLOqEQ=="],
"@opentelemetry/sdk-node/@opentelemetry/exporter-logs-otlp-http": ["@opentelemetry/exporter-logs-otlp-http@0.207.0", "", { "dependencies": { "@opentelemetry/api-logs": "0.207.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/otlp-exporter-base": "0.207.0", "@opentelemetry/otlp-transformer": "0.207.0", "@opentelemetry/sdk-logs": "0.207.0" }, "peerDependencies": { "@opentelemetry/api": "^1.3.0" } }, "sha512-JpOh7MguEUls8eRfkVVW3yRhClo5b9LqwWTOg8+i4gjr/+8eiCtquJnC7whvpTIGyff06cLZ2NsEj+CVP3Mjeg=="],
"@opentelemetry/sdk-node/@opentelemetry/sdk-logs": ["@opentelemetry/sdk-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api-logs": "0.207.0", "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.4.0 <1.10.0" } }, "sha512-4MEQmn04y+WFe6cyzdrXf58hZxilvY59lzZj2AccuHW/+BxLn/rGVN/Irsi/F0qfBOpMOrrCLKTExoSL2zoQmg=="],
- "@opentui/core/marked": ["marked@17.0.1", "", { "bin": { "marked": "bin/marked.js" } }, "sha512-boeBdiS0ghpWcSwoNm/jJBwdpFaMnZWRzjA6SkUMYb40SVaN1x7mmfGKp0jvexGcx+7y2La5zRZsYFZI6Qpypg=="],
+ "@opentelemetry/sdk-node/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentelemetry/sdk-trace-base/@opentelemetry/core": ["@opentelemetry/core@2.6.0", "", { "dependencies": { "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.0.0 <1.10.0" } }, "sha512-HLM1v2cbZ4TgYN6KEOj+Bbj8rAKriOdkF9Ed3tG25FoprSiQl7kYc+RRT6fUZGOvx0oMi5U67GoFdT+XUn8zEg=="],
- "@opentui/core/web-tree-sitter": ["web-tree-sitter@0.25.10", "", { "peerDependencies": { "@types/emscripten": "^1.40.0" }, "optionalPeers": ["@types/emscripten"] }, "sha512-Y09sF44/13XvgVKgO2cNDw5rGk6s26MgoZPXLESvMXeefBf7i6/73eFurre0IsTW6E14Y0ArIzhUMmjoc7xyzA=="],
+ "@opentelemetry/sdk-trace-base/@opentelemetry/resources": ["@opentelemetry/resources@2.6.0", "", { "dependencies": { "@opentelemetry/core": "2.6.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-D4y/+OGe3JSuYUCBxtH5T9DSAWNcvCb/nQWIga8HNtXTVPQn59j0nTBAgaAXxUVBDl40mG3Tc76b46wPlZaiJQ=="],
+
+ "@opentelemetry/sdk-trace-node/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
+ "@opentui/core/marked": ["marked@17.0.1", "", { "bin": { "marked": "bin/marked.js" } }, "sha512-boeBdiS0ghpWcSwoNm/jJBwdpFaMnZWRzjA6SkUMYb40SVaN1x7mmfGKp0jvexGcx+7y2La5zRZsYFZI6Qpypg=="],
"@oslojs/jwt/@oslojs/encoding": ["@oslojs/encoding@0.4.1", "", {}, "sha512-hkjo6MuIK/kQR5CrGNdAPZhS01ZCXuWDRJ187zh6qqF2+yMHZpD9fAYpX8q2bOO6Ryhl3XpCT6kUX76N8hhm4Q=="],
@@ -2806,6 +2900,8 @@
"@solana/web3.js/buffer": ["buffer@6.0.3", "", { "dependencies": { "base64-js": "^1.3.1", "ieee754": "^1.2.1" } }, "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA=="],
+ "@sveltejs/adapter-vercel/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=="],
+
"@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.8.1", "", { "dependencies": { "@emnapi/wasi-threads": "1.1.0", "tslib": "^2.4.0" }, "bundled": true }, "sha512-AvT9QFpxK0Zd8J0jopedNm+w/2fIzvtPKPjqyw9jwvBaReTTqPBk9Hixaz7KbjimP+QNz605/XnjFcDAL2pqBg=="],
"@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.8.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-mehfKSMWjjNol8659Z8KxEMrdSJDDot5SXMq00dM8BN4o+CLNXQ0xH2V7EchNHV4RmbZLmmPdEaXZc5H2FXmDg=="],
@@ -2820,12 +2916,6 @@
"@tanstack/react-query/@tanstack/query-core": ["@tanstack/query-core@5.90.20", "", {}, "sha512-OMD2HLpNouXEfZJWcKeVKUgQ5n+n3A2JFmBaScpNDUqSrQSjiveC7dKMe53uJUg1nDG16ttFPz2xfilz6i2uVg=="],
- "@types/connect/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "@types/graceful-fs/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "@types/ws/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"@wallet-standard/errors/commander": ["commander@13.1.0", "", {}, "sha512-/rFeCpNJQbhSZjGVwO9RFV3xPqbnERS8MmIQzCtD/zl6gpJuV/bMLuN92oG3F7d8oDEHHRrujSXNUr8fpjntKw=="],
"accepts/mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="],
@@ -2838,18 +2928,8 @@
"btca-server/zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="],
- "bun-types/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "chrome-launcher/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "chrome-launcher/escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
-
"chrome-launcher/is-wsl": ["is-wsl@2.2.0", "", { "dependencies": { "is-docker": "^2.0.0" } }, "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww=="],
- "chromium-edge-launcher/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "chromium-edge-launcher/escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
-
"chromium-edge-launcher/is-wsl": ["is-wsl@2.2.0", "", { "dependencies": { "is-docker": "^2.0.0" } }, "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww=="],
"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=="],
@@ -2864,8 +2944,6 @@
"connect/debug": ["debug@2.6.9", "", { "dependencies": { "ms": "2.0.0" } }, "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA=="],
- "convex/esbuild": ["esbuild@0.27.0", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.27.0", "@esbuild/android-arm": "0.27.0", "@esbuild/android-arm64": "0.27.0", "@esbuild/android-x64": "0.27.0", "@esbuild/darwin-arm64": "0.27.0", "@esbuild/darwin-x64": "0.27.0", "@esbuild/freebsd-arm64": "0.27.0", "@esbuild/freebsd-x64": "0.27.0", "@esbuild/linux-arm": "0.27.0", "@esbuild/linux-arm64": "0.27.0", "@esbuild/linux-ia32": "0.27.0", "@esbuild/linux-loong64": "0.27.0", "@esbuild/linux-mips64el": "0.27.0", "@esbuild/linux-ppc64": "0.27.0", "@esbuild/linux-riscv64": "0.27.0", "@esbuild/linux-s390x": "0.27.0", "@esbuild/linux-x64": "0.27.0", "@esbuild/netbsd-arm64": "0.27.0", "@esbuild/netbsd-x64": "0.27.0", "@esbuild/openbsd-arm64": "0.27.0", "@esbuild/openbsd-x64": "0.27.0", "@esbuild/openharmony-arm64": "0.27.0", "@esbuild/sunos-x64": "0.27.0", "@esbuild/win32-arm64": "0.27.0", "@esbuild/win32-ia32": "0.27.0", "@esbuild/win32-x64": "0.27.0" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-jd0f4NHbD6cALCyGElNpGAOtWxSq46l9X/sWB0Nzd5er4Kz2YTm+Vl0qKFT9KUJvD8+fiO8AvoHhFvEatfVixA=="],
-
"cosmiconfig/yaml": ["yaml@1.10.2", "", {}, "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg=="],
"crypto-random-string/type-fest": ["type-fest@1.4.0", "", {}, "sha512-yGSza74xk0UG8k+pLh5oeoYirvIiWo5t0/o3zHHAO2tRDiZcxWP7fywNlXhqb6/r6sWvwi+RsyQMWhVLe4BVuA=="],
@@ -2888,8 +2966,6 @@
"ink/react-reconciler": ["react-reconciler@0.33.0", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.0" } }, "sha512-KetWRytFv1epdpJc3J4G75I4WrplZE5jOL7Yq0p34+OVOKF4Se7WrdIdVC45XsSSmUTlht2FM/fM1FZb1mfQeA=="],
- "ink/ws": ["ws@8.19.0", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-blAT2mjOEIi0ZzruJfIhb3nps74PRWTCz1IjglWEEpQl5XS/UNama6u2/rjFkDDouqr4L67ry+1aGIALViWjDg=="],
-
"ink-confirm-input/ink-text-input": ["ink-text-input@3.3.0", "", { "dependencies": { "chalk": "^3.0.0", "prop-types": "^15.5.10" }, "peerDependencies": { "ink": "^2.0.0", "react": "^16.5.2" } }, "sha512-gO4wrOf2ie3YuEARTIwGlw37lMjFn3Gk6CKIDrMlHb46WFMagZU7DplohjM24zynlqfnXA5UDEIfC2NBcvD8kg=="],
"ink-spinner/cli-spinners": ["cli-spinners@2.9.2", "", {}, "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg=="],
@@ -2914,16 +2990,8 @@
"jayson/ws": ["ws@7.5.10", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": "^5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ=="],
- "jest-environment-node/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "jest-haste-map/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"jest-message-util/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="],
- "jest-mock/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
- "jest-util/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"jest-util/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="],
"jest-util/ci-info": ["ci-info@3.9.0", "", {}, "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ=="],
@@ -2934,8 +3002,6 @@
"jest-validate/chalk": ["chalk@4.1.2", "", { "dependencies": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA=="],
- "jest-worker/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"lighthouse-logger/debug": ["debug@2.6.9", "", { "dependencies": { "ms": "2.0.0" } }, "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA=="],
"lru-cache/yallist": ["yallist@3.1.1", "", {}, "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="],
@@ -2950,16 +3016,10 @@
"metro/mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="],
- "metro/source-map": ["source-map@0.5.7", "", {}, "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ=="],
-
"metro/ws": ["ws@7.5.10", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": "^5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ=="],
"metro-babel-transformer/hermes-parser": ["hermes-parser@0.33.3", "", { "dependencies": { "hermes-estree": "0.33.3" } }, "sha512-Yg3HgaG4CqgyowtYjX/FsnPAuZdHOqSMtnbpylbptsQ9nwwSKsy6uRWcGO5RK0EqiX12q8HvDWKgeAVajRO5DA=="],
- "metro-source-map/source-map": ["source-map@0.5.7", "", {}, "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ=="],
-
- "metro-symbolicate/source-map": ["source-map@0.5.7", "", {}, "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ=="],
-
"micromatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
"node-abi/semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="],
@@ -2976,16 +3036,10 @@
"postcss/nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="],
- "posthog-js/@opentelemetry/resources": ["@opentelemetry/resources@2.6.0", "", { "dependencies": { "@opentelemetry/core": "2.6.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-D4y/+OGe3JSuYUCBxtH5T9DSAWNcvCb/nQWIga8HNtXTVPQn59j0nTBAgaAXxUVBDl40mG3Tc76b46wPlZaiJQ=="],
-
- "posthog-js/core-js": ["core-js@3.48.0", "", {}, "sha512-zpEHTy1fjTMZCKLHUZoVeylt9XrzaIN2rbPXEt0k+q7JE5CkCZdo6bNq55bn24a69CH7ErAVLKijxJja4fw+UQ=="],
-
"pretty-format/ansi-styles": ["ansi-styles@5.2.0", "", {}, "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA=="],
"pretty-format/react-is": ["react-is@18.3.1", "", {}, "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg=="],
- "protobufjs/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
-
"qrcode/pngjs": ["pngjs@5.0.0", "", {}, "sha512-40QW5YalBNfQo5yRYmiw7Yz6TKKVr3h6970B2YE+3fQpsWcrbj1PzJgxeJ19DRQjhMbKPIuMY8rFaXc8moolVw=="],
"qrcode/yargs": ["yargs@15.4.1", "", { "dependencies": { "cliui": "^6.0.0", "decamelize": "^1.2.0", "find-up": "^4.1.0", "get-caller-file": "^2.0.1", "require-directory": "^2.1.1", "require-main-filename": "^2.0.0", "set-blocking": "^2.0.0", "string-width": "^4.2.0", "which-module": "^2.0.0", "y18n": "^4.0.0", "yargs-parser": "^18.1.2" } }, "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A=="],
@@ -3030,6 +3084,10 @@
"send/statuses": ["statuses@2.0.2", "", {}, "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw=="],
+ "source-map-support/source-map": ["source-map@0.6.1", "", {}, "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g=="],
+
+ "stack-utils/escape-string-regexp": ["escape-string-regexp@2.0.0", "", {}, "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w=="],
+
"stacktrace-parser/type-fest": ["type-fest@0.7.1", "", {}, "sha512-Ne2YiiGN8bmrmJJEuTWTLJR32nh/JdL1+PSicowtNb0WFpn59GK8/lfD61bVtzguz7b3PBt74nxpv/Pw5po5Rg=="],
"subsume/escape-string-regexp": ["escape-string-regexp@5.0.0", "", {}, "sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw=="],
@@ -3058,8 +3116,6 @@
"viem/ws": ["ws@8.18.3", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-PEIGCY5tSlUt50cqyMXfCzX+oOPqN0vuGqWzbcJ2xvnkzkq46oOpz7dQaTDBdfICb4N14+GARUDw2XV2N4tvzg=="],
- "vite/esbuild": ["esbuild@0.27.0", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.27.0", "@esbuild/android-arm": "0.27.0", "@esbuild/android-arm64": "0.27.0", "@esbuild/android-x64": "0.27.0", "@esbuild/darwin-arm64": "0.27.0", "@esbuild/darwin-x64": "0.27.0", "@esbuild/freebsd-arm64": "0.27.0", "@esbuild/freebsd-x64": "0.27.0", "@esbuild/linux-arm": "0.27.0", "@esbuild/linux-arm64": "0.27.0", "@esbuild/linux-ia32": "0.27.0", "@esbuild/linux-loong64": "0.27.0", "@esbuild/linux-mips64el": "0.27.0", "@esbuild/linux-ppc64": "0.27.0", "@esbuild/linux-riscv64": "0.27.0", "@esbuild/linux-s390x": "0.27.0", "@esbuild/linux-x64": "0.27.0", "@esbuild/netbsd-arm64": "0.27.0", "@esbuild/netbsd-x64": "0.27.0", "@esbuild/openbsd-arm64": "0.27.0", "@esbuild/openbsd-x64": "0.27.0", "@esbuild/openharmony-arm64": "0.27.0", "@esbuild/sunos-x64": "0.27.0", "@esbuild/win32-arm64": "0.27.0", "@esbuild/win32-ia32": "0.27.0", "@esbuild/win32-x64": "0.27.0" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-jd0f4NHbD6cALCyGElNpGAOtWxSq46l9X/sWB0Nzd5er4Kz2YTm+Vl0qKFT9KUJvD8+fiO8AvoHhFvEatfVixA=="],
-
"wrap-ansi/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=="],
"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=="],
@@ -3156,6 +3212,8 @@
"@opentelemetry/exporter-logs-otlp-grpc/@opentelemetry/sdk-logs/@opentelemetry/api-logs": ["@opentelemetry/api-logs@0.207.0", "", { "dependencies": { "@opentelemetry/api": "^1.3.0" } }, "sha512-lAb0jQRVyleQQGiuuvCOTDVspc14nx6XJjP4FspJ1sNARo3Regq4ZZbrc3rN4b1TYSuUCvgH+UXUPug4SLOqEQ=="],
+ "@opentelemetry/exporter-logs-otlp-http/@opentelemetry/otlp-transformer/@opentelemetry/sdk-trace-base": ["@opentelemetry/sdk-trace-base@2.2.0", "", { "dependencies": { "@opentelemetry/core": "2.2.0", "@opentelemetry/resources": "2.2.0", "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.3.0 <1.10.0" } }, "sha512-xWQgL0Bmctsalg6PaXExmzdedSp3gyKV8mQBwK/j9VGdCDu2fmXIb2gAehBKbkXCpJ4HPkgv3QfoJWRT4dHWbw=="],
+
"@react-native/dev-middleware/open/is-docker": ["is-docker@2.2.1", "", { "bin": { "is-docker": "cli.js" } }, "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ=="],
"@react-native/dev-middleware/open/is-wsl": ["is-wsl@2.2.0", "", { "dependencies": { "is-docker": "^2.0.0" } }, "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww=="],
@@ -3164,79 +3222,81 @@
"@solana-mobile/wallet-standard-mobile/bs58/base-x": ["base-x@4.0.1", "", {}, "sha512-uAZ8x6r6S3aUM9rbHGVOIsR15U/ZSc82b3ymnCPsT45Gk1DDvhDPdIgB5MrhirZWt+5K0EEPQH985kNqZgNPFw=="],
+ "@solana/codecs-strings/@solana/errors/commander": ["commander@14.0.1", "", {}, "sha512-2JkV3gUZUVrbNA+1sjBOYLsMZ5cEEl8GTFP2a4AVz5hvasAMCQ1D2l2le/cX+pV4N6ZU17zjUahLpIXRrnWL8A=="],
+
"@solana/wallet-standard-wallet-adapter-base/bs58/base-x": ["base-x@4.0.1", "", {}, "sha512-uAZ8x6r6S3aUM9rbHGVOIsR15U/ZSc82b3ymnCPsT45Gk1DDvhDPdIgB5MrhirZWt+5K0EEPQH985kNqZgNPFw=="],
- "accepts/mime-types/mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.25.12", "", { "os": "aix", "cpu": "ppc64" }, "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA=="],
- "babel-jest/chalk/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.25.12", "", { "os": "android", "cpu": "arm" }, "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg=="],
- "babel-jest/chalk/supports-color": ["supports-color@7.2.0", "", { "dependencies": { "has-flag": "^4.0.0" } }, "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.25.12", "", { "os": "android", "cpu": "arm64" }, "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg=="],
- "chrome-launcher/is-wsl/is-docker": ["is-docker@2.2.1", "", { "bin": { "is-docker": "cli.js" } }, "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.25.12", "", { "os": "android", "cpu": "x64" }, "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg=="],
- "chromium-edge-launcher/is-wsl/is-docker": ["is-docker@2.2.1", "", { "bin": { "is-docker": "cli.js" } }, "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.25.12", "", { "os": "darwin", "cpu": "arm64" }, "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg=="],
- "cliui/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.25.12", "", { "os": "darwin", "cpu": "x64" }, "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA=="],
- "cliui/string-width/is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.25.12", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg=="],
- "cliui/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.25.12", "", { "os": "freebsd", "cpu": "x64" }, "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ=="],
- "cliui/wrap-ansi/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.25.12", "", { "os": "linux", "cpu": "arm" }, "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw=="],
- "connect/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.25.12", "", { "os": "linux", "cpu": "arm64" }, "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ=="],
- "convex/esbuild/@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.27.0", "", { "os": "aix", "cpu": "ppc64" }, "sha512-KuZrd2hRjz01y5JK9mEBSD3Vj3mbCvemhT466rSuJYeE/hjuBrHfjjcjMdTm/sz7au+++sdbJZJmuBwQLuw68A=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.25.12", "", { "os": "linux", "cpu": "ia32" }, "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA=="],
- "convex/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.27.0", "", { "os": "android", "cpu": "arm" }, "sha512-j67aezrPNYWJEOHUNLPj9maeJte7uSMM6gMoxfPC9hOg8N02JuQi/T7ewumf4tNvJadFkvLZMlAq73b9uwdMyQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng=="],
- "convex/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.27.0", "", { "os": "android", "cpu": "arm64" }, "sha512-CC3vt4+1xZrs97/PKDkl0yN7w8edvU2vZvAFGD16n9F0Cvniy5qvzRXjfO1l94efczkkQE6g1x0i73Qf5uthOQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw=="],
- "convex/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.27.0", "", { "os": "android", "cpu": "x64" }, "sha512-wurMkF1nmQajBO1+0CJmcN17U4BP6GqNSROP8t0X/Jiw2ltYGLHpEksp9MpoBqkrFR3kv2/te6Sha26k3+yZ9Q=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.25.12", "", { "os": "linux", "cpu": "ppc64" }, "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA=="],
- "convex/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.27.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-uJOQKYCcHhg07DL7i8MzjvS2LaP7W7Pn/7uA0B5S1EnqAirJtbyw4yC5jQ5qcFjHK9l6o/MX9QisBg12kNkdHg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.25.12", "", { "os": "linux", "cpu": "none" }, "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w=="],
- "convex/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.27.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-8mG6arH3yB/4ZXiEnXof5MK72dE6zM9cDvUcPtxhUZsDjESl9JipZYW60C3JGreKCEP+p8P/72r69m4AZGJd5g=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.25.12", "", { "os": "linux", "cpu": "s390x" }, "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg=="],
- "convex/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.27.0", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-9FHtyO988CwNMMOE3YIeci+UV+x5Zy8fI2qHNpsEtSF83YPBmE8UWmfYAQg6Ux7Gsmd4FejZqnEUZCMGaNQHQw=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.25.12", "", { "os": "linux", "cpu": "x64" }, "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw=="],
- "convex/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.27.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-zCMeMXI4HS/tXvJz8vWGexpZj2YVtRAihHLk1imZj4efx1BQzN76YFeKqlDr3bUWI26wHwLWPd3rwh6pe4EV7g=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg=="],
- "convex/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.27.0", "", { "os": "linux", "cpu": "arm" }, "sha512-t76XLQDpxgmq2cNXKTVEB7O7YMb42atj2Re2Haf45HkaUpjM2J0UuJZDuaGbPbamzZ7bawyGFUkodL+zcE+jvQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.25.12", "", { "os": "none", "cpu": "x64" }, "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ=="],
- "convex/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.27.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-AS18v0V+vZiLJyi/4LphvBE+OIX682Pu7ZYNsdUHyUKSoRwdnOsMf6FDekwoAFKej14WAkOef3zAORJgAtXnlQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.25.12", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A=="],
- "convex/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.27.0", "", { "os": "linux", "cpu": "ia32" }, "sha512-Mz1jxqm/kfgKkc/KLHC5qIujMvnnarD9ra1cEcrs7qshTUSksPihGrWHVG5+osAIQ68577Zpww7SGapmzSt4Nw=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.25.12", "", { "os": "openbsd", "cpu": "x64" }, "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw=="],
- "convex/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-QbEREjdJeIreIAbdG2hLU1yXm1uu+LTdzoq1KCo4G4pFOLlvIspBm36QrQOar9LFduavoWX2msNFAAAY9j4BDg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.25.12", "", { "os": "none", "cpu": "arm64" }, "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg=="],
- "convex/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-sJz3zRNe4tO2wxvDpH/HYJilb6+2YJxo/ZNbVdtFiKDufzWq4JmKAiHy9iGoLjAV7r/W32VgaHGkk35cUXlNOg=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.25.12", "", { "os": "sunos", "cpu": "x64" }, "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w=="],
- "convex/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.27.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-z9N10FBD0DCS2dmSABDBb5TLAyF1/ydVb+N4pi88T45efQ/w4ohr/F/QYCkxDPnkhkp6AIpIcQKQ8F0ANoA2JA=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.25.12", "", { "os": "win32", "cpu": "arm64" }, "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg=="],
- "convex/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-pQdyAIZ0BWIC5GyvVFn5awDiO14TkT/19FTmFcPdDec94KJ1uZcmFs21Fo8auMXzD4Tt+diXu1LW1gHus9fhFQ=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.25.12", "", { "os": "win32", "cpu": "ia32" }, "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ=="],
- "convex/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.27.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-hPlRWR4eIDDEci953RI1BLZitgi5uqcsjKMxwYfmi4LcwyWo2IcRP+lThVnKjNtk90pLS8nKdroXYOqW+QQH+w=="],
+ "@sveltejs/adapter-vercel/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.25.12", "", { "os": "win32", "cpu": "x64" }, "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA=="],
- "convex/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.27.0", "", { "os": "linux", "cpu": "x64" }, "sha512-1hBWx4OUJE2cab++aVZ7pObD6s+DK4mPGpemtnAORBvb5l/g5xFGk0vc0PjSkrDs0XaXj9yyob3d14XqvnQ4gw=="],
+ "accepts/mime-types/mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="],
- "convex/esbuild/@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-6m0sfQfxfQfy1qRuecMkJlf1cIzTOgyaeXaiVaaki8/v+WB+U4hc6ik15ZW6TAllRlg/WuQXxWj1jx6C+dfy3w=="],
+ "babel-jest/chalk/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="],
- "convex/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.27.0", "", { "os": "none", "cpu": "x64" }, "sha512-xbbOdfn06FtcJ9d0ShxxvSn2iUsGd/lgPIO2V3VZIPDbEaIj1/3nBBe1AwuEZKXVXkMmpr6LUAgMkLD/4D2PPA=="],
+ "babel-jest/chalk/supports-color": ["supports-color@7.2.0", "", { "dependencies": { "has-flag": "^4.0.0" } }, "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw=="],
- "convex/esbuild/@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.27.0", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fWgqR8uNbCQ/GGv0yhzttj6sU/9Z5/Sv/VGU3F5OuXK6J6SlriONKrQ7tNlwBrJZXRYk5jUhuWvF7GYzGguBZQ=="],
+ "chrome-launcher/is-wsl/is-docker": ["is-docker@2.2.1", "", { "bin": { "is-docker": "cli.js" } }, "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ=="],
- "convex/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.27.0", "", { "os": "openbsd", "cpu": "x64" }, "sha512-aCwlRdSNMNxkGGqQajMUza6uXzR/U0dIl1QmLjPtRbLOx3Gy3otfFu/VjATy4yQzo9yFDGTxYDo1FfAD9oRD2A=="],
+ "chromium-edge-launcher/is-wsl/is-docker": ["is-docker@2.2.1", "", { "bin": { "is-docker": "cli.js" } }, "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ=="],
- "convex/esbuild/@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-nyvsBccxNAsNYz2jVFYwEGuRRomqZ149A39SHWk4hV0jWxKM0hjBPm3AmdxcbHiFLbBSwG6SbpIcUbXjgyECfA=="],
+ "cliui/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="],
- "convex/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.27.0", "", { "os": "sunos", "cpu": "x64" }, "sha512-Q1KY1iJafM+UX6CFEL+F4HRTgygmEW568YMqDA5UV97AuZSm21b7SXIrRJDwXWPzr8MGr75fUZPV67FdtMHlHA=="],
+ "cliui/string-width/is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="],
- "convex/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.27.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-W1eyGNi6d+8kOmZIwi/EDjrL9nxQIQ0MiGqe/AWc6+IaHloxHSGoeRgDRKHFISThLmsewZ5nHFvGFWdBYlgKPg=="],
+ "cliui/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="],
- "convex/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.27.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-30z1aKL9h22kQhilnYkORFYt+3wp7yZsHWus+wSKAJR8JtdfI76LJ4SBdMsCopTR3z/ORqVu5L1vtnHZWVj4cQ=="],
+ "cliui/wrap-ansi/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="],
- "convex/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.27.0", "", { "os": "win32", "cpu": "x64" }, "sha512-aIitBcjQeyOhMTImhLZmtxfdOcuNRpwlPNmlFKPcHQYPhEssw75Cl1TSXJXpMkzaua9FUetx/4OQKq7eJul5Cg=="],
+ "connect/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="],
"finalhandler/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="],
@@ -3248,7 +3308,7 @@
"inquirer/@inquirer/core/wrap-ansi": ["wrap-ansi@6.2.0", "", { "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA=="],
- "jayson/@types/ws/@types/node": ["@types/node@22.19.13", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-akNQMv0wW5uyRpD2v2IEyRSZiR+BeGuoB6L310EgGObO44HSMNT8z1xzio28V8qOrgYaopIDNA18YgdXd+qTiw=="],
+ "jayson/@types/ws/@types/node": ["@types/node@22.19.15", "", { "dependencies": { "undici-types": "~6.21.0" } }, "sha512-F0R/h2+dsy5wJAUe3tAU6oqa2qbWY5TpNfL/RGmo1y38hiyO1w3x2jPtt76wmuaJI4DQnOBu21cNXQ2STIUUWg=="],
"jest-message-util/chalk/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="],
@@ -3278,8 +3338,6 @@
"node-fetch/whatwg-url/webidl-conversions": ["webidl-conversions@3.0.1", "", {}, "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ=="],
- "posthog-js/@opentelemetry/resources/@opentelemetry/core": ["@opentelemetry/core@2.6.0", "", { "dependencies": { "@opentelemetry/semantic-conventions": "^1.29.0" }, "peerDependencies": { "@opentelemetry/api": ">=1.0.0 <1.10.0" } }, "sha512-HLM1v2cbZ4TgYN6KEOj+Bbj8rAKriOdkF9Ed3tG25FoprSiQl7kYc+RRT6fUZGOvx0oMi5U67GoFdT+XUn8zEg=="],
-
"qrcode/yargs/cliui": ["cliui@6.0.0", "", { "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.0", "wrap-ansi": "^6.2.0" } }, "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ=="],
"qrcode/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=="],
@@ -3310,58 +3368,6 @@
"vercel-minimax-ai-provider/@ai-sdk/anthropic/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@4.0.3", "", { "dependencies": { "@ai-sdk/provider": "3.0.1", "@standard-schema/spec": "^1.1.0", "eventsource-parser": "^3.0.6" }, "peerDependencies": { "zod": "^3.25.76 || ^4.1.8" } }, "sha512-Vo2p61dDld8Dy/O66zKQpE4nqHojiEEYEjZcSbICjE7h8Z6QmHzBfd+ss/paIDdyXyS0yHmC1GoRYYKo89cqZQ=="],
- "vite/esbuild/@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.27.0", "", { "os": "aix", "cpu": "ppc64" }, "sha512-KuZrd2hRjz01y5JK9mEBSD3Vj3mbCvemhT466rSuJYeE/hjuBrHfjjcjMdTm/sz7au+++sdbJZJmuBwQLuw68A=="],
-
- "vite/esbuild/@esbuild/android-arm": ["@esbuild/android-arm@0.27.0", "", { "os": "android", "cpu": "arm" }, "sha512-j67aezrPNYWJEOHUNLPj9maeJte7uSMM6gMoxfPC9hOg8N02JuQi/T7ewumf4tNvJadFkvLZMlAq73b9uwdMyQ=="],
-
- "vite/esbuild/@esbuild/android-arm64": ["@esbuild/android-arm64@0.27.0", "", { "os": "android", "cpu": "arm64" }, "sha512-CC3vt4+1xZrs97/PKDkl0yN7w8edvU2vZvAFGD16n9F0Cvniy5qvzRXjfO1l94efczkkQE6g1x0i73Qf5uthOQ=="],
-
- "vite/esbuild/@esbuild/android-x64": ["@esbuild/android-x64@0.27.0", "", { "os": "android", "cpu": "x64" }, "sha512-wurMkF1nmQajBO1+0CJmcN17U4BP6GqNSROP8t0X/Jiw2ltYGLHpEksp9MpoBqkrFR3kv2/te6Sha26k3+yZ9Q=="],
-
- "vite/esbuild/@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.27.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-uJOQKYCcHhg07DL7i8MzjvS2LaP7W7Pn/7uA0B5S1EnqAirJtbyw4yC5jQ5qcFjHK9l6o/MX9QisBg12kNkdHg=="],
-
- "vite/esbuild/@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.27.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-8mG6arH3yB/4ZXiEnXof5MK72dE6zM9cDvUcPtxhUZsDjESl9JipZYW60C3JGreKCEP+p8P/72r69m4AZGJd5g=="],
-
- "vite/esbuild/@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.27.0", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-9FHtyO988CwNMMOE3YIeci+UV+x5Zy8fI2qHNpsEtSF83YPBmE8UWmfYAQg6Ux7Gsmd4FejZqnEUZCMGaNQHQw=="],
-
- "vite/esbuild/@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.27.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-zCMeMXI4HS/tXvJz8vWGexpZj2YVtRAihHLk1imZj4efx1BQzN76YFeKqlDr3bUWI26wHwLWPd3rwh6pe4EV7g=="],
-
- "vite/esbuild/@esbuild/linux-arm": ["@esbuild/linux-arm@0.27.0", "", { "os": "linux", "cpu": "arm" }, "sha512-t76XLQDpxgmq2cNXKTVEB7O7YMb42atj2Re2Haf45HkaUpjM2J0UuJZDuaGbPbamzZ7bawyGFUkodL+zcE+jvQ=="],
-
- "vite/esbuild/@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.27.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-AS18v0V+vZiLJyi/4LphvBE+OIX682Pu7ZYNsdUHyUKSoRwdnOsMf6FDekwoAFKej14WAkOef3zAORJgAtXnlQ=="],
-
- "vite/esbuild/@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.27.0", "", { "os": "linux", "cpu": "ia32" }, "sha512-Mz1jxqm/kfgKkc/KLHC5qIujMvnnarD9ra1cEcrs7qshTUSksPihGrWHVG5+osAIQ68577Zpww7SGapmzSt4Nw=="],
-
- "vite/esbuild/@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-QbEREjdJeIreIAbdG2hLU1yXm1uu+LTdzoq1KCo4G4pFOLlvIspBm36QrQOar9LFduavoWX2msNFAAAY9j4BDg=="],
-
- "vite/esbuild/@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-sJz3zRNe4tO2wxvDpH/HYJilb6+2YJxo/ZNbVdtFiKDufzWq4JmKAiHy9iGoLjAV7r/W32VgaHGkk35cUXlNOg=="],
-
- "vite/esbuild/@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.27.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-z9N10FBD0DCS2dmSABDBb5TLAyF1/ydVb+N4pi88T45efQ/w4ohr/F/QYCkxDPnkhkp6AIpIcQKQ8F0ANoA2JA=="],
-
- "vite/esbuild/@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.27.0", "", { "os": "linux", "cpu": "none" }, "sha512-pQdyAIZ0BWIC5GyvVFn5awDiO14TkT/19FTmFcPdDec94KJ1uZcmFs21Fo8auMXzD4Tt+diXu1LW1gHus9fhFQ=="],
-
- "vite/esbuild/@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.27.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-hPlRWR4eIDDEci953RI1BLZitgi5uqcsjKMxwYfmi4LcwyWo2IcRP+lThVnKjNtk90pLS8nKdroXYOqW+QQH+w=="],
-
- "vite/esbuild/@esbuild/linux-x64": ["@esbuild/linux-x64@0.27.0", "", { "os": "linux", "cpu": "x64" }, "sha512-1hBWx4OUJE2cab++aVZ7pObD6s+DK4mPGpemtnAORBvb5l/g5xFGk0vc0PjSkrDs0XaXj9yyob3d14XqvnQ4gw=="],
-
- "vite/esbuild/@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-6m0sfQfxfQfy1qRuecMkJlf1cIzTOgyaeXaiVaaki8/v+WB+U4hc6ik15ZW6TAllRlg/WuQXxWj1jx6C+dfy3w=="],
-
- "vite/esbuild/@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.27.0", "", { "os": "none", "cpu": "x64" }, "sha512-xbbOdfn06FtcJ9d0ShxxvSn2iUsGd/lgPIO2V3VZIPDbEaIj1/3nBBe1AwuEZKXVXkMmpr6LUAgMkLD/4D2PPA=="],
-
- "vite/esbuild/@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.27.0", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-fWgqR8uNbCQ/GGv0yhzttj6sU/9Z5/Sv/VGU3F5OuXK6J6SlriONKrQ7tNlwBrJZXRYk5jUhuWvF7GYzGguBZQ=="],
-
- "vite/esbuild/@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.27.0", "", { "os": "openbsd", "cpu": "x64" }, "sha512-aCwlRdSNMNxkGGqQajMUza6uXzR/U0dIl1QmLjPtRbLOx3Gy3otfFu/VjATy4yQzo9yFDGTxYDo1FfAD9oRD2A=="],
-
- "vite/esbuild/@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.27.0", "", { "os": "none", "cpu": "arm64" }, "sha512-nyvsBccxNAsNYz2jVFYwEGuRRomqZ149A39SHWk4hV0jWxKM0hjBPm3AmdxcbHiFLbBSwG6SbpIcUbXjgyECfA=="],
-
- "vite/esbuild/@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.27.0", "", { "os": "sunos", "cpu": "x64" }, "sha512-Q1KY1iJafM+UX6CFEL+F4HRTgygmEW568YMqDA5UV97AuZSm21b7SXIrRJDwXWPzr8MGr75fUZPV67FdtMHlHA=="],
-
- "vite/esbuild/@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.27.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-W1eyGNi6d+8kOmZIwi/EDjrL9nxQIQ0MiGqe/AWc6+IaHloxHSGoeRgDRKHFISThLmsewZ5nHFvGFWdBYlgKPg=="],
-
- "vite/esbuild/@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.27.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-30z1aKL9h22kQhilnYkORFYt+3wp7yZsHWus+wSKAJR8JtdfI76LJ4SBdMsCopTR3z/ORqVu5L1vtnHZWVj4cQ=="],
-
- "vite/esbuild/@esbuild/win32-x64": ["@esbuild/win32-x64@0.27.0", "", { "os": "win32", "cpu": "x64" }, "sha512-aIitBcjQeyOhMTImhLZmtxfdOcuNRpwlPNmlFKPcHQYPhEssw75Cl1TSXJXpMkzaua9FUetx/4OQKq7eJul5Cg=="],
-
"yargs/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="],
"yargs/string-width/is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="],
diff --git a/package.json b/package.json
index 14c16af3..96cec2db 100644
--- a/package.json
+++ b/package.json
@@ -22,7 +22,9 @@
},
"packageManager": "bun@1.3.10",
"scripts": {
- "dev:web": "turbo dev --filter=@btca/web",
+ "dev:web": "turbo dev --ui=tui --filter=@btca/web --filter=@btca/convex",
+ "dev:desktop": "turbo dev --ui=tui --filter=@btca/web --filter=@btca/desktop --filter=@btca/convex",
+ "dev:convex": "turbo dev --ui=tui --filter=@btca/convex",
"dev:docs": "bun run --cwd apps/docs dev",
"dev:cli": "bun run --watch apps/cli/src/index.ts",
"cli": "bun run apps/cli/src/index.ts",
@@ -38,6 +40,8 @@
"check:docs": "turbo check --filter=@btca/docs",
"check:cli": "turbo check --filter=btca",
"check:web": "turbo check --filter=@btca/web",
+ "check:desktop": "turbo check --filter=@btca/desktop",
+ "check:convex": "turbo check --filter=@btca/convex",
"check:shared": "turbo check --filter=@btca/shared",
"check:server": "turbo check --filter=btca-server",
"check:sandbox": "turbo check --filter=btca-sandbox",
@@ -45,6 +49,8 @@
"format:all": "turbo format",
"format:cli": "turbo format --filter=btca",
"format:web": "turbo format --filter=@btca/web",
+ "format:desktop": "turbo format --filter=@btca/desktop",
+ "format:convex": "turbo format --filter=@btca/convex",
"format:shared": "turbo format --filter=@btca/shared",
"format:server": "turbo format --filter=btca-server",
"format:sandbox": "turbo format --filter=btca-sandbox",
diff --git a/packages/convex/.gitignore b/packages/convex/.gitignore
new file mode 100644
index 00000000..8c5fbb9c
--- /dev/null
+++ b/packages/convex/.gitignore
@@ -0,0 +1,2 @@
+
+.env.local
diff --git a/packages/convex/README.md b/packages/convex/README.md
new file mode 100644
index 00000000..ba78de08
--- /dev/null
+++ b/packages/convex/README.md
@@ -0,0 +1,17 @@
+# @btca/convex
+
+Shared Convex backend package for btca.
+
+## Dev
+
+From the repo root:
+
+```sh
+bun run dev:convex
+```
+
+Or from this package directly:
+
+```sh
+bun run dev
+```
diff --git a/apps/web/convex.json b/packages/convex/convex.json
similarity index 82%
rename from apps/web/convex.json
rename to packages/convex/convex.json
index 5e83eb8b..bdd8bffa 100644
--- a/apps/web/convex.json
+++ b/packages/convex/convex.json
@@ -1,4 +1,4 @@
{
- "functions": "src/convex/",
+ "functions": "src/",
"$schema": "https://raw.githubusercontent.com/get-convex/convex-backend/refs/heads/main/npm-packages/convex/schemas/convex.schema.json"
}
diff --git a/packages/convex/package.json b/packages/convex/package.json
new file mode 100644
index 00000000..b6b2f2b1
--- /dev/null
+++ b/packages/convex/package.json
@@ -0,0 +1,41 @@
+{
+ "name": "@btca/convex",
+ "version": "0.0.1",
+ "private": true,
+ "type": "module",
+ "exports": {
+ "./api": "./src/api.ts",
+ "./apiHelpers": "./src/apiHelpers.ts",
+ "./dataModel": "./src/dataModel.ts",
+ "./errors": "./src/errors.ts",
+ "./instanceErrors": "./src/instanceErrors.ts",
+ "./webSandboxModels": "./src/webSandboxModels.ts",
+ "./aiBudget": "./src/aiBudget.ts"
+ },
+ "scripts": {
+ "dev": "convex dev",
+ "deploy:web": "convex deploy --cmd-url-env-var-name PUBLIC_CONVEX_URL --cmd 'bun run --cwd ../../apps/web build'",
+ "check": "bunx tsc -p src/tsconfig.json --noEmit",
+ "format": "prettier --write ."
+ },
+ "devDependencies": {
+ "@types/bun": "^1.3.10",
+ "prettier": "^3.8.1",
+ "typescript": "^5.9.3"
+ },
+ "dependencies": {
+ "@ai-sdk/openai-compatible": "^2.0.29",
+ "@btca/shared": "workspace:*",
+ "@clerk/backend": "^2.30.1",
+ "@convex-dev/migrations": "^0.3.1",
+ "@daytonaio/sdk": "^0.149.0",
+ "ai": "^6.0.81",
+ "autumn-js": "^0.1.82",
+ "better-result": "^2.7.0",
+ "btca-sandbox": "workspace:*",
+ "convex": "^1.31.7",
+ "nanoid": "^5.1.6",
+ "posthog-node": "^5.24.15",
+ "zod": "^4.3.6"
+ }
+}
diff --git a/apps/web/src/convex/README.md b/packages/convex/src/README.md
similarity index 100%
rename from apps/web/src/convex/README.md
rename to packages/convex/src/README.md
diff --git a/packages/convex/src/_generated/api.d.ts b/packages/convex/src/_generated/api.d.ts
new file mode 100644
index 00000000..8202f330
--- /dev/null
+++ b/packages/convex/src/_generated/api.d.ts
@@ -0,0 +1,203 @@
+/* eslint-disable */
+/**
+ * Generated `api` utility.
+ *
+ * THIS CODE IS AUTOMATICALLY GENERATED.
+ *
+ * To regenerate, run `npx convex dev`.
+ * @module
+ */
+
+import type * as aiBudget from '../aiBudget.js';
+import type * as analytics from '../analytics.js';
+import type * as analyticsEvents from '../analyticsEvents.js';
+import type * as api_ from '../api.js';
+import type * as apiHelpers from '../apiHelpers.js';
+import type * as authHelpers from '../authHelpers.js';
+import type * as clerkApiKeys from '../clerkApiKeys.js';
+import type * as clerkApiKeysQueries from '../clerkApiKeysQueries.js';
+import type * as crons from '../crons.js';
+import type * as dataModel from '../dataModel.js';
+import type * as errors from '../errors.js';
+import type * as githubApp from '../githubApp.js';
+import type * as githubAuth from '../githubAuth.js';
+import type * as githubConnections from '../githubConnections.js';
+import type * as http from '../http.js';
+import type * as instanceErrors from '../instanceErrors.js';
+import type * as instances_actions from '../instances/actions.js';
+import type * as instances_mutations from '../instances/mutations.js';
+import type * as instances_queries from '../instances/queries.js';
+import type * as lib_billing_aiBudget from '../lib/billing/aiBudget.js';
+import type * as lib_instanceErrors from '../lib/instanceErrors.js';
+import type * as lib_models_webSandboxModels from '../lib/models/webSandboxModels.js';
+import type * as lib_result_errors from '../lib/result/errors.js';
+import type * as mcp from '../mcp.js';
+import type * as mcp_resourceContract from '../mcp/resourceContract.js';
+import type * as mcpInternal from '../mcpInternal.js';
+import type * as mcpQuestions from '../mcpQuestions.js';
+import type * as messages from '../messages.js';
+import type * as migrations from '../migrations.js';
+import type * as privateWrappers from '../privateWrappers.js';
+import type * as projects from '../projects.js';
+import type * as resourceActions from '../resourceActions.js';
+import type * as resources from '../resources.js';
+import type * as scheduled_queries from '../scheduled/queries.js';
+import type * as scheduled_updates from '../scheduled/updates.js';
+import type * as scheduled_versionCheck from '../scheduled/versionCheck.js';
+import type * as streamSessions from '../streamSessions.js';
+import type * as threadTitle from '../threadTitle.js';
+import type * as threads from '../threads.js';
+import type * as usage from '../usage.js';
+import type * as users from '../users.js';
+import type * as webSandboxModels from '../webSandboxModels.js';
+
+import type { ApiFromModules, FilterApi, FunctionReference } from 'convex/server';
+
+declare const fullApi: ApiFromModules<{
+ aiBudget: typeof aiBudget;
+ analytics: typeof analytics;
+ analyticsEvents: typeof analyticsEvents;
+ api: typeof api_;
+ apiHelpers: typeof apiHelpers;
+ authHelpers: typeof authHelpers;
+ clerkApiKeys: typeof clerkApiKeys;
+ clerkApiKeysQueries: typeof clerkApiKeysQueries;
+ crons: typeof crons;
+ dataModel: typeof dataModel;
+ errors: typeof errors;
+ githubApp: typeof githubApp;
+ githubAuth: typeof githubAuth;
+ githubConnections: typeof githubConnections;
+ http: typeof http;
+ instanceErrors: typeof instanceErrors;
+ 'instances/actions': typeof instances_actions;
+ 'instances/mutations': typeof instances_mutations;
+ 'instances/queries': typeof instances_queries;
+ 'lib/billing/aiBudget': typeof lib_billing_aiBudget;
+ 'lib/instanceErrors': typeof lib_instanceErrors;
+ 'lib/models/webSandboxModels': typeof lib_models_webSandboxModels;
+ 'lib/result/errors': typeof lib_result_errors;
+ mcp: typeof mcp;
+ 'mcp/resourceContract': typeof mcp_resourceContract;
+ mcpInternal: typeof mcpInternal;
+ mcpQuestions: typeof mcpQuestions;
+ messages: typeof messages;
+ migrations: typeof migrations;
+ privateWrappers: typeof privateWrappers;
+ projects: typeof projects;
+ resourceActions: typeof resourceActions;
+ resources: typeof resources;
+ 'scheduled/queries': typeof scheduled_queries;
+ 'scheduled/updates': typeof scheduled_updates;
+ 'scheduled/versionCheck': typeof scheduled_versionCheck;
+ streamSessions: typeof streamSessions;
+ threadTitle: typeof threadTitle;
+ threads: typeof threads;
+ usage: typeof usage;
+ users: typeof users;
+ webSandboxModels: typeof webSandboxModels;
+}>;
+
+/**
+ * A utility for referencing Convex functions in your app's public API.
+ *
+ * Usage:
+ * ```js
+ * const myFunctionReference = api.myModule.myFunction;
+ * ```
+ */
+export declare const api: FilterApi>;
+
+/**
+ * A utility for referencing Convex functions in your app's internal API.
+ *
+ * Usage:
+ * ```js
+ * const myFunctionReference = internal.myModule.myFunction;
+ * ```
+ */
+export declare const internal: FilterApi>;
+
+export declare const components: {
+ migrations: {
+ lib: {
+ cancel: FunctionReference<
+ 'mutation',
+ 'internal',
+ { name: string },
+ {
+ batchSize?: number;
+ cursor?: string | null;
+ error?: string;
+ isDone: boolean;
+ latestEnd?: number;
+ latestStart: number;
+ name: string;
+ next?: Array;
+ processed: number;
+ state: 'inProgress' | 'success' | 'failed' | 'canceled' | 'unknown';
+ }
+ >;
+ cancelAll: FunctionReference<
+ 'mutation',
+ 'internal',
+ { sinceTs?: number },
+ Array<{
+ batchSize?: number;
+ cursor?: string | null;
+ error?: string;
+ isDone: boolean;
+ latestEnd?: number;
+ latestStart: number;
+ name: string;
+ next?: Array;
+ processed: number;
+ state: 'inProgress' | 'success' | 'failed' | 'canceled' | 'unknown';
+ }>
+ >;
+ clearAll: FunctionReference<'mutation', 'internal', { before?: number }, null>;
+ getStatus: FunctionReference<
+ 'query',
+ 'internal',
+ { limit?: number; names?: Array },
+ Array<{
+ batchSize?: number;
+ cursor?: string | null;
+ error?: string;
+ isDone: boolean;
+ latestEnd?: number;
+ latestStart: number;
+ name: string;
+ next?: Array;
+ processed: number;
+ state: 'inProgress' | 'success' | 'failed' | 'canceled' | 'unknown';
+ }>
+ >;
+ migrate: FunctionReference<
+ 'mutation',
+ 'internal',
+ {
+ batchSize?: number;
+ cursor?: string | null;
+ dryRun: boolean;
+ fnHandle: string;
+ name: string;
+ next?: Array<{ fnHandle: string; name: string }>;
+ oneBatchOnly?: boolean;
+ },
+ {
+ batchSize?: number;
+ cursor?: string | null;
+ error?: string;
+ isDone: boolean;
+ latestEnd?: number;
+ latestStart: number;
+ name: string;
+ next?: Array;
+ processed: number;
+ state: 'inProgress' | 'success' | 'failed' | 'canceled' | 'unknown';
+ }
+ >;
+ };
+ };
+};
diff --git a/apps/web/src/convex/_generated/api.js b/packages/convex/src/_generated/api.js
similarity index 87%
rename from apps/web/src/convex/_generated/api.js
rename to packages/convex/src/_generated/api.js
index 44bf9858..24593c74 100644
--- a/apps/web/src/convex/_generated/api.js
+++ b/packages/convex/src/_generated/api.js
@@ -8,7 +8,7 @@
* @module
*/
-import { anyApi, componentsGeneric } from "convex/server";
+import { anyApi, componentsGeneric } from 'convex/server';
/**
* A utility for referencing Convex functions in your app's API.
diff --git a/apps/web/src/convex/_generated/dataModel.d.ts b/packages/convex/src/_generated/dataModel.d.ts
similarity index 81%
rename from apps/web/src/convex/_generated/dataModel.d.ts
rename to packages/convex/src/_generated/dataModel.d.ts
index f97fd194..5428df6f 100644
--- a/apps/web/src/convex/_generated/dataModel.d.ts
+++ b/packages/convex/src/_generated/dataModel.d.ts
@@ -9,13 +9,13 @@
*/
import type {
- DataModelFromSchemaDefinition,
- DocumentByName,
- TableNamesInDataModel,
- SystemTableNames,
-} from "convex/server";
-import type { GenericId } from "convex/values";
-import schema from "../schema.js";
+ DataModelFromSchemaDefinition,
+ DocumentByName,
+ TableNamesInDataModel,
+ SystemTableNames
+} from 'convex/server';
+import type { GenericId } from 'convex/values';
+import schema from '../schema.js';
/**
* The names of all of your Convex tables.
@@ -27,10 +27,7 @@ export type TableNames = TableNamesInDataModel;
*
* @typeParam TableName - A string literal type of the table name (like "users").
*/
-export type Doc = DocumentByName<
- DataModel,
- TableName
->;
+export type Doc = DocumentByName;
/**
* An identifier for a document in Convex.
@@ -45,8 +42,7 @@ export type Doc = DocumentByName<
*
* @typeParam TableName - A string literal type of the table name (like "users").
*/
-export type Id =
- GenericId;
+export type Id = GenericId;
/**
* A type describing your Convex data model.
diff --git a/apps/web/src/convex/_generated/server.d.ts b/packages/convex/src/_generated/server.d.ts
similarity index 89%
rename from apps/web/src/convex/_generated/server.d.ts
rename to packages/convex/src/_generated/server.d.ts
index bec05e68..1cc047ef 100644
--- a/apps/web/src/convex/_generated/server.d.ts
+++ b/packages/convex/src/_generated/server.d.ts
@@ -9,17 +9,17 @@
*/
import {
- ActionBuilder,
- HttpActionBuilder,
- MutationBuilder,
- QueryBuilder,
- GenericActionCtx,
- GenericMutationCtx,
- GenericQueryCtx,
- GenericDatabaseReader,
- GenericDatabaseWriter,
-} from "convex/server";
-import type { DataModel } from "./dataModel.js";
+ ActionBuilder,
+ HttpActionBuilder,
+ MutationBuilder,
+ QueryBuilder,
+ GenericActionCtx,
+ GenericMutationCtx,
+ GenericQueryCtx,
+ GenericDatabaseReader,
+ GenericDatabaseWriter
+} from 'convex/server';
+import type { DataModel } from './dataModel.js';
/**
* Define a query in this Convex app's public API.
@@ -29,7 +29,7 @@ import type { DataModel } from "./dataModel.js";
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
*/
-export declare const query: QueryBuilder;
+export declare const query: QueryBuilder;
/**
* Define a query that is only accessible from other Convex functions (but not from the client).
@@ -39,7 +39,7 @@ export declare const query: QueryBuilder;
* @param func - The query function. It receives a {@link QueryCtx} as its first argument.
* @returns The wrapped query. Include this as an `export` to name it and make it accessible.
*/
-export declare const internalQuery: QueryBuilder;
+export declare const internalQuery: QueryBuilder;
/**
* Define a mutation in this Convex app's public API.
@@ -49,7 +49,7 @@ export declare const internalQuery: QueryBuilder;
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
*/
-export declare const mutation: MutationBuilder;
+export declare const mutation: MutationBuilder;
/**
* Define a mutation that is only accessible from other Convex functions (but not from the client).
@@ -59,7 +59,7 @@ export declare const mutation: MutationBuilder;
* @param func - The mutation function. It receives a {@link MutationCtx} as its first argument.
* @returns The wrapped mutation. Include this as an `export` to name it and make it accessible.
*/
-export declare const internalMutation: MutationBuilder;
+export declare const internalMutation: MutationBuilder;
/**
* Define an action in this Convex app's public API.
@@ -72,7 +72,7 @@ export declare const internalMutation: MutationBuilder;
* @param func - The action. It receives an {@link ActionCtx} as its first argument.
* @returns The wrapped action. Include this as an `export` to name it and make it accessible.
*/
-export declare const action: ActionBuilder;
+export declare const action: ActionBuilder;
/**
* Define an action that is only accessible from other Convex functions (but not from the client).
@@ -80,7 +80,7 @@ export declare const action: ActionBuilder;
* @param func - The function. It receives an {@link ActionCtx} as its first argument.
* @returns The wrapped function. Include this as an `export` to name it and make it accessible.
*/
-export declare const internalAction: ActionBuilder;
+export declare const internalAction: ActionBuilder;
/**
* Define an HTTP action.
diff --git a/apps/web/src/convex/_generated/server.js b/packages/convex/src/_generated/server.js
similarity index 95%
rename from apps/web/src/convex/_generated/server.js
rename to packages/convex/src/_generated/server.js
index bf3d25ad..a18aa285 100644
--- a/apps/web/src/convex/_generated/server.js
+++ b/packages/convex/src/_generated/server.js
@@ -9,14 +9,14 @@
*/
import {
- actionGeneric,
- httpActionGeneric,
- queryGeneric,
- mutationGeneric,
- internalActionGeneric,
- internalMutationGeneric,
- internalQueryGeneric,
-} from "convex/server";
+ actionGeneric,
+ httpActionGeneric,
+ queryGeneric,
+ mutationGeneric,
+ internalActionGeneric,
+ internalMutationGeneric,
+ internalQueryGeneric
+} from 'convex/server';
/**
* Define a query in this Convex app's public API.
diff --git a/packages/convex/src/aiBudget.ts b/packages/convex/src/aiBudget.ts
new file mode 100644
index 00000000..cc2363ce
--- /dev/null
+++ b/packages/convex/src/aiBudget.ts
@@ -0,0 +1 @@
+export * from './lib/billing/aiBudget.ts';
diff --git a/apps/web/src/convex/analytics.ts b/packages/convex/src/analytics.ts
similarity index 100%
rename from apps/web/src/convex/analytics.ts
rename to packages/convex/src/analytics.ts
diff --git a/apps/web/src/convex/analyticsEvents.ts b/packages/convex/src/analyticsEvents.ts
similarity index 100%
rename from apps/web/src/convex/analyticsEvents.ts
rename to packages/convex/src/analyticsEvents.ts
diff --git a/packages/convex/src/api.ts b/packages/convex/src/api.ts
new file mode 100644
index 00000000..c22a0e11
--- /dev/null
+++ b/packages/convex/src/api.ts
@@ -0,0 +1 @@
+export { api, components, internal } from './_generated/api.js';
diff --git a/apps/web/src/convex/apiHelpers.ts b/packages/convex/src/apiHelpers.ts
similarity index 100%
rename from apps/web/src/convex/apiHelpers.ts
rename to packages/convex/src/apiHelpers.ts
diff --git a/apps/web/src/convex/auth.config.ts b/packages/convex/src/auth.config.ts
similarity index 100%
rename from apps/web/src/convex/auth.config.ts
rename to packages/convex/src/auth.config.ts
diff --git a/apps/web/src/convex/authHelpers.ts b/packages/convex/src/authHelpers.ts
similarity index 99%
rename from apps/web/src/convex/authHelpers.ts
rename to packages/convex/src/authHelpers.ts
index 4b60bada..0d19d574 100644
--- a/apps/web/src/convex/authHelpers.ts
+++ b/packages/convex/src/authHelpers.ts
@@ -4,7 +4,7 @@ import { Result } from 'better-result';
import type { QueryCtx, MutationCtx, ActionCtx } from './_generated/server';
import type { Id, Doc } from './_generated/dataModel';
import { instances } from './apiHelpers';
-import { WebAuthError } from '../lib/result/errors';
+import { WebAuthError } from './lib/result/errors';
type DbCtx = QueryCtx | MutationCtx;
type AuthContextError = WebAuthError;
diff --git a/apps/web/src/convex/clerkApiKeys.ts b/packages/convex/src/clerkApiKeys.ts
similarity index 100%
rename from apps/web/src/convex/clerkApiKeys.ts
rename to packages/convex/src/clerkApiKeys.ts
diff --git a/apps/web/src/convex/clerkApiKeysQueries.ts b/packages/convex/src/clerkApiKeysQueries.ts
similarity index 100%
rename from apps/web/src/convex/clerkApiKeysQueries.ts
rename to packages/convex/src/clerkApiKeysQueries.ts
diff --git a/apps/web/src/convex/convex.config.ts b/packages/convex/src/convex.config.ts
similarity index 100%
rename from apps/web/src/convex/convex.config.ts
rename to packages/convex/src/convex.config.ts
diff --git a/apps/web/src/convex/crons.ts b/packages/convex/src/crons.ts
similarity index 100%
rename from apps/web/src/convex/crons.ts
rename to packages/convex/src/crons.ts
diff --git a/packages/convex/src/dataModel.ts b/packages/convex/src/dataModel.ts
new file mode 100644
index 00000000..6268c703
--- /dev/null
+++ b/packages/convex/src/dataModel.ts
@@ -0,0 +1 @@
+export type { DataModel, Doc, Id, TableNames } from './_generated/dataModel.js';
diff --git a/packages/convex/src/errors.ts b/packages/convex/src/errors.ts
new file mode 100644
index 00000000..0a07bf33
--- /dev/null
+++ b/packages/convex/src/errors.ts
@@ -0,0 +1 @@
+export * from './lib/result/errors.ts';
diff --git a/apps/web/src/convex/githubApp.ts b/packages/convex/src/githubApp.ts
similarity index 99%
rename from apps/web/src/convex/githubApp.ts
rename to packages/convex/src/githubApp.ts
index d256d5f8..36b64ba4 100644
--- a/apps/web/src/convex/githubApp.ts
+++ b/packages/convex/src/githubApp.ts
@@ -5,7 +5,7 @@ import {
WebConfigMissingError,
WebUnhandledError,
WebValidationError
-} from '../lib/result/errors';
+} from './lib/result/errors';
const GITHUB_API_BASE_URL = 'https://api.github.com';
const GITHUB_API_VERSION = '2022-11-28';
diff --git a/apps/web/src/convex/githubAuth.ts b/packages/convex/src/githubAuth.ts
similarity index 98%
rename from apps/web/src/convex/githubAuth.ts
rename to packages/convex/src/githubAuth.ts
index 6b801d6f..7cb1e06a 100644
--- a/apps/web/src/convex/githubAuth.ts
+++ b/packages/convex/src/githubAuth.ts
@@ -8,7 +8,7 @@ import type { Id } from './_generated/dataModel';
import { action } from './_generated/server';
import { instances } from './apiHelpers';
import { getInstallationSnapshot } from './githubApp';
-import { WebAuthError, WebUnhandledError } from '../lib/result/errors';
+import { WebAuthError, WebUnhandledError } from './lib/result/errors';
type InternalGithubConnections = {
getByInstanceId: FunctionReference<
diff --git a/apps/web/src/convex/githubConnections.ts b/packages/convex/src/githubConnections.ts
similarity index 100%
rename from apps/web/src/convex/githubConnections.ts
rename to packages/convex/src/githubConnections.ts
diff --git a/apps/web/src/convex/http.ts b/packages/convex/src/http.ts
similarity index 99%
rename from apps/web/src/convex/http.ts
rename to packages/convex/src/http.ts
index ee62ff08..80aa0d89 100644
--- a/apps/web/src/convex/http.ts
+++ b/packages/convex/src/http.ts
@@ -16,9 +16,9 @@ import {
INSTANCE_DISK_FULL_MESSAGE,
getInstanceErrorKind,
getUserFacingInstanceError
-} from '../lib/instanceErrors';
-import { getWebSandboxModel } from '../lib/models/webSandboxModels.ts';
-import { WebConfigMissingError, WebUnhandledError, type WebError } from '../lib/result/errors';
+} from './lib/instanceErrors';
+import { getWebSandboxModel } from './lib/models/webSandboxModels.ts';
+import { WebConfigMissingError, WebUnhandledError, type WebError } from './lib/result/errors';
import { withInstanceRuntimeConfigLock } from './runtimeConfigLock.js';
type HttpFlowResult = Result;
diff --git a/packages/convex/src/instanceErrors.ts b/packages/convex/src/instanceErrors.ts
new file mode 100644
index 00000000..e40aa2e9
--- /dev/null
+++ b/packages/convex/src/instanceErrors.ts
@@ -0,0 +1 @@
+export * from './lib/instanceErrors.ts';
diff --git a/apps/web/src/convex/instances/actions.ts b/packages/convex/src/instances/actions.ts
similarity index 99%
rename from apps/web/src/convex/instances/actions.ts
rename to packages/convex/src/instances/actions.ts
index 6eec3bef..8c197dd4 100644
--- a/apps/web/src/convex/instances/actions.ts
+++ b/packages/convex/src/instances/actions.ts
@@ -19,15 +19,15 @@ import {
parseGitHubRepoRef
} from '../githubApp';
import { privateAction, withPrivateApiKey } from '../privateWrappers';
-import { getInstanceErrorKind, getUserFacingInstanceError } from '../../lib/instanceErrors';
-import { getWebSandboxModel } from '../../lib/models/webSandboxModels.ts';
+import { getInstanceErrorKind, getUserFacingInstanceError } from '../lib/instanceErrors';
+import { getWebSandboxModel } from '../lib/models/webSandboxModels.ts';
import {
WebAuthError,
WebConfigMissingError,
WebUnhandledError,
WebValidationError,
type WebError
-} from '../../lib/result/errors';
+} from '../lib/result/errors';
import { withInstanceRuntimeConfigLock } from '../runtimeConfigLock.js';
const instanceQueries = instances.queries;
diff --git a/apps/web/src/convex/instances/mutations.ts b/packages/convex/src/instances/mutations.ts
similarity index 100%
rename from apps/web/src/convex/instances/mutations.ts
rename to packages/convex/src/instances/mutations.ts
diff --git a/apps/web/src/convex/instances/queries.ts b/packages/convex/src/instances/queries.ts
similarity index 99%
rename from apps/web/src/convex/instances/queries.ts
rename to packages/convex/src/instances/queries.ts
index c0e5fd75..5f6c4c2f 100644
--- a/apps/web/src/convex/instances/queries.ts
+++ b/packages/convex/src/instances/queries.ts
@@ -4,7 +4,7 @@ import { BTCA_SNAPSHOT_NAME } from 'btca-sandbox/shared';
import { internalQuery, query } from '../_generated/server';
import { requireInstanceOwnershipResult, unwrapAuthResult } from '../authHelpers';
-import { WebAuthError } from '../../lib/result/errors';
+import { WebAuthError } from '../lib/result/errors';
// Instance validator
const instanceValidator = v.object({
diff --git a/apps/web/src/lib/billing/aiBudget.ts b/packages/convex/src/lib/billing/aiBudget.ts
similarity index 100%
rename from apps/web/src/lib/billing/aiBudget.ts
rename to packages/convex/src/lib/billing/aiBudget.ts
diff --git a/apps/web/src/lib/instanceErrors.ts b/packages/convex/src/lib/instanceErrors.ts
similarity index 100%
rename from apps/web/src/lib/instanceErrors.ts
rename to packages/convex/src/lib/instanceErrors.ts
diff --git a/apps/web/src/lib/models/webSandboxModels.ts b/packages/convex/src/lib/models/webSandboxModels.ts
similarity index 100%
rename from apps/web/src/lib/models/webSandboxModels.ts
rename to packages/convex/src/lib/models/webSandboxModels.ts
diff --git a/apps/web/src/lib/result/errors.ts b/packages/convex/src/lib/result/errors.ts
similarity index 100%
rename from apps/web/src/lib/result/errors.ts
rename to packages/convex/src/lib/result/errors.ts
diff --git a/apps/web/src/convex/mcp.ts b/packages/convex/src/mcp.ts
similarity index 98%
rename from apps/web/src/convex/mcp.ts
rename to packages/convex/src/mcp.ts
index 938912d8..d2aa4bc3 100644
--- a/apps/web/src/convex/mcp.ts
+++ b/packages/convex/src/mcp.ts
@@ -15,9 +15,9 @@ import {
INSTANCE_DISK_FULL_APP_MESSAGE,
getInstanceErrorKind,
getUserFacingInstanceError
-} from '../lib/instanceErrors';
+} from './lib/instanceErrors';
+import { toWebError, type WebError } from './lib/result/errors';
import { withInstanceRuntimeConfigLock } from './runtimeConfigLock.js';
-import { toWebError, type WebError } from '../lib/result/errors';
const instanceActions = instances.actions;
const instanceMutations = instances.mutations;
@@ -566,6 +566,13 @@ type SyncResult = {
}>;
};
+type SyncExistingResource = {
+ name: string;
+ url?: string;
+ branch?: string;
+ visibility?: string;
+};
+
/**
* Sync remote config with cloud - authenticated via API key
*/
@@ -677,14 +684,11 @@ export const sync = action({
const projectId = projectIdResult.value;
// Get current resources for this project
- const existingResources = (
+ const existingResources: SyncExistingResource[] = (
await ctx.runQuery(internal.resources.listByProject, {
projectId
})
- ).filter(
- (resource: { name: string; url?: string; branch?: string; visibility?: string }) =>
- resource.visibility !== 'private'
- );
+ ).filter((resource: SyncExistingResource) => resource.visibility !== 'private');
const synced: string[] = [];
const errors: string[] = [];
diff --git a/apps/web/src/convex/mcp/resourceContract.test.ts b/packages/convex/src/mcp/resourceContract.test.ts
similarity index 100%
rename from apps/web/src/convex/mcp/resourceContract.test.ts
rename to packages/convex/src/mcp/resourceContract.test.ts
diff --git a/apps/web/src/convex/mcp/resourceContract.ts b/packages/convex/src/mcp/resourceContract.ts
similarity index 100%
rename from apps/web/src/convex/mcp/resourceContract.ts
rename to packages/convex/src/mcp/resourceContract.ts
diff --git a/apps/web/src/convex/mcpInternal.ts b/packages/convex/src/mcpInternal.ts
similarity index 97%
rename from apps/web/src/convex/mcpInternal.ts
rename to packages/convex/src/mcpInternal.ts
index bbafe059..33af5e69 100644
--- a/apps/web/src/convex/mcpInternal.ts
+++ b/packages/convex/src/mcpInternal.ts
@@ -3,8 +3,8 @@ import { Result } from 'better-result';
import type { Id } from './_generated/dataModel';
import { internalMutation } from './_generated/server';
-import { isWebSandboxModelId } from '../lib/models/webSandboxModels.ts';
-import { WebConflictError, WebValidationError, type WebError } from '../lib/result/errors';
+import { isWebSandboxModelId } from './lib/models/webSandboxModels.ts';
+import { WebConflictError, WebValidationError, type WebError } from './lib/result/errors';
type McpInternalResult = Result;
diff --git a/apps/web/src/convex/mcpQuestions.ts b/packages/convex/src/mcpQuestions.ts
similarity index 97%
rename from apps/web/src/convex/mcpQuestions.ts
rename to packages/convex/src/mcpQuestions.ts
index b9343d7b..ec59e0cd 100644
--- a/apps/web/src/convex/mcpQuestions.ts
+++ b/packages/convex/src/mcpQuestions.ts
@@ -2,7 +2,7 @@ import { v } from 'convex/values';
import { query } from './_generated/server';
import { getAuthenticatedInstanceResult, unwrapAuthResult } from './authHelpers';
-import { WebValidationError, type WebError } from '../lib/result/errors';
+import { WebValidationError, type WebError } from './lib/result/errors';
const throwMcpQuestionError = (error: WebError): never => {
throw error;
diff --git a/apps/web/src/convex/messages.ts b/packages/convex/src/messages.ts
similarity index 98%
rename from apps/web/src/convex/messages.ts
rename to packages/convex/src/messages.ts
index 85eda214..7fa0ab55 100644
--- a/apps/web/src/convex/messages.ts
+++ b/packages/convex/src/messages.ts
@@ -7,7 +7,7 @@ import {
requireThreadOwnershipResult,
unwrapAuthResult
} from './authHelpers';
-import { WebValidationError, type WebError } from '../lib/result/errors';
+import { WebValidationError, type WebError } from './lib/result/errors';
const throwMessageError = (error: WebError): never => {
throw error;
diff --git a/apps/web/src/convex/migrations.ts b/packages/convex/src/migrations.ts
similarity index 100%
rename from apps/web/src/convex/migrations.ts
rename to packages/convex/src/migrations.ts
diff --git a/apps/web/src/convex/privateWrappers.ts b/packages/convex/src/privateWrappers.ts
similarity index 100%
rename from apps/web/src/convex/privateWrappers.ts
rename to packages/convex/src/privateWrappers.ts
diff --git a/apps/web/src/convex/projects.ts b/packages/convex/src/projects.ts
similarity index 99%
rename from apps/web/src/convex/projects.ts
rename to packages/convex/src/projects.ts
index f7b0f59d..7c5faac8 100644
--- a/apps/web/src/convex/projects.ts
+++ b/packages/convex/src/projects.ts
@@ -6,8 +6,8 @@ import type { Doc, Id } from './_generated/dataModel';
import { internalQuery, mutation, query } from './_generated/server';
import { AnalyticsEvents } from './analyticsEvents';
import { getAuthenticatedInstanceResult, unwrapAuthResult } from './authHelpers';
-import { isWebSandboxModelId } from '../lib/models/webSandboxModels.ts';
-import { WebConflictError, WebValidationError, type WebError } from '../lib/result/errors';
+import { isWebSandboxModelId } from './lib/models/webSandboxModels.ts';
+import { WebConflictError, WebValidationError, type WebError } from './lib/result/errors';
// Project validator
const projectValidator = v.object({
diff --git a/apps/web/src/convex/resourceActions.ts b/packages/convex/src/resourceActions.ts
similarity index 99%
rename from apps/web/src/convex/resourceActions.ts
rename to packages/convex/src/resourceActions.ts
index bf4c6dd7..6d34ac05 100644
--- a/apps/web/src/convex/resourceActions.ts
+++ b/packages/convex/src/resourceActions.ts
@@ -16,7 +16,7 @@ import {
parseGitHubRepoRef,
resolveAccessibleRepo
} from './githubApp';
-import { WebAuthError, WebUnhandledError, WebValidationError } from '../lib/result/errors';
+import { WebAuthError, WebUnhandledError, WebValidationError } from './lib/result/errors';
type InternalResources = {
addCustomResourceInternal: FunctionReference<
diff --git a/apps/web/src/convex/resources.ts b/packages/convex/src/resources.ts
similarity index 99%
rename from apps/web/src/convex/resources.ts
rename to packages/convex/src/resources.ts
index d8956292..94ef4881 100644
--- a/apps/web/src/convex/resources.ts
+++ b/packages/convex/src/resources.ts
@@ -11,8 +11,8 @@ import {
requireUserResourceOwnershipResult,
unwrapAuthResult
} from './authHelpers';
-import { WebValidationError } from '../lib/result/errors';
-import type { WebError } from '../lib/result/errors';
+import { WebValidationError } from './lib/result/errors';
+import type { WebError } from './lib/result/errors';
type ResourceNameResult = Result;
diff --git a/apps/web/src/convex/runtimeConfigLock.ts b/packages/convex/src/runtimeConfigLock.ts
similarity index 100%
rename from apps/web/src/convex/runtimeConfigLock.ts
rename to packages/convex/src/runtimeConfigLock.ts
diff --git a/apps/web/src/convex/scheduled/queries.ts b/packages/convex/src/scheduled/queries.ts
similarity index 100%
rename from apps/web/src/convex/scheduled/queries.ts
rename to packages/convex/src/scheduled/queries.ts
diff --git a/apps/web/src/convex/scheduled/updates.ts b/packages/convex/src/scheduled/updates.ts
similarity index 100%
rename from apps/web/src/convex/scheduled/updates.ts
rename to packages/convex/src/scheduled/updates.ts
diff --git a/apps/web/src/convex/scheduled/versionCheck.ts b/packages/convex/src/scheduled/versionCheck.ts
similarity index 100%
rename from apps/web/src/convex/scheduled/versionCheck.ts
rename to packages/convex/src/scheduled/versionCheck.ts
diff --git a/apps/web/src/convex/schema.ts b/packages/convex/src/schema.ts
similarity index 100%
rename from apps/web/src/convex/schema.ts
rename to packages/convex/src/schema.ts
diff --git a/apps/web/src/convex/streamSessions.ts b/packages/convex/src/streamSessions.ts
similarity index 100%
rename from apps/web/src/convex/streamSessions.ts
rename to packages/convex/src/streamSessions.ts
diff --git a/apps/web/src/convex/threadTitle.ts b/packages/convex/src/threadTitle.ts
similarity index 100%
rename from apps/web/src/convex/threadTitle.ts
rename to packages/convex/src/threadTitle.ts
diff --git a/apps/web/src/convex/threads.ts b/packages/convex/src/threads.ts
similarity index 99%
rename from apps/web/src/convex/threads.ts
rename to packages/convex/src/threads.ts
index 2c5f6c84..fd7623c7 100644
--- a/apps/web/src/convex/threads.ts
+++ b/packages/convex/src/threads.ts
@@ -8,7 +8,7 @@ import {
requireThreadOwnershipResult,
unwrapAuthResult
} from './authHelpers';
-import { WebValidationError } from '../lib/result/errors';
+import { WebValidationError } from './lib/result/errors';
// Shared validators
const threadValidator = v.object({
diff --git a/apps/web/src/convex/tsconfig.json b/packages/convex/src/tsconfig.json
similarity index 95%
rename from apps/web/src/convex/tsconfig.json
rename to packages/convex/src/tsconfig.json
index a84f9282..ffe6d34b 100644
--- a/apps/web/src/convex/tsconfig.json
+++ b/packages/convex/src/tsconfig.json
@@ -6,6 +6,7 @@
"compilerOptions": {
/* These settings are not required by Convex and can be modified. */
"allowJs": true,
+ "allowImportingTsExtensions": true,
"strict": true,
"moduleResolution": "Bundler",
"jsx": "react-jsx",
diff --git a/apps/web/src/convex/usage.ts b/packages/convex/src/usage.ts
similarity index 99%
rename from apps/web/src/convex/usage.ts
rename to packages/convex/src/usage.ts
index 7d62a8ca..5449eea0 100644
--- a/apps/web/src/convex/usage.ts
+++ b/packages/convex/src/usage.ts
@@ -13,15 +13,15 @@ import {
PRO_AI_BUDGET_MICROS,
getPreflightAiBudgetMicros,
totalAiBudgetMicros
-} from '../lib/billing/aiBudget';
-import { getWebSandboxModel } from '../lib/models/webSandboxModels';
+} from './lib/billing/aiBudget.ts';
+import { getWebSandboxModel } from './lib/models/webSandboxModels.ts';
import {
WebConfigMissingError,
WebExternalDependencyError,
WebUnhandledError,
WebValidationError,
type WebError
-} from '../lib/result/errors.js';
+} from './lib/result/errors.js';
type FeatureMetrics = {
usage: number;
diff --git a/apps/web/src/convex/users.ts b/packages/convex/src/users.ts
similarity index 100%
rename from apps/web/src/convex/users.ts
rename to packages/convex/src/users.ts
diff --git a/packages/convex/src/webSandboxModels.ts b/packages/convex/src/webSandboxModels.ts
new file mode 100644
index 00000000..e17bdca9
--- /dev/null
+++ b/packages/convex/src/webSandboxModels.ts
@@ -0,0 +1 @@
+export * from './lib/models/webSandboxModels.ts';