diff --git a/.gitignore b/.gitignore index 4d79e99..bee4bf8 100644 --- a/.gitignore +++ b/.gitignore @@ -57,3 +57,6 @@ # Finder (MacOS) folder config .DS_Store + +# Journals +journals/ diff --git a/.vscode/launch.json b/.vscode/launch.json index 1d85594..1bff3ec 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -4,19 +4,22 @@ { "type": "node", "request": "launch", - "name": "Debug Electron (Bun)", + "name": "Tauri dev (Bun)", "runtimeExecutable": "bun", "runtimeArgs": [ - "run", - "electron", - "." + "tauri", + "dev" ], "cwd": "${workspaceFolder}", - "protocol": "inspector", - "env": { - "NODE_ENV": "development" - }, - "console": "integratedTerminal" + "console": "integratedTerminal", + "internalConsoleOptions": "neverOpen" + }, + { + "type": "chrome", + "request": "launch", + "name": "Debug Webview", + "url": "http://localhost:1420", + "webRoot": "${workspaceFolder}/src/frontend" } ] -} +} \ No newline at end of file diff --git a/.vscode/tasks.json b/.vscode/tasks.json index affedd3..febeb1f 100644 --- a/.vscode/tasks.json +++ b/.vscode/tasks.json @@ -2,25 +2,40 @@ "version": "2.0.0", "tasks": [ { - "label": "Run Electron MCP Server", + "label": "tauri:dev", "type": "shell", - "command": "bun start", + "command": "bun tauri dev", "isBackground": true, "problemMatcher": { + "owner": "rust", + "fileLocation": [ + "relative", + "${workspaceFolder}/src-tauri" + ], "pattern": [ { - "regexp": ".", + "regexp": "^(warning|error): (.*)$", + "kind": 1, + "message": 2 + }, + { + "regexp": "^[ ]*--> (.*):(\\d+):(\\d+)$", "file": 1, - "location": 2, - "message": 3 + "line": 2, + "column": 3 } ], "background": { "activeOnStart": true, - "beginsPattern": "Electron app is ready", - "endsPattern": "MCP SSE Server listening" + "beginsPattern": "^\\s*(Compiling|Building)", + "endsPattern": "^\\s*(Finished|Running)" } } + }, + { + "label": "tauri:build", + "type": "shell", + "command": "bun tauri build" } ] } \ No newline at end of file diff --git a/README.md b/README.md index 613097c..f7169cf 100644 --- a/README.md +++ b/README.md @@ -1,291 +1,103 @@ -# SQLite Vector MCP Server (Electron + Bun) +# SQLite Vector MCP Server (Tauri + Rust) -> 🚀 Bun と Electron を使用した常駐型 MCP サーバー。SQLite + sqlite-vec でベクトル検索機能を提供。 +> 🦀 **Tauri 2 + Rust + SeaORM** を使用した、極めて軽量で高速な常駐型 MCP サーバー。 +> SQLite + `sqlite-vec` によるベクトル検索機能と、`llama.cpp` による LLM 連携を統合。 -[![License: ISC](https://img.shields.io/badge/License-ISC-blue.svg)](LICENSE) -[![Bun](https://img.shields.io/badge/Bun-v1.3.8-orange)](https://bun.sh) -[![Electron](https://img.shields.io/badge/Electron-v40.1.0-blue)](https://www.electronjs.org) +## 概要 + +このプロジェクトは、元々 Electron + Bun で構成されていた `sqlitevector` アプリケーションを、パフォーマンスとリソース効率を最大化するために **Tauri + Rust** へ完全に移植したものです。バックエンドには型安全な **SeaORM** を採用し、データベース操作の堅牢性を高めています。 ## 特徴 -- 🎯 **常駐アプリ**:システムトレイで常時実行 -- 🔍 **ベクトル検索**:sqlite-vec で高速な近傍検索 -- 🧠 **LLM 統合**:llama.cpp でテキスト生成・埋め込み生成 -- 📡 **MCP サーバー**:SSE(Server-Sent Events)で接続可能 -- ✅ **テストスイート**:35 テスト、100% 成功 -- 🔧 **設定可能**:環境変数で柔軟にカスタマイズ +- 🎯 **常駐型デザイン**: システムトレイに格納され、最小限のメモリで MCP サーバーとして動作。 +- 🔍 **ベクトル検索**: `sqlite-vec` を Rust からネイティブ操作し、高速な近傍検索(MATCH 句)を実現。 +- 🧠 **LLM 統合**: `llama.cpp` サーバーと連携し、Embedding 取得やテキスト補完をバックエンドで完結。 +- 🛠 **SeaORM**: 型安全なクエリビルダにより、複雑なデータベース操作を確実に実行。 +- 🚀 **開発体験**: `bun` と `tauri` CLI による、シンプルかつ高速なビルドフロー。 -## 対応する MCP ツール - -| ツール名 | 説明 | 入力 | -|---------|------|------| -| `add_item_text` | テキストから埋め込みを自動生成して保存 | `content: string`, `path?: string` | -| `add_item` | ベクトルを直接指定して保存 | `content: string`, `vector: number[]` | -| `search_text` | テキストから埋め込みを生成して検索 | `content: string` | -| `search_vector` | ベクトルで直接検索 | `vector: number[]` | -| `llm_generate` | llama.cpp でテキスト生成 | `prompt: string`, `options?: object` | - -## クイックスタート - -### 前提条件 - -- **Node.js/Bun**: `bun 1.3.8` 以上 -- **Electron**: アプリケーション内で自動インストール -- **llama.cpp**: 埋め込み・生成機能を使う場合は別途サーバー起動 - -### インストール - -```bash -git clone -cd sqlitevector -bun install -``` - -### 環境設定 - -`.env` ファイルを作成(またはコピー): - -```bash -# 既存の .env をコピー -cp .env.example .env # 存在する場合 - -# または手動作成 -cat > .env << 'EOF' -LLAMA_CPP_BASE_URL=http://127.0.0.1:8080 -LLAMA_CPP_EMBEDDING_MODEL=nomic-embed-text -LLAMA_CPP_MODEL=mistral -VEC_DIM=3 -MCP_PORT=3000 -EOF -``` - -### 起動 - -```bash -# Electron アプリとして起動 -bun start - -# ポート指定で起動 -MCP_PORT=3001 bun start -``` - -アプリケーション起動後、システムトレイに SQLite Vector MCP アイコンが表示されます。 - -## 設定 - -### 環境変数 - -| 変数 | デフォルト | 説明 | -|------|-----------|------| -| `MCP_PORT` | `3000` | MCP SSE サーバーのポート | -| `LLAMA_CPP_BASE_URL` | `http://127.0.0.1:8080` | llama.cpp サーバー URL | -| `LLAMA_CPP_EMBEDDING_MODEL` | - | 埋め込み用モデル名(例: `nomic-embed-text`) | -| `LLAMA_CPP_MODEL` | - | テキスト生成用モデル名(例: `mistral`) | -| `VEC_DIM` | `3` | ベクトル次元数(埋め込みモデルと一致させる) | - -### MCP クライアント設定 - -`mcp.json` または Claude Desktop など MCP クライアントの設定ファイルに: - -```json -{ - "mcpServers": { - "sqlite-vector-electron": { - "url": "http://localhost:3000/sse" - } - } -} -``` - -## 使用方法 - -### MCP ツール呼び出し例 - -```javascript -// テキストから自動的に埋め込みを生成して保存 -await callTool("add_item_text", { - content: "This is a sample document", - path: "/docs/sample.txt" -}); - -// ベクトルで検索 -await callTool("search_vector", { - vector: [0.1, 0.2, 0.3] -}); - -// テキストで検索(自動的に埋め込みを生成) -await callTool("search_text", { - content: "Similar documents" -}); - -// llama.cpp でテキスト生成 -await callTool("llm_generate", { - prompt: "What is machine learning?", - options: { temperature: 0.7, n_predict: 128 } -}); -``` - -## テスト - -### すべてのテストを実行 - -```bash -bun test -``` - -### 特定のテストモジュール実行 - -```bash -bun test test/db.test.js # DB テスト(5テスト) -bun test test/mcp-tools.test.js # ツール定義テスト(11テスト) -bun test test/mcp-handlers.test.js # ハンドラーテスト(6テスト) -bun test test/llama-client.test.js # LLama テスト(9テスト) -bun test test/integration.test.js # 統合テスト(4テスト) -``` - -### ウォッチモード(ファイル変更時に自動実行) - -```bash -bun run test:watch -``` - -**テスト結果**: 35 テスト全て成功 (139 expect calls) - -## プロジェクト構造 - -``` -sqlitevector/ -├── src/ -│ ├── main.js # Electron 起動・常駐処理 -│ ├── mcp-server.js # MCP SSE サーバー -│ ├── mcp-tools.js # ツール定義 -│ ├── mcp-handlers.js # ツール実装 -│ ├── db.js # DB 接続・初期化 -│ ├── llama-client.js # llama.cpp API -│ └── index.html # GUI ウィンドウ -├── test/ -│ ├── setup.js # テスト用ユーティリティ -│ ├── db.test.js # DB テスト -│ ├── mcp-tools.test.js # ツール定義テスト -│ ├── mcp-handlers.test.js -│ ├── llama-client.test.js -│ ├── integration.test.js -│ └── README.md # テスト詳細ドキュメント -├── document/ -│ ├── overview.md # アーキテクチャ解説 -│ └── openapi.yaml # REST API 仕様 -├── journals/ -│ └── 20260206-0000-案件.md # 作業報告書 -├── .env # 環境変数 -├── .gitignore -├── package.json -└── README.md (this file) -``` - -## アーキテクチャ - -### データフロー +## システムアーキテクチャ ```mermaid graph TD - A[MCP Client] -->|SSE| B[Express /sse] - B --> C[MCP Server] - C --> D{Tool Handler} - D -->|embedding| L[llama.cpp API] - D -->|insert| E[knex: items] - D -->|insert| F[sqlite-vec: vec_items] - D -->|search| F - F --> G[Results] - E --> G - G --> C - C --> B - B --> A + subgraph "Frontend (Webview2)" + UI[User Interface / React] + end + + subgraph "Tauri Backend (Rust / v2)" + CMD[Tauri Commands] + MCP[MCP SSE Server: Axum] + DB[(SQLite + SeaORM)] + VEC[sqlite-vec Extension] + LLM[Llama Client: reqwest] + end + + subgraph "External Resources" + llama[llama.cpp Server] + end + + UI <--> CMD + MCP <--> DB + DB <--> VEC + DB <--> LLM + LLM <--> llama ``` -### データベーススキーマ +## フォルダ構成 -``` -items table: - ├── id (PRIMARY KEY) - ├── content (TEXT) - アイテムテキスト - ├── path (TEXT) - ファイルパス参照 - ├── created_at (TIMESTAMP) - └── updated_at (TIMESTAMP) - -vec_items table (sqlite-vec): - ├── id (PRIMARY KEY) - └── embedding (float[VEC_DIM]) - ベクトル +```text +sqlitevector/ +├── src/ +│ └── frontend/ # UI 資産 (HTML/JS/CSS) +├── src-tauri/ +│ ├── src/ +│ │ ├── entities/ # SeaORM エンティティ定義 +│ │ ├── db.rs # データベース接続・初期化 +│ │ ├── llama.rs # LLM (llama.cpp) クライアント +│ │ ├── mcp.rs # MCP SSE ハンドラー実装 +│ │ └── lib.rs # エントリーポイント・Tauri 設定 +│ ├── tests/ # 統合テスト +│ └── Cargo.toml # Rust 依存関係 +├── journals/ # 開発記録 +└── package.json # Node.js/Bun 設定 ``` -## トラブルシューティング +## セットアップ -### `better-sqlite3 compilation error` +### プリリクエスト + +- [Rust](https://www.rust-lang.org/) +- [Bun](https://bun.sh/) +- [llama.cpp](https://github.com/ggerganov/llama.cpp) (サーバーモードで起動していること) + +### 開発・起動 + +1. **パッケージのインストール** + + ```bash + bun install + ``` + +2. **開発サーバーの起動** + + ```bash + bun tauri dev + ``` + +3. **テストの実行** + + ```bash + cd src-tauri + cargo test --lib + ``` + +## ビルド + +リリース用バイナリ (EXE/MSI) を生成します。 ```bash -electron-rebuild +bun tauri build ``` -### MCP サーバーが接続できない - -1. ポート確認: `netstat -an | grep 3000`(Windows: `netstat -ano | findstr "3000"`) -2. `mcp.json` の URL を確認: `http://localhost:3000/sse` -3. `.env` の `MCP_PORT` を確認 - -### llama.cpp 接続エラー - -```bash -# llama.cpp サーバーが起動しているか確認 -curl http://127.0.0.1:8080/health - -# .env の LLAMA_CPP_BASE_URL を確認 -``` - -### テストが失敗する - -```bash -# DB ファイルをクリア -rm -f vector.db test-*.db - -# テストを再実行 -bun test -``` - -## 開発ガイド - -### 新しいツールを追加 - -1. `src/mcp-tools.js` で定義 -2. `src/mcp-handlers.js` で実装 -3. `test/` でテスト作成 - -### コード品質 - -- すべてのツール変更は `test/` でテスト作成 -- 既存の 35 テストは常に成功を維持 - -## ⚠️ 重要な注意事項 - -### knex はベクトル検索に対応していません - -ベクトル検索は **必ず** `db.prepare()` + raw SQL で実装してください。詳細は [document/overview.md](document/overview.md#%EF%B8%8F-%E9%87%8D%E8%A6%81%E3%81%AA%E6%B3%A8%E6%84%8F%E4%BA%8B%E9%A0%85) を参照。 - -### sqlite-vec は Alpha 版 - -- 本番環境での使用は慎重に検討してください -- 安定版のリリースを待つか、代替ベクトルDB を検討してください - ## ライセンス -ISC License - 詳細は [LICENSE](LICENSE) を参照 - -## 貢献 - -プルリクエストを歓迎します! - -## 参考リンク - -- [MCP Specification](https://modelcontextprotocol.io) -- [sqlite-vec](https://github.com/asg017/sqlite-vec) -- [llama.cpp](https://github.com/ggerganov/llama.cpp) -- [Electron](https://www.electronjs.org) -- [Bun](https://bun.sh) +ISC License diff --git a/bun.lock b/bun.lock index b4c0b5d..9e6df27 100644 --- a/bun.lock +++ b/bun.lock @@ -13,6 +13,7 @@ "sqlite-vec": "^0.1.7-alpha.2", }, "devDependencies": { + "@tauri-apps/cli": "^2.10.0", "@types/bun": "latest", "@types/express": "^5.0.6", "electron": "^40.1.0", @@ -42,6 +43,30 @@ "@szmarczak/http-timer": ["@szmarczak/http-timer@4.0.6", "", { "dependencies": { "defer-to-connect": "^2.0.0" } }, "sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w=="], + "@tauri-apps/cli": ["@tauri-apps/cli@2.10.0", "", { "optionalDependencies": { "@tauri-apps/cli-darwin-arm64": "2.10.0", "@tauri-apps/cli-darwin-x64": "2.10.0", "@tauri-apps/cli-linux-arm-gnueabihf": "2.10.0", "@tauri-apps/cli-linux-arm64-gnu": "2.10.0", "@tauri-apps/cli-linux-arm64-musl": "2.10.0", "@tauri-apps/cli-linux-riscv64-gnu": "2.10.0", "@tauri-apps/cli-linux-x64-gnu": "2.10.0", "@tauri-apps/cli-linux-x64-musl": "2.10.0", "@tauri-apps/cli-win32-arm64-msvc": "2.10.0", "@tauri-apps/cli-win32-ia32-msvc": "2.10.0", "@tauri-apps/cli-win32-x64-msvc": "2.10.0" }, "bin": { "tauri": "tauri.js" } }, "sha512-ZwT0T+7bw4+DPCSWzmviwq5XbXlM0cNoleDKOYPFYqcZqeKY31KlpoMW/MOON/tOFBPgi31a2v3w9gliqwL2+Q=="], + + "@tauri-apps/cli-darwin-arm64": ["@tauri-apps/cli-darwin-arm64@2.10.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-avqHD4HRjrMamE/7R/kzJPcAJnZs0IIS+1nkDP5b+TNBn3py7N2aIo9LIpy+VQq0AkN8G5dDpZtOOBkmWt/zjA=="], + + "@tauri-apps/cli-darwin-x64": ["@tauri-apps/cli-darwin-x64@2.10.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-keDmlvJRStzVFjZTd0xYkBONLtgBC9eMTpmXnBXzsHuawV2q9PvDo2x6D5mhuoMVrJ9QWjgaPKBBCFks4dK71Q=="], + + "@tauri-apps/cli-linux-arm-gnueabihf": ["@tauri-apps/cli-linux-arm-gnueabihf@2.10.0", "", { "os": "linux", "cpu": "arm" }, "sha512-e5u0VfLZsMAC9iHaOEANumgl6lfnJx0Dtjkd8IJpysZ8jp0tJ6wrIkto2OzQgzcYyRCKgX72aKE0PFgZputA8g=="], + + "@tauri-apps/cli-linux-arm64-gnu": ["@tauri-apps/cli-linux-arm64-gnu@2.10.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-YrYYk2dfmBs5m+OIMCrb+JH/oo+4FtlpcrTCgiFYc7vcs6m3QDd1TTyWu0u01ewsCtK2kOdluhr/zKku+KP7HA=="], + + "@tauri-apps/cli-linux-arm64-musl": ["@tauri-apps/cli-linux-arm64-musl@2.10.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-GUoPdVJmrJRIXFfW3Rkt+eGK9ygOdyISACZfC/bCSfOnGt8kNdQIQr5WRH9QUaTVFIwxMlQyV3m+yXYP+xhSVA=="], + + "@tauri-apps/cli-linux-riscv64-gnu": ["@tauri-apps/cli-linux-riscv64-gnu@2.10.0", "", { "os": "linux", "cpu": "none" }, "sha512-JO7s3TlSxshwsoKNCDkyvsx5gw2QAs/Y2GbR5UE2d5kkU138ATKoPOtxn8G1fFT1aDW4LH0rYAAfBpGkDyJJnw=="], + + "@tauri-apps/cli-linux-x64-gnu": ["@tauri-apps/cli-linux-x64-gnu@2.10.0", "", { "os": "linux", "cpu": "x64" }, "sha512-Uvh4SUUp4A6DVRSMWjelww0GnZI3PlVy7VS+DRF5napKuIehVjGl9XD0uKoCoxwAQBLctvipyEK+pDXpJeoHng=="], + + "@tauri-apps/cli-linux-x64-musl": ["@tauri-apps/cli-linux-x64-musl@2.10.0", "", { "os": "linux", "cpu": "x64" }, "sha512-AP0KRK6bJuTpQ8kMNWvhIpKUkQJfcPFeba7QshOQZjJ8wOS6emwTN4K5g/d3AbCMo0RRdnZWwu67MlmtJyxC1Q=="], + + "@tauri-apps/cli-win32-arm64-msvc": ["@tauri-apps/cli-win32-arm64-msvc@2.10.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-97DXVU3dJystrq7W41IX+82JEorLNY+3+ECYxvXWqkq7DBN6FsA08x/EFGE8N/b0LTOui9X2dvpGGoeZKKV08g=="], + + "@tauri-apps/cli-win32-ia32-msvc": ["@tauri-apps/cli-win32-ia32-msvc@2.10.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-EHyQ1iwrWy1CwMalEm9z2a6L5isQ121pe7FcA2xe4VWMJp+GHSDDGvbTv/OPdkt2Lyr7DAZBpZHM6nvlHXEc4A=="], + + "@tauri-apps/cli-win32-x64-msvc": ["@tauri-apps/cli-win32-x64-msvc@2.10.0", "", { "os": "win32", "cpu": "x64" }, "sha512-NTpyQxkpzGmU6ceWBTY2xRIEaS0ZLbVx1HE1zTA3TY/pV3+cPoPPOs+7YScr4IMzXMtOw7tLw5LEXo5oIG3qaQ=="], + "@tootallnate/once": ["@tootallnate/once@2.0.0", "", {}, "sha512-XCuKFP5PS55gnMVu3dty8KPatLqUoy/ZYzDzAGCQ8JNFCkLXzmI7vNHCR+XpbZaMWQK/vQubr7PkYq8g470J/A=="], "@types/body-parser": ["@types/body-parser@1.19.6", "", { "dependencies": { "@types/connect": "*", "@types/node": "*" } }, "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g=="], diff --git a/launch.cmd b/launch.cmd new file mode 100644 index 0000000..ce1edf8 --- /dev/null +++ b/launch.cmd @@ -0,0 +1,3 @@ +@echo off +echo Starting SQLite Vector (Tauri version)... +bun tauri dev diff --git a/package.json b/package.json index 619f539..bbc50e5 100644 --- a/package.json +++ b/package.json @@ -33,9 +33,11 @@ "test:tools": "bun test test/mcp-tools.test.js", "test:handlers": "bun test test/mcp-handlers.test.js", "test:integration": "bun test test/integration.test.js", + "tauri": "tauri", "test:watch": "bun test --watch test/**/*.test.js" }, "devDependencies": { + "@tauri-apps/cli": "^2.10.0", "@types/bun": "latest", "@types/express": "^5.0.6", "electron": "^40.1.0", @@ -53,4 +55,4 @@ "sharer.js": "^0.5.3", "sqlite-vec": "^0.1.7-alpha.2" } -} +} \ No newline at end of file diff --git a/src-tauri/.gitignore b/src-tauri/.gitignore new file mode 100644 index 0000000..502406b --- /dev/null +++ b/src-tauri/.gitignore @@ -0,0 +1,4 @@ +# Generated by Cargo +# will have compiled files and executables +/target/ +/gen/schemas diff --git a/src-tauri/Cargo.lock b/src-tauri/Cargo.lock new file mode 100644 index 0000000..e407414 --- /dev/null +++ b/src-tauri/Cargo.lock @@ -0,0 +1,6646 @@ +# 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 = "ahash" +version = "0.8.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a15f179cd60c4584b8a8c596927aadc462e27f2ca70c04e0071964a73ba7a75" +dependencies = [ + "cfg-if", + "once_cell", + "version_check", + "zerocopy", +] + +[[package]] +name = "aho-corasick" +version = "1.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ddd31a130427c27518df266943a5308ed92d4b226cc639f5a8f1002816174301" +dependencies = [ + "memchr", +] + +[[package]] +name = "aliasable" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "250f629c0161ad8107cf89319e990051fae62832fd343083bea452d93e2205fd" + +[[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 = "allocator-api2" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" + +[[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.101" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f0e0fee31ef5ed1ba1316088939cea399010ed7731dba877ed44aeb407a75ea" + +[[package]] +name = "app" +version = "0.1.0" +dependencies = [ + "anyhow", + "axum", + "dotenvy", + "futures", + "log", + "mockito", + "reqwest 0.12.28", + "rusqlite", + "sea-orm", + "serde", + "serde_json", + "tauri", + "tauri-build", + "tauri-plugin-log", + "tokio", + "tower-http 0.5.2", +] + +[[package]] +name = "arrayvec" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" + +[[package]] +name = "assert-json-diff" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47e4f2b81832e72834d7518d8487a0396a28cc408186a2e8854c0f98011faf12" +dependencies = [ + "serde", + "serde_json", +] + +[[package]] +name = "async-stream" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b5a71a6f37880a80d1d7f19efd781e4b5de42c88f0722cc13bcb6cc2cfe8476" +dependencies = [ + "async-stream-impl", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-stream-impl" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7c24de15d275a1ecfd47a380fb4d5ec9bfe0933f309ed5e705b775596a3574d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "async-trait" +version = "0.1.89" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9035ad2d096bed7955a320ee7e2230574d28fd3c3a0f186cbea1ff3c7eed5dbb" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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 = "atoi" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f28d99ec8bfea296261ca1af174f24225171fea9664ba9003cbebee704810528" +dependencies = [ + "num-traits", +] + +[[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 = "axum" +version = "0.7.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edca88bc138befd0323b20752846e6587272d3b03b0343c8ea28a6f819e6e71f" +dependencies = [ + "async-trait", + "axum-core", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "itoa", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "serde_json", + "serde_path_to_error", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tower", + "tower-layer", + "tower-service", + "tracing", +] + +[[package]] +name = "axum-core" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09f2bd6146b97ae3359fa0cc6d6b376d9539582c7b4220f041a33ec24c226199" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http", + "http-body", + "http-body-util", + "mime", + "pin-project-lite", + "rustversion", + "sync_wrapper", + "tower-layer", + "tower-service", + "tracing", +] + +[[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 = "base64ct" +version = "1.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2af50177e190e07a26ab74f8b1efbfe2ef87da2116221318cb1c2e82baf7de06" + +[[package]] +name = "bigdecimal" +version = "0.4.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d6867f1565b3aad85681f1015055b087fcfd840d6aeee6eee7f2da317603695" +dependencies = [ + "autocfg", + "libm", + "num-bigint", + "num-integer", + "num-traits", + "serde", +] + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bitflags" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "812e12b5285cc515a9c72a5c1d3b6d46a19dac5acfef5265968c166106e31dd3" +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.4.0", + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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 = "bumpalo" +version = "3.19.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dd9dc738b7a8311c7ade152424974d8115f2cdad61e8dab8dac9f2362298510" + +[[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.10.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.11+spec-1.1.0", +] + +[[package]] +name = "cc" +version = "1.2.55" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47b26a0954ae34af09b50f0de26458fa95369a0d478d8236d3f93082b219bd29" +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.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fac4744fb15ae8337dc853fee7fb3f4e48c0fbaa23d0afe49c447b4fab126118" +dependencies = [ + "iana-time-zone", + "num-traits", + "serde", + "windows-link 0.2.1", +] + +[[package]] +name = "colored" +version = "3.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "faf9468729b8cbcea668e36183cb69d317348c2e08e994829fb56ebfdfbaac34" +dependencies = [ + "windows-sys 0.61.2", +] + +[[package]] +name = "combine" +version = "4.6.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd" +dependencies = [ + "bytes", + "memchr", +] + +[[package]] +name = "concurrent-queue" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ca0197aee26d1ae37445ee532fefce43251d24cc7c166799f4d46817f1d3973" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "const-oid" +version = "0.9.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2459377285ad874054d797f3ccebf984978aa39129f6eafde5cdc8315b612f8" + +[[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.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[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.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa95a34622365fa5bbf40b20b75dba8dfa8c94c734aea8ac9a5ca38af14316f1" +dependencies = [ + "bitflags 2.10.0", + "core-foundation 0.10.1", + "core-graphics-types", + "foreign-types 0.5.0", + "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.10.0", + "core-foundation 0.10.1", + "libc", +] + +[[package]] +name = "cpufeatures" +version = "0.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280" +dependencies = [ + "libc", +] + +[[package]] +name = "crc" +version = "3.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5eb8a2a1cd12ab0d987a5d5e825195d372001a4094a0376319d5a0ad71c1ba0d" +dependencies = [ + "crc-catalog", +] + +[[package]] +name = "crc-catalog" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5" + +[[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-queue" +version = "0.3.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f58bbc28f91df819d0aa2a2c00cd19754769c2fad90579b3592b1c9ba7a3115" +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.114", +] + +[[package]] +name = "ctor" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32a2785755761f3ddc1492979ce1e48d2c00d09311c39e4466429188f3dd6501" +dependencies = [ + "quote", + "syn 2.0.114", +] + +[[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.114", +] + +[[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.114", +] + +[[package]] +name = "der" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e7c1832837b905bbfb5101e07cc24c8deddf52f93225eee6ead5f4d63d53ddcb" +dependencies = [ + "const-oid", + "pem-rfc7468", + "zeroize", +] + +[[package]] +name = "deranged" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ececcb659e7ba858fb4f10388c250a7252eb0a27373f1a72b8748afdd248e587" +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.114", +] + +[[package]] +name = "derive_more" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d751e9e49156b02b44f9c1815bcb94b984cdcc4396ecc32521c739452808b134" +dependencies = [ + "derive_more-impl", +] + +[[package]] +name = "derive_more-impl" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "799a97264921d8623a957f6c3b9011f3b5492f557bbb7a5a19b7fa6d06ba8dcb" +dependencies = [ + "proc-macro2", + "quote", + "rustc_version", + "syn 2.0.114", + "unicode-xid", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "const-oid", + "crypto-common", + "subtle", +] + +[[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 = "dispatch" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd0c93bb4b0c6d9b77f4435b0ae98c24d17f1c45b2ff844c6151a07256ca923b" + +[[package]] +name = "dispatch2" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89a09f22a6c6069a18470eb92d2298acf25463f14256d24778e1230d789a2aec" +dependencies = [ + "bitflags 2.10.0", + "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.114", +] + +[[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.114", +] + +[[package]] +name = "dotenvy" +version = "0.15.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b" + +[[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 = "either" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48c757948c5ede0e46177b7add2e67155f70e33c07fea8284df6576da70b3719" +dependencies = [ + "serde", +] + +[[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.11+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 = "encoding_rs" +version = "0.8.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75030f3c4f45dafd7586dd6780965a8c7e8e285a5ecb86713e63a79c5b2766f3" +dependencies = [ + "cfg-if", +] + +[[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.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89e8918065695684b2b0702da20382d5ae6065cf3327bc2d6436bd49a71ce9f3" +dependencies = [ + "serde", + "serde_core", + "typeid", +] + +[[package]] +name = "errno" +version = "0.3.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "39cab71617ae0d63f51a36d69f866391735b51691dbda63cf6f96d042b63efeb" +dependencies = [ + "libc", + "windows-sys 0.61.2", +] + +[[package]] +name = "etcetera" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "136d1b5283a1ab77bd9257427ffd09d8667ced0570b6f938942bc7568ed5b943" +dependencies = [ + "cfg-if", + "home", + "windows-sys 0.48.0", +] + +[[package]] +name = "event-listener" +version = "5.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13b66accf52311f30a0db42147dadea9850cb48cd070028831ae5f5d4b856ab" +dependencies = [ + "concurrent-queue", + "parking", + "pin-project-lite", +] + +[[package]] +name = "fallible-iterator" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2acce4a10f12dc2fb14a218589d4f1f62ef011b2d0cc4b3cb1bba8e94da14649" + +[[package]] +name = "fallible-streaming-iterator" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7360491ce676a36bf9bb3c56c1aa791658183a54d2744120f27285738d90465a" + +[[package]] +name = "fastrand" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be" + +[[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 = "flume" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da0e4dd2a88388a1f4ccc7c9ce104604dab68d9f408dc34cd45823d5a9069095" +dependencies = [ + "futures-core", + "futures-sink", + "spin", +] + +[[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.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" +dependencies = [ + "foreign-types-shared 0.1.1", +] + +[[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 0.3.1", +] + +[[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.114", +] + +[[package]] +name = "foreign-types-shared" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" + +[[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" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65bc07b1a8bc7c85c5f2e110c476c7389b4554ba72af57d8445ea63a576b0876" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2dff15bf788c671c1934e366d07e30c1814a8ef514e1af724a602e8a2fbe1b10" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e" + +[[package]] +name = "futures-executor" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e28d1d997f585e54aebc3f97d39e72338912123a67330d723fdbb564d646c9f" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-intrusive" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d930c203dd0b6ff06e0201a4a2fe9149b43c684fd4420555b26d21b1a02956f" +dependencies = [ + "futures-core", + "lock_api", + "parking_lot", +] + +[[package]] +name = "futures-io" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e5c1b78ca4aae1ac06c48a526a655760685149f0d465d21f37abfe57ce075c6" + +[[package]] +name = "futures-macro" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "futures-sink" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e575fab7d1e0dcb8d0c7bcf9a63ee213816ab51902e6d244a95819acacf1d4f7" + +[[package]] +name = "futures-task" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f90f7dce0722e95104fcb095585910c0977252f286e354b5e3bd38902cd99988" + +[[package]] +name = "futures-util" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "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", + "wasip2", +] + +[[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.10.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.114", +] + +[[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.114", +] + +[[package]] +name = "h2" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f44da3a8150a6703ed5d34e164b875fd14c2cdab9af1252a9a1020bde2bdc54" +dependencies = [ + "atomic-waker", + "bytes", + "fnv", + "futures-core", + "futures-sink", + "http", + "indexmap 2.13.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" +dependencies = [ + "ahash 0.7.8", +] + +[[package]] +name = "hashbrown" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" +dependencies = [ + "ahash 0.8.12", +] + +[[package]] +name = "hashbrown" +version = "0.15.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9229cfe53dfd69f0609a49f65461bd93001ea1ef889cd5529dd176593f5338a1" +dependencies = [ + "allocator-api2", + "equivalent", + "foldhash", +] + +[[package]] +name = "hashbrown" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "841d1cc9bed7f9236f321df977030373f4a4163ae1a7dbfe1a51a2c1a51d9100" + +[[package]] +name = "hashlink" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ba4ff7128dee98c7dc9794b6a411377e1404dba1c97deb8d1a55297bd25d8af" +dependencies = [ + "hashbrown 0.14.5", +] + +[[package]] +name = "hashlink" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7382cf6263419f2d8df38c55d7da83da5c18aef87fc7a7fc1fb1e344edfe14c1" +dependencies = [ + "hashbrown 0.15.5", +] + +[[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 = "hkdf" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b5f8eb2ad728638ea2c7d47a21db23b7b58a72ed6a38256b8a1849f15fbbdf7" +dependencies = [ + "hmac", +] + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest", +] + +[[package]] +name = "home" +version = "0.5.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc627f471c528ff0c4a49e1d5e60450c8f6461dd6d10ba9dcd3a61d3dff7728d" +dependencies = [ + "windows-sys 0.61.2", +] + +[[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 = "httpdate" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" + +[[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", + "h2", + "http", + "http-body", + "httparse", + "httpdate", + "itoa", + "pin-project-lite", + "pin-utils", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3c93eb611681b207e1fe55d5a71ecf91572ec8a6705cdb6857f7d8d5242cf58" +dependencies = [ + "http", + "hyper", + "hyper-util", + "rustls", + "rustls-pki-types", + "tokio", + "tokio-rustls", + "tower-service", +] + +[[package]] +name = "hyper-tls" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70206fc6890eaca9fde8a0bf71caa2ddfc9fe045ac9e5c70df101a7dbde866e0" +dependencies = [ + "bytes", + "http-body-util", + "hyper", + "hyper-util", + "native-tls", + "tokio", + "tokio-native-tls", + "tower-service", +] + +[[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", + "system-configuration", + "tokio", + "tower-service", + "tracing", + "windows-registry", +] + +[[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 = "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 = "inherent" +version = "1.0.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c727f80bfa4a6c6e2508d2f05b6f4bfce242030bd88ed15ae5331c5b5d30fba7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "ipnet" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "469fb0b9cefa57e3ef31275ee7cacb78f2fdca44e4765491884a2b119d4eb130" + +[[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.85" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c942ebf8e95485ca0d52d97da7c5a2c387d0e7f0ba4c35e93bfcaee045955b3" +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.10.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 = "lazy_static" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" +dependencies = [ + "spin", +] + +[[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.180" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bcc35a38544a891a5f7c865aca548a982ccb3b8650a5b06d0fd33a10283c56fc" + +[[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 = "libm" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6d2cec3eae94f9f509c767b45932f1ada8350c4bdb85af2fcab4a3c14807981" + +[[package]] +name = "libredox" +version = "0.1.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d0b95e02c851351f877147b7deea7b1afb1df71b63aa5f8270716e0c5720616" +dependencies = [ + "bitflags 2.10.0", + "libc", + "redox_syscall 0.7.0", +] + +[[package]] +name = "libsqlite3-sys" +version = "0.30.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e99fb7a497b1e3339bc746195567ed8d3e24945ecd636e3619d20b9de9e9149" +dependencies = [ + "cc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "linux-raw-sys" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df1d3c3b53da64cf5760482273a98e575c651a67eec7f77df96b5b642de8f039" + +[[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.114", +] + +[[package]] +name = "matches" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5" + +[[package]] +name = "matchit" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" + +[[package]] +name = "md-5" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d89e7ee0cfbedfc4da3340218492196241d89eefb6dab27de5df917a6d2e78cf" +dependencies = [ + "cfg-if", + "digest", +] + +[[package]] +name = "memchr" +version = "2.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f52b00d39961fc5b2736ea853c9cc86238e165017a493d1d5c8eac6bdc4cc273" + +[[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 = "mockito" +version = "1.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90820618712cab19cfc46b274c6c22546a82affcb3c3bdf0f29e3db8e1bb92c0" +dependencies = [ + "assert-json-diff", + "bytes", + "colored", + "futures-core", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "log", + "pin-project-lite", + "rand 0.9.2", + "regex", + "serde_json", + "serde_urlencoded", + "similar", + "tokio", +] + +[[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 = "native-tls" +version = "0.2.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87de3442987e9dbec73158d5c715e7ad9072fda936bb03d19d7fa10e00520f0e" +dependencies = [ + "libc", + "log", + "openssl", + "openssl-probe", + "openssl-sys", + "schannel", + "security-framework", + "security-framework-sys", + "tempfile", +] + +[[package]] +name = "ndk" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3f42e7bbe13d351b6bead8286a43aac9534b82bd3cc43e47037f012ebfd62d4" +dependencies = [ + "bitflags 2.10.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-bigint" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9" +dependencies = [ + "num-integer", + "num-traits", +] + +[[package]] +name = "num-bigint-dig" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e661dda6640fad38e827a6d4a310ff4763082116fe217f279885c97f511bb0b7" +dependencies = [ + "lazy_static", + "libm", + "num-integer", + "num-iter", + "num-traits", + "rand 0.8.5", + "smallvec", + "zeroize", +] + +[[package]] +name = "num-conv" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf97ec579c3c42f953ef76dbf8d55ac91fb219dde70e49aa4a6b7d74e9919050" + +[[package]] +name = "num-integer" +version = "0.1.46" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f" +dependencies = [ + "num-traits", +] + +[[package]] +name = "num-iter" +version = "0.1.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1429034a0490724d0075ebb2bc9e875d6503c3cf69e235a8941aa757d83ef5bf" +dependencies = [ + "autocfg", + "num-integer", + "num-traits", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", + "libm", +] + +[[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.4.0", + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7c2599ce0ec54857b29ce62166b0ed9b4f6f1a70ccc9a71165b6154caca8c05" +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.10.0", + "block2", + "libc", + "objc2", + "objc2-cloud-kit", + "objc2-core-data", + "objc2-core-foundation", + "objc2-core-graphics", + "objc2-core-image", + "objc2-core-text", + "objc2-core-video", + "objc2-foundation", + "objc2-quartz-core", +] + +[[package]] +name = "objc2-cloud-kit" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73ad74d880bb43877038da939b7427bba67e9dd42004a18b809ba7d87cee241c" +dependencies = [ + "bitflags 2.10.0", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-data" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b402a653efbb5e82ce4df10683b6b28027616a2715e90009947d50b8dd298fa" +dependencies = [ + "bitflags 2.10.0", + "objc2", + "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.10.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.10.0", + "dispatch2", + "objc2", + "objc2-core-foundation", + "objc2-io-surface", +] + +[[package]] +name = "objc2-core-image" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5d563b38d2b97209f8e861173de434bd0214cf020e3423a52624cd1d989f006" +dependencies = [ + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-text" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0cde0dfb48d25d2b4862161a4d5fcc0e3c24367869ad306b0c9ec0073bfed92d" +dependencies = [ + "bitflags 2.10.0", + "objc2", + "objc2-core-foundation", + "objc2-core-graphics", +] + +[[package]] +name = "objc2-core-video" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d425caf1df73233f29fd8a5c3e5edbc30d2d4307870f802d18f00d83dc5141a6" +dependencies = [ + "bitflags 2.10.0", + "objc2", + "objc2-core-foundation", + "objc2-core-graphics", + "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.10.0", + "block2", + "libc", + "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.10.0", + "objc2", + "objc2-core-foundation", +] + +[[package]] +name = "objc2-javascript-core" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a1e6550c4caed348956ce3370c9ffeca70bb1dbed4fa96112e7c6170e074586" +dependencies = [ + "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.10.0", + "objc2", + "objc2-core-foundation", + "objc2-foundation", +] + +[[package]] +name = "objc2-security" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "709fe137109bd1e8b5a99390f77a7d8b2961dafc1a1c5db8f2e60329ad6d895a" +dependencies = [ + "bitflags 2.10.0", + "objc2", + "objc2-core-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.10.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.10.0", + "block2", + "objc2", + "objc2-app-kit", + "objc2-core-foundation", + "objc2-foundation", + "objc2-javascript-core", + "objc2-security", +] + +[[package]] +name = "once_cell" +version = "1.21.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42f5e15c9953c5e4ccceeb2e7382a716482c34515315f7b03532b8b4e8393d2d" + +[[package]] +name = "openssl" +version = "0.10.75" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08838db121398ad17ab8531ce9de97b244589089e290a384c900cb9ff7434328" +dependencies = [ + "bitflags 2.10.0", + "cfg-if", + "foreign-types 0.3.2", + "libc", + "once_cell", + "openssl-macros", + "openssl-sys", +] + +[[package]] +name = "openssl-macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "openssl-probe" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d05e27ee213611ffe7d6348b942e8f942b37114c00cc03cec254295a4a17852e" + +[[package]] +name = "openssl-sys" +version = "0.9.111" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "82cab2d520aa75e3c58898289429321eb788c3106963d0dc886ec7a5f4adc321" +dependencies = [ + "cc", + "libc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "option-ext" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" + +[[package]] +name = "ordered-float" +version = "4.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7bb71e1b3fa6ca1c61f383464aaf2bb0e2f8e772a1f01d486832464de363b951" +dependencies = [ + "num-traits", +] + +[[package]] +name = "ouroboros" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e0f050db9c44b97a94723127e6be766ac5c340c48f2c4bb3ffa11713744be59" +dependencies = [ + "aliasable", + "ouroboros_macro", + "static_assertions", +] + +[[package]] +name = "ouroboros_macro" +version = "0.18.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c7028bdd3d43083f6d8d4d5187680d0d3560d54df4cc9d752005268b41e64d0" +dependencies = [ + "heck 0.4.1", + "proc-macro2", + "proc-macro2-diagnostics", + "quote", + "syn 2.0.114", +] + +[[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" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" + +[[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 0.5.18", + "smallvec", + "windows-link 0.2.1", +] + +[[package]] +name = "pem-rfc7468" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "88b39c9bfcfc231068454382784bb460aae594343fb030d46e9f50a645418412" +dependencies = [ + "base64ct", +] + +[[package]] +name = "percent-encoding" +version = "2.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220" + +[[package]] +name = "pgvector" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fc58e2d255979a31caa7cabfa7aac654af0354220719ab7a68520ae7a91e8c0b" +dependencies = [ + "serde", +] + +[[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.114", +] + +[[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.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b3cff922bd51709b605d9ead9aa71031d81447142d828eb4a6eba76fe619f9b" + +[[package]] +name = "pin-utils" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + +[[package]] +name = "pkcs1" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8ffb9f10fa047879315e6625af03c164b16962a5368d724ed16323b68ace47f" +dependencies = [ + "der", + "pkcs8", + "spki", +] + +[[package]] +name = "pkcs8" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f950b2377845cebe5cf8b5165cb3cc1a5e0fa5cfa3e1f7f55707d8fd82e0a7b7" +dependencies = [ + "der", + "spki", +] + +[[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 = "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.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "219cb19e96be00ab2e37d6e299658a0cfa83e52429179969b0f0121b4ac46983" +dependencies = [ + "toml_edit 0.23.10+spec-1.0.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-error-attr2" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96de42df36bb9bba5542fe9f1a054b8cc87e172759a1868aa05c1f3acc89dfc5" +dependencies = [ + "proc-macro2", + "quote", +] + +[[package]] +name = "proc-macro-error2" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11ec05c52be0a07b08061f7dd003e7d7092e0472bc731b4af7bb1ef876109802" +dependencies = [ + "proc-macro-error-attr2", + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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 = "proc-macro2-diagnostics" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af066a9c399a26e020ada66a034357a868728e72cd426f3adcd35f80d88d88c8" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", + "version_check", + "yansi", +] + +[[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.44" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21b2ebcf727b7760c461f091f9f0f539b77b8e87f2fd88131e7f1b433b3cece4" +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 = "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" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6db2770f06117d490610c7488547d543617b21bfa07796d7a12f6f1bd53850d1" +dependencies = [ + "rand_chacha 0.9.0", + "rand_core 0.9.5", +] + +[[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_chacha" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3022b5f1df60f26e1ffddd6c66e8aa15de382ae63b3a0c1bfc0e4d3e3f325cb" +dependencies = [ + "ppv-lite86", + "rand_core 0.9.5", +] + +[[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_core" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76afc826de14238e6e8c374ddcc1fa19e374fd8dd986b0d2af0d02377261d83c" +dependencies = [ + "getrandom 0.3.4", +] + +[[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.10.0", +] + +[[package]] +name = "redox_syscall" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49f3fe0889e69e2ae9e41f4d6c4c0181701d00e4697b356fb1f74173a5e0ee27" +dependencies = [ + "bitflags 2.10.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.114", +] + +[[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.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a96887878f22d7bad8a3b6dc5b7440e0ada9a245242924394987b21cf2210a4c" + +[[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.12.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eddd3ca559203180a307f12d114c268abf583f59b03cb906fd0b3ff8646c1147" +dependencies = [ + "base64 0.22.1", + "bytes", + "encoding_rs", + "futures-core", + "h2", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-rustls", + "hyper-tls", + "hyper-util", + "js-sys", + "log", + "mime", + "native-tls", + "percent-encoding", + "pin-project-lite", + "rustls-pki-types", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tokio-native-tls", + "tower", + "tower-http 0.6.8", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "reqwest" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04e9018c9d814e5f30cc16a0f03271aeab3571e609612d9fe78c1aa8d11c2f62" +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 0.6.8", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-streams", + "web-sys", +] + +[[package]] +name = "ring" +version = "0.17.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4689e6c2294d81e88dc6261c768b63bc4fcdb852be6d1352498b114f61383b7" +dependencies = [ + "cc", + "cfg-if", + "getrandom 0.2.17", + "libc", + "untrusted", + "windows-sys 0.52.0", +] + +[[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 = "rsa" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8573f03f5883dcaebdfcf4725caa1ecb9c15b2ef50c43a07b816e06799bb12d" +dependencies = [ + "const-oid", + "digest", + "num-bigint-dig", + "num-integer", + "num-traits", + "pkcs1", + "pkcs8", + "rand_core 0.6.4", + "signature", + "spki", + "subtle", + "zeroize", +] + +[[package]] +name = "rusqlite" +version = "0.32.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7753b721174eb8ff87a9a0e799e2d7bc3749323e773db92e0984debb00019d6e" +dependencies = [ + "bitflags 2.10.0", + "fallible-iterator", + "fallible-streaming-iterator", + "hashlink 0.9.1", + "libsqlite3-sys", + "smallvec", +] + +[[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 = "rustix" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "146c9e247ccc180c1f61615433868c99f3de3ae256a30a43b49f67c2d9171f34" +dependencies = [ + "bitflags 2.10.0", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.61.2", +] + +[[package]] +name = "rustls" +version = "0.23.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c665f33d38cea657d9614f766881e4d510e0eda4239891eea56b4cadcf01801b" +dependencies = [ + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-pki-types" +version = "1.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be040f8b0a225e40375822a563fa9524378b9d63112f53e19ffff34df5d33fdd" +dependencies = [ + "zeroize", +] + +[[package]] +name = "rustls-webpki" +version = "0.103.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7df23109aa6c1567d1c575b9952556388da57401e4ace1d15f79eedad0d8f53" +dependencies = [ + "ring", + "rustls-pki-types", + "untrusted", +] + +[[package]] +name = "rustversion" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b39cdef0fa800fc44525c84ccb54a029961a8215f9619753635a9c0d2538d46d" + +[[package]] +name = "ryu" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a50f4cf475b65d88e057964e0e9bb1f0aa9bbb2036dc65c64596b42932536984" + +[[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 = "schannel" +version = "0.1.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "891d81b926048e76efe18581bf793546b4c0eaf8448d72be8de2bbee5fd166e1" +dependencies = [ + "windows-sys 0.61.2", +] + +[[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.114", +] + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "sea-bae" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f694a6ab48f14bc063cfadff30ab551d3c7e46d8f81836c51989d548f44a2a25" +dependencies = [ + "heck 0.4.1", + "proc-macro-error2", + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "sea-orm" +version = "1.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d945f62558fac19e5988680d2fdf747b734c2dbc6ce2cb81ba33ed8dde5b103" +dependencies = [ + "async-stream", + "async-trait", + "bigdecimal", + "chrono", + "derive_more 2.1.1", + "futures-util", + "log", + "ouroboros", + "pgvector", + "rust_decimal", + "sea-orm-macros", + "sea-query", + "sea-query-binder", + "serde", + "serde_json", + "sqlx", + "strum", + "thiserror 2.0.18", + "time", + "tracing", + "url", + "uuid", +] + +[[package]] +name = "sea-orm-macros" +version = "1.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "84c2e64a50a9cc8339f10a27577e10062c7f995488e469f2c95762c5ee847832" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "sea-bae", + "syn 2.0.114", + "unicode-ident", +] + +[[package]] +name = "sea-query" +version = "0.32.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a5d1c518eaf5eda38e5773f902b26ab6d5e9e9e2bb2349ca6c64cf96f80448c" +dependencies = [ + "bigdecimal", + "chrono", + "inherent", + "ordered-float", + "rust_decimal", + "serde_json", + "time", + "uuid", +] + +[[package]] +name = "sea-query-binder" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b0019f47430f7995af63deda77e238c17323359af241233ec768aba1faea7608" +dependencies = [ + "bigdecimal", + "chrono", + "rust_decimal", + "sea-query", + "serde_json", + "sqlx", + "time", + "uuid", +] + +[[package]] +name = "seahash" +version = "4.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c107b6f4780854c8b126e228ea8869f4d7b71260f962fefb57b996b8959ba6b" + +[[package]] +name = "security-framework" +version = "2.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02" +dependencies = [ + "bitflags 2.10.0", + "core-foundation 0.9.4", + "core-foundation-sys", + "libc", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc1f0cbffaac4852523ce30d8bd3c5cdc873501d96ff467ca09b6767bb8cd5c0" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[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 0.99.20", + "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.114", +] + +[[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.114", +] + +[[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_path_to_error" +version = "0.1.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10a9ff822e371bb5403e391ecd83e182e0e77ba7f6fe0160b795797109d1b457" +dependencies = [ + "itoa", + "serde", + "serde_core", +] + +[[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.114", +] + +[[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_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "serde_with" +version = "3.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4fa237f2807440d238e0364a218270b98f767a00d3dada77b1c53ae88940e2e7" +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.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52a8e3ca0ca629121f70ab50f95249e5a6f925cc0f6ffe8256c45b728875706c" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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.114", +] + +[[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 = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + +[[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 = "signal-hook-registry" +version = "1.4.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4db69cba1110affc0e9f7bcd48bbf87b3f4fc7c61fc9155afd4c469eb3d6c1b" +dependencies = [ + "errno", + "libc", +] + +[[package]] +name = "signature" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de" +dependencies = [ + "digest", + "rand_core 0.6.4", +] + +[[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 = "similar" +version = "2.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbbb5d9659141646ae647b42fe094daf6c6192d1620870b449d9557f748b2daa" + +[[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" +dependencies = [ + "serde", +] + +[[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 0.5.18", + "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 = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +dependencies = [ + "lock_api", +] + +[[package]] +name = "spki" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d91ed6c858b01f942cd56b37a94b3e0a1798290327d1236e4d9cf4eaca44d29d" +dependencies = [ + "base64ct", + "der", +] + +[[package]] +name = "sqlx" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fefb893899429669dcdd979aff487bd78f4064e5e7907e4269081e0ef7d97dc" +dependencies = [ + "sqlx-core", + "sqlx-macros", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", +] + +[[package]] +name = "sqlx-core" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee6798b1838b6a0f69c007c133b8df5866302197e404e8b6ee8ed3e3a5e68dc6" +dependencies = [ + "base64 0.22.1", + "bigdecimal", + "bytes", + "chrono", + "crc", + "crossbeam-queue", + "either", + "event-listener", + "futures-core", + "futures-intrusive", + "futures-io", + "futures-util", + "hashbrown 0.15.5", + "hashlink 0.10.0", + "indexmap 2.13.0", + "log", + "memchr", + "once_cell", + "percent-encoding", + "rust_decimal", + "rustls", + "serde", + "serde_json", + "sha2", + "smallvec", + "thiserror 2.0.18", + "time", + "tokio", + "tokio-stream", + "tracing", + "url", + "uuid", + "webpki-roots 0.26.11", +] + +[[package]] +name = "sqlx-macros" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2d452988ccaacfbf5e0bdbc348fb91d7c8af5bee192173ac3636b5fb6e6715d" +dependencies = [ + "proc-macro2", + "quote", + "sqlx-core", + "sqlx-macros-core", + "syn 2.0.114", +] + +[[package]] +name = "sqlx-macros-core" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19a9c1841124ac5a61741f96e1d9e2ec77424bf323962dd894bdb93f37d5219b" +dependencies = [ + "dotenvy", + "either", + "heck 0.5.0", + "hex", + "once_cell", + "proc-macro2", + "quote", + "serde", + "serde_json", + "sha2", + "sqlx-core", + "sqlx-mysql", + "sqlx-postgres", + "sqlx-sqlite", + "syn 2.0.114", + "tokio", + "url", +] + +[[package]] +name = "sqlx-mysql" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa003f0038df784eb8fecbbac13affe3da23b45194bd57dba231c8f48199c526" +dependencies = [ + "atoi", + "base64 0.22.1", + "bigdecimal", + "bitflags 2.10.0", + "byteorder", + "bytes", + "chrono", + "crc", + "digest", + "dotenvy", + "either", + "futures-channel", + "futures-core", + "futures-io", + "futures-util", + "generic-array", + "hex", + "hkdf", + "hmac", + "itoa", + "log", + "md-5", + "memchr", + "once_cell", + "percent-encoding", + "rand 0.8.5", + "rsa", + "rust_decimal", + "serde", + "sha1", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror 2.0.18", + "time", + "tracing", + "uuid", + "whoami", +] + +[[package]] +name = "sqlx-postgres" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db58fcd5a53cf07c184b154801ff91347e4c30d17a3562a635ff028ad5deda46" +dependencies = [ + "atoi", + "base64 0.22.1", + "bigdecimal", + "bitflags 2.10.0", + "byteorder", + "chrono", + "crc", + "dotenvy", + "etcetera", + "futures-channel", + "futures-core", + "futures-util", + "hex", + "hkdf", + "hmac", + "home", + "itoa", + "log", + "md-5", + "memchr", + "num-bigint", + "once_cell", + "rand 0.8.5", + "rust_decimal", + "serde", + "serde_json", + "sha2", + "smallvec", + "sqlx-core", + "stringprep", + "thiserror 2.0.18", + "time", + "tracing", + "uuid", + "whoami", +] + +[[package]] +name = "sqlx-sqlite" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2d12fe70b2c1b4401038055f90f151b78208de1f9f89a7dbfd41587a10c3eea" +dependencies = [ + "atoi", + "chrono", + "flume", + "futures-channel", + "futures-core", + "futures-executor", + "futures-intrusive", + "futures-util", + "libsqlite3-sys", + "log", + "percent-encoding", + "serde", + "serde_urlencoded", + "sqlx-core", + "thiserror 2.0.18", + "time", + "tracing", + "url", + "uuid", +] + +[[package]] +name = "stable_deref_trait" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ce2be8dc25455e1f91df71bfa12ad37d7af1092ae736f3a6cd0e37bc7810596" + +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + +[[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 = "stringprep" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b4df3d392d81bd458a8a621b8bffbd2302a12ffe288a9d931670948749463b1" +dependencies = [ + "unicode-bidi", + "unicode-normalization", + "unicode-properties", +] + +[[package]] +name = "strsim" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" + +[[package]] +name = "strum" +version = "0.26.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fec0f0aef304996cf250b31b5a10dee7980c85da9d759361292b8bca5a18f06" + +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + +[[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.114" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d4d107df263a3013ef9b1879b0df87d706ff80f65a86ea879bd9c31f9b307c2a" +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.114", +] + +[[package]] +name = "system-configuration" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a13f3d0daba03132c0aa9767f98351b3488edc2c100cda2d2ec2b04f3d8d3c8b" +dependencies = [ + "bitflags 2.10.0", + "core-foundation 0.9.4", + "system-configuration-sys", +] + +[[package]] +name = "system-configuration-sys" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e1d1b10ced5ca923a1fcb8d03e96b8d3268065d724548c0211415ff6ac6bac4" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[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.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3a753bdc39c07b192151523a3f77cd0394aa75413802c883a0f6f6a0e5ee2e7" +dependencies = [ + "bitflags 2.10.0", + "block2", + "core-foundation 0.10.1", + "core-graphics", + "crossbeam-channel", + "dispatch", + "dlopen2", + "dpi", + "gdkwayland-sys", + "gdkx11-sys", + "gtk", + "jni", + "lazy_static", + "libc", + "log", + "ndk", + "ndk-context", + "ndk-sys", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "once_cell", + "parking_lot", + "raw-window-handle", + "scopeguard", + "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.114", +] + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "463ae8677aa6d0f063a900b9c41ecd4ac2b7ca82f0b058cc4491540e55b20129" +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 0.13.1", + "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.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca7bd893329425df750813e95bd2b643d5369d929438da96d5bbb7cc2c918f74" +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.11+spec-1.1.0", + "walkdir", +] + +[[package]] +name = "tauri-codegen" +version = "2.5.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aac423e5859d9f9ccdd32e3cf6a5866a15bedbf25aa6630bcb2acde9468f6ae3" +dependencies = [ + "base64 0.22.1", + "brotli", + "ico", + "json-patch", + "plist", + "png", + "proc-macro2", + "quote", + "semver", + "serde", + "serde_json", + "sha2", + "syn 2.0.114", + "tauri-utils", + "thiserror 2.0.18", + "time", + "url", + "uuid", + "walkdir", +] + +[[package]] +name = "tauri-macros" +version = "2.5.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b6a1bd2861ff0c8766b1d38b32a6a410f6dc6532d4ef534c47cfb2236092f59" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "syn 2.0.114", + "tauri-codegen", + "tauri-utils", +] + +[[package]] +name = "tauri-plugin" +version = "2.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "692a77abd8b8773e107a42ec0e05b767b8d2b7ece76ab36c6c3947e34df9f53f" +dependencies = [ + "anyhow", + "glob", + "plist", + "schemars 0.8.22", + "serde", + "serde_json", + "tauri-utils", + "toml 0.9.11+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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b885ffeac82b00f1f6fd292b6e5aabfa7435d537cef57d11e38a489956535651" +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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5204682391625e867d16584fedc83fc292fb998814c9f7918605c789cd876314" +dependencies = [ + "gtk", + "http", + "jni", + "log", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fcd169fccdff05eff2c1033210b9b94acd07a47e6fa9a3431cf09cfd4f01c87e" +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.11+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.11+spec-1.1.0", +] + +[[package]] +name = "tempfile" +version = "3.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "655da9c7eb6305c55742045d5a8d2037996d61d8de95806335c7c86ce0f82e9c" +dependencies = [ + "fastrand", + "getrandom 0.3.4", + "once_cell", + "rustix", + "windows-sys 0.61.2", +] + +[[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.114", +] + +[[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.114", +] + +[[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.49.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72a2903cd7736441aac9df9d7688bd0ce48edccaadf181c3b90be801e81d3d86" +dependencies = [ + "bytes", + "libc", + "mio", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "windows-sys 0.61.2", +] + +[[package]] +name = "tokio-macros" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af407857209536a95c8e56f8231ef2c2e2aff839b22e07a1ffcbc617e9db9fa5" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[package]] +name = "tokio-native-tls" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbae76ab933c85776efabc971569dd6119c580d8f5d448769dec1764bf796ef2" +dependencies = [ + "native-tls", + "tokio", +] + +[[package]] +name = "tokio-rustls" +version = "0.26.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1729aa945f29d91ba541258c8df89027d5792d85a8841fb65e8bf0f4ede4ef61" +dependencies = [ + "rustls", + "tokio", +] + +[[package]] +name = "tokio-stream" +version = "0.1.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32da49809aab5c3bc678af03902d4ccddea2a87d028d86392a4b1560c6906c70" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[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.11+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3afc9a848309fe1aaffaed6e1546a7a14de1f935dc9d89d32afd9a44bab7c46" +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.14", +] + +[[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_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.23.10+spec-1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "84c8b9f757e028cee9fa244aea147aab2a9ec09d5325a9b01e0a49730c2b5269" +dependencies = [ + "indexmap 2.13.0", + "toml_datetime 0.7.5+spec-1.1.0", + "toml_parser", + "winnow 0.7.14", +] + +[[package]] +name = "toml_parser" +version = "1.0.6+spec-1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3198b4b0a8e11f09dd03e133c0280504d0801269e9afa46362ffde1cbeebf44" +dependencies = [ + "winnow 0.7.14", +] + +[[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", + "tracing", +] + +[[package]] +name = "tower-http" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e9cd434a998747dd2c4276bc96ee2e0c7a2eadf3cae88e52be55a05fa9053f5" +dependencies = [ + "bitflags 2.10.0", + "bytes", + "http", + "http-body", + "http-body-util", + "pin-project-lite", + "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.10.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 = [ + "log", + "pin-project-lite", + "tracing-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7490cfa5ec963746568740651ac6781f701c9c5ea257c58e057f3ba8cf69e8da" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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-bidi" +version = "0.3.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c1cb5db39152898a79168971543b1cb5020dff7fe43c8dc468b0885f5e29df5" + +[[package]] +name = "unicode-ident" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9312f7c4f6ff9069b165498234ce8be658059c6728633667c526e27dc2cf1df5" + +[[package]] +name = "unicode-normalization" +version = "0.1.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5fd4f6878c9cb28d874b009da9e8d183b5abc80117c40bbd187a1fde336be6e8" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-properties" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7df058c713841ad818f1dc5d3fd88063241cc61f49f5fbea4b951e8cf5a8d71d" + +[[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 = "untrusted" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" + +[[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.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee48d38b119b0cd71fe4141b30f5ba9c7c5d9f4e7a3a8b4a674e4b6ef789976f" +dependencies = [ + "getrandom 0.3.4", + "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 = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + +[[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 = "wasite" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8dad83b4f25e74f184f64c43b150b91efe7647395b42289f38e50566d82855b" + +[[package]] +name = "wasm-bindgen" +version = "0.2.108" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64024a30ec1e37399cf85a7ffefebdb72205ca1c972291c51512360d90bd8566" +dependencies = [ + "cfg-if", + "once_cell", + "rustversion", + "wasm-bindgen-macro", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.58" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70a6e77fd0ae8029c9ea0063f87c46fde723e7d887703d74ad2616d792e51e6f" +dependencies = [ + "cfg-if", + "futures-util", + "js-sys", + "once_cell", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.108" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "008b239d9c740232e71bd39e8ef6429d27097518b6b30bdf9086833bd5b6d608" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.108" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5256bae2d58f54820e6490f9839c49780dff84c65aeab9e772f15d5f0e913a55" +dependencies = [ + "bumpalo", + "proc-macro2", + "quote", + "syn 2.0.114", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.108" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f01b580c9ac74c8d8f0c0e4afb04eeef2acf145458e52c03845ee9cd23e3d12" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "wasm-streams" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15053d8d85c7eccdbefef60f06769760a563c7f0a9d6902a13d35c7800b0ad65" +dependencies = [ + "futures-util", + "js-sys", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "web-sys" +version = "0.3.85" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "312e32e551d92129218ea9a2452120f4aabc03529ef03e4d0d82fb2780608598" +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 = "webpki-roots" +version = "0.26.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "521bc38abb08001b01866da9f51eb7c5d647a19260e00054a8c7fd5f9e57f7a9" +dependencies = [ + "webpki-roots 1.0.6", +] + +[[package]] +name = "webpki-roots" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22cfaf3c063993ff62e73cb4311efde4db1efb31ab78a3e5c457939ad5cc0bed" +dependencies = [ + "rustls-pki-types", +] + +[[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.114", +] + +[[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 = "whoami" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d4a4db5077702ca3015d3d02d74974948aba2ad9e12ab7df718ee64ccd7e97d" +dependencies = [ + "libredox", + "wasite", +] + +[[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.114", +] + +[[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.114", +] + +[[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-registry" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02752bf7fbdcce7f2a27a742f798510f3e5ad88dbe84871e5168e2120c3d5720" +dependencies = [ + "windows-link 0.2.1", + "windows-result 0.4.1", + "windows-strings 0.5.1", +] + +[[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.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.5", +] + +[[package]] +name = "windows-sys" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" +dependencies = [ + "windows-targets 0.52.6", +] + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" +dependencies = [ + "windows_aarch64_gnullvm 0.48.5", + "windows_aarch64_msvc 0.48.5", + "windows_i686_gnu 0.48.5", + "windows_i686_msvc 0.48.5", + "windows_x86_64_gnu 0.48.5", + "windows_x86_64_gnullvm 0.48.5", + "windows_x86_64_msvc 0.48.5", +] + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[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.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + +[[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.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a5364e9d77fcdeeaa6062ced926ee3381faa2ee02d3eb83a5c27a8825540829" +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" + +[[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.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ed1a195b0375491dd15a7066a10251be217ce743cf4bbbbdcf5391d6473bee0" +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 = "yansi" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cfe53a6657fd280eaa890a3bc59152892ffa3e30101319d168b781ed6529b049" + +[[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.114", + "synstructure", +] + +[[package]] +name = "zerocopy" +version = "0.8.39" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db6d35d663eadb6c932438e763b262fe1a70987f9ae936e60158176d710cae4a" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.8.39" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4122cd3169e94605190e77839c9a40d40ed048d305bfdc146e7df40ab0f3e517" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.114", +] + +[[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.114", + "synstructure", +] + +[[package]] +name = "zeroize" +version = "1.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b97154e67e32c85465826e8bcc1c59429aaaf107c1e4a9e53c8d8ccd5eff88d0" + +[[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.114", +] + +[[package]] +name = "zmij" +version = "1.0.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3ff05f8caa9038894637571ae6b9e29466c1f4f829d26c9b28f869a29cbe3445" diff --git a/src-tauri/Cargo.toml b/src-tauri/Cargo.toml new file mode 100644 index 0000000..576718e --- /dev/null +++ b/src-tauri/Cargo.toml @@ -0,0 +1,37 @@ +[package] +name = "app" +version = "0.1.0" +description = "A Tauri App" +authors = ["you"] +license = "" +repository = "" +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 = "app_lib" +crate-type = ["staticlib", "cdylib", "rlib"] + +[build-dependencies] +tauri-build = { version = "2.5.4", features = [] } + +[dependencies] +serde_json = "1.0" +serde = { version = "1.0", features = ["derive"] } +log = "0.4" +tauri = { version = "2.10.0", features = ["tray-icon"] } +tauri-plugin-log = "2" +rusqlite = { version = "0.32", features = ["load_extension", "bundled"] } +reqwest = { version = "0.12", features = ["json"] } +tokio = { version = "1", features = ["full"] } +axum = { version = "0.7" } +anyhow = "1.0" +tower-http = { version = "0.5", features = ["cors"] } +dotenvy = "0.15" +sea-orm = { version = "1.1", features = ["sqlx-sqlite", "runtime-tokio-rustls", "macros"] } +futures = "0.3" + +[dev-dependencies] +mockito = "1.4" diff --git a/src-tauri/build.rs b/src-tauri/build.rs new file mode 100644 index 0000000..d860e1e --- /dev/null +++ b/src-tauri/build.rs @@ -0,0 +1,3 @@ +fn main() { + tauri_build::build() +} diff --git a/src-tauri/build_errors.txt b/src-tauri/build_errors.txt new file mode 100644 index 0000000..d554403 --- /dev/null +++ b/src-tauri/build_errors.txt @@ -0,0 +1,47 @@ + Checking app v0.1.0 (D:\develop\sqlitevector\src-tauri) +error[E0433]: failed to resolve: use of unresolved module or unlinked crate `futures` + --> src\mcp.rs:7:5 + | +7 | use futures::stream::{self, Stream}; + | ^^^^^^^ use of unresolved module or unlinked crate `futures` + | + = help: if you wanted to use a crate named `futures`, use `cargo add futures` to add it to your `Cargo.toml` + +error[E0432]: unresolved import `futures` + --> src\mcp.rs:7:5 + | +7 | use futures::stream::{self, Stream}; + | ^^^^^^^ use of unresolved module or unlinked crate `futures` + | + = help: if you wanted to use a crate named `futures`, use `cargo add futures` to add it to your `Cargo.toml` + +warning: unused import: `std::path::Path` + --> src\db.rs:3:5 + | +3 | use std::path::Path; + | ^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +warning: unused import: `Mutex` + --> src\mcp.rs:10:22 + | +10 | use std::sync::{Arc, Mutex}; + | ^^^^^ + +warning: unused import: `crate::llama::LlamaClient` + --> src\mcp.rs:12:5 + | +12 | use crate::llama::LlamaClient; + | ^^^^^^^^^^^^^^^^^^^^^^^^^ + +warning: unused import: `items::Entity as ItemsEntity` + --> src\mcp.rs:13:30 + | +13 | use crate::entities::{items, items::Entity as ItemsEntity}; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Some errors have detailed explanations: E0432, E0433. +For more information about an error, try `rustc --explain E0432`. +warning: `app` (lib) generated 4 warnings +error: could not compile `app` (lib) due to 2 previous errors; 4 warnings emitted diff --git a/src-tauri/build_errors_2.txt b/src-tauri/build_errors_2.txt new file mode 100644 index 0000000..2e17d34 --- /dev/null +++ b/src-tauri/build_errors_2.txt @@ -0,0 +1,57 @@ + Updating crates.io index + Locking 1 package to latest compatible version + Adding futures v0.3.31 + Downloading crates ... + Downloaded futures v0.3.31 + Compiling futures-macro v0.3.31 + Checking futures-util v0.3.31 + Checking futures-executor v0.3.31 + Checking sqlx-core v0.8.6 + Checking hyper-util v0.1.20 + Checking tower v0.5.3 + Checking axum-core v0.4.5 + Checking futures v0.3.31 + Checking tower-http v0.6.8 + Checking hyper-tls v0.6.0 + Checking reqwest v0.12.28 + Checking axum v0.7.9 + Checking sqlx-sqlite v0.8.6 + Checking sqlx v0.8.6 + Checking sea-query-binder v0.7.0 + Checking sea-orm v1.1.19 + Checking app v0.1.0 (D:\develop\sqlitevector\src-tauri) +warning: unused import: `items::Entity as ItemsEntity` + --> src\mcp.rs:12:30 + | +12 | use crate::entities::{items, items::Entity as ItemsEntity}; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +warning: unused variable: `db_path` + --> src\lib.rs:37:17 + | +37 | let db_path = "vector.db"; + | ^^^^^^^ help: if this is intentional, prefix it with an underscore: `_db_path` + | + = note: `#[warn(unused_variables)]` (part of `#[warn(unused)]`) on by default + +warning: unused variable: `extension_path` + --> src\lib.rs:38:17 + | +38 | let extension_path = "../node_modules/sqlite-vec-windows-x64/vec0.dll"; + | ^^^^^^^^^^^^^^ help: if this is intentional, prefix it with an underscore: `_extension_path` + +warning: field `jsonrpc` is never read + --> src\mcp.rs:17:9 + | +16 | pub struct JsonRpcRequest { + | -------------- field in this struct +17 | pub jsonrpc: String, + | ^^^^^^^ + | + = note: `JsonRpcRequest` has a derived impl for the trait `Debug`, but this is intentionally ignored during dead code analysis + = note: `#[warn(dead_code)]` (part of `#[warn(unused)]`) on by default + +warning: `app` (lib) generated 4 warnings (run `cargo fix --lib -p app` to apply 3 suggestions) + Finished `dev` profile [unoptimized + debuginfo] target(s) in 16.77s diff --git a/src-tauri/build_errors_3.txt b/src-tauri/build_errors_3.txt new file mode 100644 index 0000000..186d825 --- /dev/null +++ b/src-tauri/build_errors_3.txt @@ -0,0 +1,28 @@ + Checking app v0.1.0 (D:\develop\sqlitevector\src-tauri) +warning: unused import: `items::Entity as ItemsEntity` + --> src\mcp.rs:12:30 + | +12 | use crate::entities::{items, items::Entity as ItemsEntity}; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +warning: unused import: `crate::entities::items` + --> src\lib.rs:6:5 + | +6 | use crate::entities::items; + | ^^^^^^^^^^^^^^^^^^^^^^ + +warning: field `jsonrpc` is never read + --> src\mcp.rs:17:9 + | +16 | pub struct JsonRpcRequest { + | -------------- field in this struct +17 | pub jsonrpc: String, + | ^^^^^^^ + | + = note: `JsonRpcRequest` has a derived impl for the trait `Debug`, but this is intentionally ignored during dead code analysis + = note: `#[warn(dead_code)]` (part of `#[warn(unused)]`) on by default + +warning: `app` (lib) generated 3 warnings (run `cargo fix --lib -p app` to apply 2 suggestions) + Finished `dev` profile [unoptimized + debuginfo] target(s) in 1.95s diff --git a/src-tauri/capabilities/default.json b/src-tauri/capabilities/default.json new file mode 100644 index 0000000..c135d7f --- /dev/null +++ b/src-tauri/capabilities/default.json @@ -0,0 +1,11 @@ +{ + "$schema": "../gen/schemas/desktop-schema.json", + "identifier": "default", + "description": "enables the default permissions", + "windows": [ + "main" + ], + "permissions": [ + "core:default" + ] +} diff --git a/src-tauri/icons/128x128.png b/src-tauri/icons/128x128.png new file mode 100644 index 0000000..77e7d23 --- /dev/null +++ b/src-tauri/icons/128x128.png Binary files differ diff --git a/src-tauri/icons/128x128@2x.png b/src-tauri/icons/128x128@2x.png new file mode 100644 index 0000000..0f7976f --- /dev/null +++ b/src-tauri/icons/128x128@2x.png Binary files differ diff --git a/src-tauri/icons/32x32.png b/src-tauri/icons/32x32.png new file mode 100644 index 0000000..98fda06 --- /dev/null +++ b/src-tauri/icons/32x32.png Binary files differ diff --git a/src-tauri/icons/Square107x107Logo.png b/src-tauri/icons/Square107x107Logo.png new file mode 100644 index 0000000..f35d84f --- /dev/null +++ b/src-tauri/icons/Square107x107Logo.png Binary files differ diff --git a/src-tauri/icons/Square142x142Logo.png b/src-tauri/icons/Square142x142Logo.png new file mode 100644 index 0000000..1823bb2 --- /dev/null +++ b/src-tauri/icons/Square142x142Logo.png Binary files differ diff --git a/src-tauri/icons/Square150x150Logo.png b/src-tauri/icons/Square150x150Logo.png new file mode 100644 index 0000000..dc2b22c --- /dev/null +++ b/src-tauri/icons/Square150x150Logo.png Binary files differ diff --git a/src-tauri/icons/Square284x284Logo.png b/src-tauri/icons/Square284x284Logo.png new file mode 100644 index 0000000..0ed3984 --- /dev/null +++ b/src-tauri/icons/Square284x284Logo.png Binary files differ diff --git a/src-tauri/icons/Square30x30Logo.png b/src-tauri/icons/Square30x30Logo.png new file mode 100644 index 0000000..60bf0ea --- /dev/null +++ b/src-tauri/icons/Square30x30Logo.png Binary files differ diff --git a/src-tauri/icons/Square310x310Logo.png b/src-tauri/icons/Square310x310Logo.png new file mode 100644 index 0000000..c8ca0ad --- /dev/null +++ b/src-tauri/icons/Square310x310Logo.png Binary files differ diff --git a/src-tauri/icons/Square44x44Logo.png b/src-tauri/icons/Square44x44Logo.png new file mode 100644 index 0000000..8756459 --- /dev/null +++ b/src-tauri/icons/Square44x44Logo.png Binary files differ diff --git a/src-tauri/icons/Square71x71Logo.png b/src-tauri/icons/Square71x71Logo.png new file mode 100644 index 0000000..2c8023c --- /dev/null +++ b/src-tauri/icons/Square71x71Logo.png Binary files differ diff --git a/src-tauri/icons/Square89x89Logo.png b/src-tauri/icons/Square89x89Logo.png new file mode 100644 index 0000000..2c5e603 --- /dev/null +++ b/src-tauri/icons/Square89x89Logo.png Binary files differ diff --git a/src-tauri/icons/StoreLogo.png b/src-tauri/icons/StoreLogo.png new file mode 100644 index 0000000..17d142c --- /dev/null +++ b/src-tauri/icons/StoreLogo.png Binary files differ diff --git a/src-tauri/icons/icon.icns b/src-tauri/icons/icon.icns new file mode 100644 index 0000000..a2993ad --- /dev/null +++ b/src-tauri/icons/icon.icns Binary files differ diff --git a/src-tauri/icons/icon.ico b/src-tauri/icons/icon.ico new file mode 100644 index 0000000..06c23c8 --- /dev/null +++ b/src-tauri/icons/icon.ico Binary files differ diff --git a/src-tauri/icons/icon.png b/src-tauri/icons/icon.png new file mode 100644 index 0000000..d1756ce --- /dev/null +++ b/src-tauri/icons/icon.png Binary files differ diff --git a/src-tauri/integration_test_errors.txt b/src-tauri/integration_test_errors.txt new file mode 100644 index 0000000..1ca1f3b --- /dev/null +++ b/src-tauri/integration_test_errors.txt @@ -0,0 +1,126 @@ +warning: unused import: `items::Entity as ItemsEntity` + --> src\mcp.rs:12:30 + | +12 | ...s, items::Entity as ItemsEntity}; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +warning: unused import: `crate::entities::items` + --> src\lib.rs:6:5 + | +6 | use crate::entities::items; + | ^^^^^^^^^^^^^^^^^^^^^^ + +warning: field `jsonrpc` is never read + --> src\mcp.rs:17:9 + | +16 | pub struct JsonRpcRequest { + | -------------- field in this struct +17 | pub jsonrpc: String, + | ^^^^^^^ + | + = note: `JsonRpcRequest` has a derived impl for the trait `Debug`, but this is intentionally ignored during dead code analysis + = note: `#[warn(dead_code)]` (part of `#[warn(unused)]`) on by default + +warning: `app` (lib) generated 3 warnings (run `cargo fix --lib -p app` to apply 2 suggestions) + Compiling app v0.1.0 (D:\develop\sqlitevector\src-tauri) +error[E0603]: module `mcp` is private + --> tests\integration_test.rs:1:25 + | +1 | use app_lib::{AppState, mcp, db, llama::LlamaClient}; + | ^^^ private module + | +note: the module `mcp` is defined here + --> D:\develop\sqlitevector\src-tauri\src\lib.rs:3:1 + | +3 | mod mcp; + | ^^^^^^^ + +error[E0603]: module `db` is private + --> tests\integration_test.rs:1:30 + | +1 | use app_lib::{AppState, mcp, db, llama::LlamaClient}; + | ^^ private module + | +note: the module `db` is defined here + --> D:\develop\sqlitevector\src-tauri\src\lib.rs:1:1 + | +1 | mod db; + | ^^^^^^ + +error[E0603]: module `llama` is private + --> tests\integration_test.rs:1:34 + | +1 | use app_lib::{AppState, mcp, db, llama::LlamaClient}; + | ^^^^^ private module + | +note: the module `llama` is defined here + --> D:\develop\sqlitevector\src-tauri\src\lib.rs:2:1 + | +2 | mod llama; + | ^^^^^^^^^ + +warning: unused import: `sea_orm::DatabaseConnection` + --> tests\integration_test.rs:3:5 + | +3 | use sea_orm::DatabaseConnection; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +error[E0599]: no method named `get_database_backend` found for enum `DatabaseConnection` in the current scope + --> tests\integration_test.rs:22:27 + | +22 | let db_backend = conn.get_database_backend(); + | ^^^^^^^^^^^^^^^^^^^^ method not found in `DatabaseConnection` + | + ::: C:\Users\bythe\.cargo\registry\src\index.crates.io-1949cf8c6b5b557f\sea-orm-1.1.19\src\database\connection.rs:13:8 + | +13 | fn get_database_backend(&self) -> DbBackend; + | -------------------- the method is available for `DatabaseConnection` here + | + = help: items from traits can only be used if the trait is in scope +help: trait `ConnectionTrait` which provides `get_database_backend` is implemented but not in scope; perhaps you want to import it + | + 1 + use sea_orm::ConnectionTrait; + | + +error[E0599]: no method named `execute` found for enum `DatabaseConnection` in the current scope + --> tests\integration_test.rs:23:10 + | + 23 | conn.execute(db_backend.build(sea_orm::sea_query::Table::create() + | -----^^^^^^^ method not found in `DatabaseConnection` + | + ::: C:\Users\bythe\.cargo\registry\src\index.crates.io-1949cf8c6b5b557f\sea-orm-1.1.19\src\database\connection.rs:16:14 + | + 16 | async fn execute(&self, stmt: Statement) -> Result; + | ------- the method is available for `DatabaseConnection` here + | +note: the method `execute` exists on the type `sea_orm::SqlxSqlitePoolConnection` + --> C:\Users\bythe\.cargo\registry\src\index.crates.io-1949cf8c6b5b557f\sea-orm-1.1.19\src\driver\sqlx_sqlite.rs:131:5 + | +131 | pub async fn execute(&self, stmt: Statement) -> Result { + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + = help: items from traits can only be used if the trait is in scope +help: trait `ConnectionTrait` which provides `execute` is implemented but not in scope; perhaps you want to import it + | + 1 + use sea_orm::ConnectionTrait; + | + +error[E0282]: type annotations needed + --> tests\integration_test.rs:23:5 + | +23 | / conn.execute(db_backend.build(sea_orm::sea_query::Table::create() +24 | | .table(app_lib::entities::items::Entity) +25 | | .if_not_exists() +26 | | .col(sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::Id).integer().not_null().auto_increment().primary_key()) +... | +30 | | .col(sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::UpdatedAt).string()) +31 | | )).await.expect("Failed to create table"); + | |____________^ cannot infer type + +Some errors have detailed explanations: E0282, E0599, E0603. +For more information about an error, try `rustc --explain E0282`. +warning: `app` (test "integration_test") generated 1 warning +error: could not compile `app` (test "integration_test") due to 6 previous errors; 1 warning emitted diff --git a/src-tauri/src/db.rs b/src-tauri/src/db.rs new file mode 100644 index 0000000..f5790b4 --- /dev/null +++ b/src-tauri/src/db.rs @@ -0,0 +1,67 @@ +use sea_orm::{ConnectOptions, Database, DatabaseConnection}; +use std::time::Duration; + +pub async fn init_db(db_path: &str, extension_path: &str) -> anyhow::Result { + // SeaORM (sqlx) は URL 形式でパスを指定する + let db_url = format!("sqlite:{}?mode=rwc", db_path); + + let mut opt = ConnectOptions::new(db_url); + opt.max_connections(10) + .min_connections(5) + .connect_timeout(Duration::from_secs(8)) + .idle_timeout(Duration::from_secs(8)) + .max_lifetime(Duration::from_secs(8)) + .sqlx_logging(true); + + // 拡張機能のロードが必要なため、sqlx の初期化フックがあればいいが、 + // SeaORM/sqlx で動的にロードするのは少し工夫が必要。 + // ここではまず rusqlite か何かでスキーマ初期化と拡張ロードの確認をしてから SeaORM で繋ぐか、 + // sqlx の ConnectOptions にて初期化SQLを流す。 + + // NOTE: sqlite-vec のロードは非常に特殊なため、 + // 生の rusqlite で拡張をロードしてテーブルを作った後、SeaORM で接続する。 + { + use rusqlite::Connection; + let conn = Connection::open(db_path)?; + unsafe { + conn.load_extension(extension_path, None)?; + } + conn.execute_batch( + "PRAGMA journal_mode = WAL; + CREATE TABLE IF NOT EXISTS items ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + content TEXT NOT NULL, + path TEXT, + created_at TEXT DEFAULT (datetime('now')), + updated_at TEXT DEFAULT (datetime('now')) + ); + CREATE VIRTUAL TABLE IF NOT EXISTS vec_items USING vec0( + id INTEGER PRIMARY KEY, + embedding FLOAT[3] + );", + )?; + } + + let db = Database::connect(opt).await?; + + // SeaORM の接続後にも拡張をロードしておく必要がある場合がある(セッション毎) + // sqlx-sqlite の場合、初期化 SQL でロードできるケースがあるが設定が複雑。 + // 今回は rusqlite での初期化を優先し、クエリ実行時に MATCH 句が動くか確認する。 + + Ok(db) +} + +#[cfg(test)] +mod tests { + use super::*; + + #[tokio::test] + async fn test_init_db_basic() { + // NOTE: We test with a dummy extension path and expect it to fail if it's not found, + // or we test just the SeaORM part if we could separation. + // For now, let's just ensure we can connect to an in-memory sqlite via SeaORM. + let mut opt = ConnectOptions::new("sqlite::memory:"); + let db = Database::connect(opt).await; + assert!(db.is_ok()); + } +} diff --git a/src-tauri/src/entities/items.rs b/src-tauri/src/entities/items.rs new file mode 100644 index 0000000..6cdc409 --- /dev/null +++ b/src-tauri/src/entities/items.rs @@ -0,0 +1,18 @@ +use sea_orm::entity::prelude::*; +use serde::{Deserialize, Serialize}; + +#[derive(Clone, Debug, PartialEq, DeriveEntityModel, Serialize, Deserialize)] +#[sea_orm(table_name = "items")] +pub struct Model { + #[sea_orm(primary_key)] + pub id: i32, + pub content: String, + pub path: Option, + pub created_at: String, + pub updated_at: String, +} + +#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)] +pub enum Relation {} + +impl ActiveModelBehavior for ActiveModel {} diff --git a/src-tauri/src/entities/mod.rs b/src-tauri/src/entities/mod.rs new file mode 100644 index 0000000..339b22d --- /dev/null +++ b/src-tauri/src/entities/mod.rs @@ -0,0 +1 @@ +pub mod items; diff --git a/src-tauri/src/lib.rs b/src-tauri/src/lib.rs new file mode 100644 index 0000000..5d86108 --- /dev/null +++ b/src-tauri/src/lib.rs @@ -0,0 +1,115 @@ +pub mod db; +pub mod entities; +pub mod llama; +pub mod mcp; + +use crate::llama::LlamaClient; +use dotenvy::dotenv; +use sea_orm::DatabaseConnection; +use std::env; +use std::sync::Arc; +use tauri::menu::{Menu, MenuItem}; +use tauri::tray::{TrayIconBuilder, TrayIconEvent}; +use tauri::Manager; + +pub struct AppState { + pub db: DatabaseConnection, + pub llama: Arc, +} + +#[tauri::command] +fn greet(name: &str) -> String { + format!("Hello, {}! You've been greeted from Rust!", name) +} + +#[cfg_attr(mobile, tauri::mobile_entry_point)] +pub fn run() { + tauri::Builder::default() + .invoke_handler(tauri::generate_handler![greet]) + .on_window_event(|window, event| { + if let tauri::WindowEvent::CloseRequested { api, .. } = event { + api.prevent_close(); + let _ = window.hide(); + } + }) + .setup(|app| { + dotenv().ok(); + + let app_handle = app.handle().clone(); + tauri::async_runtime::block_on(async move { + let db_path = "vector.db"; + let extension_path = "../node_modules/sqlite-vec-windows-x64/vec0.dll"; + + let conn = db::init_db(db_path, extension_path) + .await + .expect("Failed to init database"); + let llama_base_url = env::var("LLAMA_CPP_BASE_URL") + .unwrap_or_else(|_| "http://localhost:8080".to_string()); + let embedding_model = env::var("LLAMA_CPP_EMBEDDING_MODEL") + .unwrap_or_else(|_| "nomic-embed-text".to_string()); + let completion_model = + env::var("LLAMA_CPP_MODEL").unwrap_or_else(|_| "mistral".to_string()); + + let llama = LlamaClient::new(llama_base_url, embedding_model, completion_model); + + let state = Arc::new(AppState { + db: conn, + llama: Arc::new(llama), + }); + + app_handle.manage(state.clone()); + + // Start MCP Server (SSE) + let mcp_port = env::var("MCP_PORT") + .unwrap_or_else(|_| "3000".to_string()) + .parse::() + .unwrap_or(3000); + + tokio::spawn(async move { + mcp::start_mcp_server(state, mcp_port).await; + }); + }); + + if cfg!(debug_assertions) { + app.handle().plugin( + tauri_plugin_log::Builder::default() + .level(log::LevelFilter::Info) + .build(), + )?; + } + + let quit_i = MenuItem::with_id(app, "quit", "Quit", true, None::<&str>)?; + let show_i = MenuItem::with_id(app, "show", "Show", true, None::<&str>)?; + let menu = Menu::with_items(app, &[&show_i, &quit_i])?; + + let _tray = TrayIconBuilder::new() + .icon(app.default_window_icon().unwrap().clone()) + .menu(&menu) + .on_menu_event(|app, event| match event.id.as_ref() { + "quit" => { + app.exit(0); + } + "show" => { + if let Some(window) = app.get_webview_window("main") { + let _ = window.show(); + let _ = window.set_focus(); + } + } + _ => {} + }) + .on_tray_icon_event(|tray, event| { + if let TrayIconEvent::Click { .. } = event { + let app = tray.app_handle(); + if let Some(window) = app.get_webview_window("main") { + let _ = window.show(); + let _ = window.set_focus(); + } + } + }) + .build(app)?; + + Ok(()) + }) + .run(tauri::generate_context!()) + .expect("error while running tauri application"); +} diff --git a/src-tauri/src/llama.rs b/src-tauri/src/llama.rs new file mode 100644 index 0000000..e5b82f2 --- /dev/null +++ b/src-tauri/src/llama.rs @@ -0,0 +1,145 @@ +use anyhow::Result; +use reqwest::Client; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Serialize)] +struct EmbeddingRequest { + model: String, + input: String, +} + +#[derive(Debug, Deserialize)] +struct EmbeddingResponse { + data: Vec, +} + +#[derive(Debug, Deserialize)] +struct EmbeddingData { + embedding: Vec, +} + +#[derive(Debug, Serialize)] +struct CompletionRequest { + model: String, + prompt: String, + n_predict: i32, + temperature: f32, + stream: bool, +} + +#[derive(Debug, Deserialize)] +struct CompletionResponse { + content: String, +} + +pub struct LlamaClient { + client: Client, + base_url: String, + embedding_model: String, + completion_model: String, +} + +impl LlamaClient { + pub fn new(base_url: String, embedding_model: String, completion_model: String) -> Self { + Self { + client: Client::new(), + base_url, + embedding_model, + completion_model, + } + } + + pub async fn get_embedding(&self, text: &str) -> Result> { + let url = format!("{}/embeddings", self.base_url); + let req = EmbeddingRequest { + model: self.embedding_model.clone(), + input: text.to_string(), + }; + + let res = self + .client + .post(&url) + .json(&req) + .send() + .await? + .json::() + .await?; + + Ok(res.data[0].embedding.clone()) + } + + pub async fn completion( + &self, + prompt: &str, + n_predict: i32, + temperature: f32, + ) -> Result { + let url = format!("{}/completion", self.base_url); + let req = CompletionRequest { + model: self.completion_model.clone(), + prompt: prompt.to_string(), + n_predict, + temperature, + stream: false, + }; + + let res = self + .client + .post(&url) + .json(&req) + .send() + .await? + .json::() + .await?; + + Ok(res.content) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use mockito::Server; + + #[tokio::test] + async fn test_get_embedding() -> Result<()> { + let mut server = Server::new_async().await; + let url = server.url(); + + let mock = server + .mock("POST", "/embeddings") + .with_status(200) + .with_header("content-type", "application/json") + .with_body(r#"{"data":[{"embedding":[0.1, 0.2, 0.3]}]}"#) + .create_async() + .await; + + let client = LlamaClient::new(url, "test-model".to_string(), "test-model".to_string()); + let embedding = client.get_embedding("hello").await?; + + assert_eq!(embedding, vec![0.1, 0.2, 0.3]); + mock.assert_async().await; + Ok(()) + } + + #[tokio::test] + async fn test_completion() -> Result<()> { + let mut server = Server::new_async().await; + let url = server.url(); + + let mock = server + .mock("POST", "/completion") + .with_status(200) + .with_header("content-type", "application/json") + .with_body(r#"{"content":"Hello, world!"}"#) + .create_async() + .await; + + let client = LlamaClient::new(url, "test-model".to_string(), "test-model".to_string()); + let result = client.completion("hi", 10, 0.7).await?; + + assert_eq!(result, "Hello, world!"); + mock.assert_async().await; + Ok(()) + } +} diff --git a/src-tauri/src/main.rs b/src-tauri/src/main.rs new file mode 100644 index 0000000..69c3a72 --- /dev/null +++ b/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() { + app_lib::run(); +} diff --git a/src-tauri/src/mcp.rs b/src-tauri/src/mcp.rs new file mode 100644 index 0000000..97f6cbb --- /dev/null +++ b/src-tauri/src/mcp.rs @@ -0,0 +1,358 @@ +use crate::entities::items; +use crate::AppState; +use axum::{ + extract::State, + response::sse::{Event, Sse}, + routing::{get, post}, + Json, Router, +}; +use futures::stream::{self, Stream}; +use sea_orm::*; +use serde::{Deserialize, Serialize}; +use std::convert::Infallible; +use std::sync::Arc; + +#[derive(Debug, Deserialize)] +pub struct JsonRpcRequest { + pub jsonrpc: String, + pub method: String, + pub params: serde_json::Value, + pub id: Option, +} + +#[derive(Debug, Serialize)] +pub struct JsonRpcResponse { + pub jsonrpc: String, + pub result: Option, + pub error: Option, + pub id: serde_json::Value, +} + +pub async fn start_mcp_server(state: Arc, port: u16) { + let app = Router::new() + .route("/sse", get(sse_handler)) + .route("/messages", post(message_handler)) + .with_state(state); + + let listener = tokio::net::TcpListener::bind(format!("0.0.0.0:{}", port)) + .await + .unwrap(); + axum::serve(listener, app).await.unwrap(); +} + +async fn sse_handler( + State(_state): State>, +) -> Sse>> { + // Basic SSE handler for MCP + // Real MCP SDK would send 'endpoint' event here + let stream = stream::once(async { Ok(Event::default().event("endpoint").data("/messages")) }); + + Sse::new(stream) +} + +pub async fn message_handler( + State(state): State>, + Json(payload): Json, +) -> Json { + let result = match payload.method.as_str() { + "tools/list" => Some(serde_json::json!({ + "tools": [ + { + "name": "add_item_text", + "description": "Add item from text", + "inputSchema": { + "type": "object", + "properties": { + "content": { "type": "string" }, + "path": { "type": "string" } + } + } + }, + { + "name": "search_text", + "description": "Search items by text", + "inputSchema": { + "type": "object", + "properties": { + "content": { "type": "string" }, + "limit": { "type": "number" } + } + } + }, + { + "name": "add_item", + "description": "Add item with vector", + "inputSchema": { + "type": "object", + "properties": { + "content": { "type": "string" }, + "vector": { "type": "array", "items": { "type": "number" } }, + "path": { "type": "string" } + } + } + }, + { + "name": "search_vector", + "description": "Search items by vector", + "inputSchema": { + "type": "object", + "properties": { + "vector": { "type": "array", "items": { "type": "number" } }, + "limit": { "type": "number" } + } + } + }, + { + "name": "llm_generate", + "description": "Generate text via LLM", + "inputSchema": { + "type": "object", + "properties": { + "prompt": { "type": "string" }, + "n_predict": { "type": "number" }, + "temperature": { "type": "number" } + } + } + } + ] + })), + "tools/call" => { + let tool_name = payload.params["name"].as_str().unwrap_or(""); + let args = &payload.params["arguments"]; + + match tool_name { + "add_item_text" => { + let content = args["content"].as_str().unwrap_or(""); + let path = args["path"].as_str().unwrap_or(""); + match handle_add_item_text(&state, content, path).await { + Ok(res) => Some(res), + Err(e) => { + return Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result: None, + error: Some( + serde_json::json!({ "code": -32000, "message": e.to_string() }), + ), + id: payload.id.unwrap_or(serde_json::Value::Null), + }) + } + } + } + "add_item" => { + let content = args["content"].as_str().unwrap_or(""); + let vector: Vec = args["vector"] + .as_array() + .unwrap_or(&vec![]) + .iter() + .map(|v| v.as_f64().unwrap_or(0.0) as f32) + .collect(); + let path = args["path"].as_str().unwrap_or(""); + match handle_add_item(&state, content, vector, path).await { + Ok(res) => Some(res), + Err(e) => { + return Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result: None, + error: Some( + serde_json::json!({ "code": -32000, "message": e.to_string() }), + ), + id: payload.id.unwrap_or(serde_json::Value::Null), + }) + } + } + } + "search_text" => { + let content = args["content"].as_str().unwrap_or(""); + let limit = args["limit"].as_u64().unwrap_or(10) as usize; + match handle_search_text(&state, content, limit).await { + Ok(res) => Some(res), + Err(e) => { + return Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result: None, + error: Some( + serde_json::json!({ "code": -32000, "message": e.to_string() }), + ), + id: payload.id.unwrap_or(serde_json::Value::Null), + }) + } + } + } + "search_vector" => { + let vector: Vec = args["vector"] + .as_array() + .unwrap_or(&vec![]) + .iter() + .map(|v| v.as_f64().unwrap_or(0.0) as f32) + .collect(); + let limit = args["limit"].as_u64().unwrap_or(10) as usize; + match handle_search_vector(&state, vector, limit).await { + Ok(res) => Some(res), + Err(e) => { + return Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result: None, + error: Some( + serde_json::json!({ "code": -32000, "message": e.to_string() }), + ), + id: payload.id.unwrap_or(serde_json::Value::Null), + }) + } + } + } + "llm_generate" => { + let prompt = args["prompt"].as_str().unwrap_or(""); + let n_predict = args["n_predict"].as_i64().unwrap_or(128) as i32; + let temperature = args["temperature"].as_f64().unwrap_or(0.7) as f32; + match handle_llm_generate(&state, prompt, n_predict, temperature).await { + Ok(res) => Some(res), + Err(e) => { + return Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result: None, + error: Some( + serde_json::json!({ "code": -32000, "message": e.to_string() }), + ), + id: payload.id.unwrap_or(serde_json::Value::Null), + }) + } + } + } + _ => Some(serde_json::json!({ "error": "Unknown tool" })), + } + } + _ => Some(serde_json::json!({ "error": "Method not found" })), + }; + + Json(JsonRpcResponse { + jsonrpc: "2.0".to_string(), + result, + error: None, + id: payload.id.unwrap_or(serde_json::Value::Null), + }) +} + +async fn handle_add_item_text( + state: &AppState, + content: &str, + path: &str, +) -> anyhow::Result { + let embedding = state.llama.get_embedding(content).await?; + + // SeaORM insert + let new_item = items::ActiveModel { + content: Set(content.to_owned()), + path: Set(Some(path.to_owned())), + ..Default::default() + }; + + let db = &state.db; + let res = new_item.insert(db).await?; + let id = res.id; + + // vec0 table insert (SeaORM raw SQL for now as it's a virtual table) + let embedding_bytes: Vec = embedding.iter().flat_map(|f| f.to_le_bytes()).collect(); + + db.execute(Statement::from_sql_and_values( + DatabaseBackend::Sqlite, + "INSERT INTO vec_items (id, embedding) VALUES (?, ?)", + [id.into(), embedding_bytes.into()], + )) + .await?; + + Ok(serde_json::json!({ + "content": [{ "type": "text", "text": format!("Added item with id {}", id) }] + })) +} + +async fn handle_add_item( + state: &AppState, + content: &str, + vector: Vec, + path: &str, +) -> anyhow::Result { + let db = &state.db; + + let new_item = items::ActiveModel { + content: Set(content.to_owned()), + path: Set(Some(path.to_owned())), + ..Default::default() + }; + let res = new_item.insert(db).await?; + let id = res.id; + + let embedding_bytes: Vec = vector.iter().flat_map(|f| f.to_le_bytes()).collect(); + + db.execute(Statement::from_sql_and_values( + DatabaseBackend::Sqlite, + "INSERT INTO vec_items (id, embedding) VALUES (?, ?)", + [id.into(), embedding_bytes.into()], + )) + .await?; + + Ok(serde_json::json!({ + "content": [{ "type": "text", "text": format!("Added item with id {}", id) }] + })) +} + +async fn handle_search_text( + state: &AppState, + content: &str, + limit: usize, +) -> anyhow::Result { + let embedding = state.llama.get_embedding(content).await?; + handle_search_vector(state, embedding, limit).await +} + +async fn handle_search_vector( + state: &AppState, + vector: Vec, + limit: usize, +) -> anyhow::Result { + let embedding_bytes: Vec = vector.iter().flat_map(|f| f.to_le_bytes()).collect(); + + let db = &state.db; + + // raw SQL query via SeaORM for vector search + let results = db + .query_all(Statement::from_sql_and_values( + DatabaseBackend::Sqlite, + "SELECT i.id, i.content, i.path, v.distance + FROM vec_items v + JOIN items i ON v.id = i.id + WHERE embedding MATCH ? + ORDER BY distance + LIMIT ?", + [embedding_bytes.into(), (limit as i64).into()], + )) + .await?; + + let mut out = Vec::new(); + for res in results { + out.push(serde_json::json!({ + "id": res.try_get::("", "id").map_err(|e| anyhow::anyhow!(e))?, + "content": res.try_get::("", "content").map_err(|e| anyhow::anyhow!(e))?, + "path": res.try_get::>("", "path").map_err(|e| anyhow::anyhow!(e))?, + "distance": res.try_get::("", "distance").map_err(|e| anyhow::anyhow!(e))? + })); + } + + Ok(serde_json::json!({ + "content": [{ "type": "text", "text": serde_json::to_string_pretty(&out)? }] + })) +} + +async fn handle_llm_generate( + state: &AppState, + prompt: &str, + n_predict: i32, + temperature: f32, +) -> anyhow::Result { + let text = state + .llama + .completion(prompt, n_predict, temperature) + .await?; + Ok(serde_json::json!({ + "content": [{ "type": "text", "text": text }] + })) +} diff --git a/src-tauri/tauri.conf.json b/src-tauri/tauri.conf.json new file mode 100644 index 0000000..fad6511 --- /dev/null +++ b/src-tauri/tauri.conf.json @@ -0,0 +1,36 @@ +{ + "$schema": "https://schema.tauri.app/config/2", + "productName": "sqlitevector", + "version": "0.1.0", + "identifier": "com.sqlitevector.app", + "build": { + "frontendDist": "../src/frontend", + "beforeDevCommand": "", + "beforeBuildCommand": "" + }, + "app": { + "windows": [ + { + "title": "SQLite Vector", + "width": 800, + "height": 600, + "resizable": true, + "fullscreen": false + } + ], + "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" + ] + } +} \ No newline at end of file diff --git a/src-tauri/test_errors.txt b/src-tauri/test_errors.txt new file mode 100644 index 0000000..7bd47a6 --- /dev/null +++ b/src-tauri/test_errors.txt @@ -0,0 +1,153 @@ + Compiling app v0.1.0 (D:\develop\sqlitevector\src-tauri) +error[E0432]: unresolved import `crate::AppState` + --> src\mcp.rs:11:5 + | +11 | use crate::AppState; + | ^^^^^^^^^^^^^^^ no `AppState` in the root + +error[E0433]: failed to resolve: use of unresolved module or unlinked crate `env` + --> src\lib.rs:37:38 + | +37 | let llama_base_url = env::var("LLAMA_CPP_BASE_URL").unwrap_or_else(|_| "http://localhost:8080".to_string()); + | ^^^ use of unresolved module or unlinked crate `env` + | + = help: if you wanted to use a crate named `env`, use `cargo add env` to add it to your `Cargo.toml` +help: consider importing this module + | + 6 + use std::env; + | + +error[E0433]: failed to resolve: use of unresolved module or unlinked crate `env` + --> src\lib.rs:38:39 + | +38 | let embedding_model = env::var("LLAMA_CPP_EMBEDDING_MODEL").unwrap_or_else(|_| "nomic-embed-text".to_string()); + | ^^^ use of unresolved module or unlinked crate `env` + | + = help: if you wanted to use a crate named `env`, use `cargo add env` to add it to your `Cargo.toml` +help: consider importing this module + | + 6 + use std::env; + | + +error[E0433]: failed to resolve: use of unresolved module or unlinked crate `env` + --> src\lib.rs:39:40 + | +39 | let completion_model = env::var("LLAMA_CPP_MODEL").unwrap_or_else(|_| "mistral".to_string()); + | ^^^ use of unresolved module or unlinked crate `env` + | + = help: if you wanted to use a crate named `env`, use `cargo add env` to add it to your `Cargo.toml` +help: consider importing this module + | + 6 + use std::env; + | + +error[E0422]: cannot find struct, variant or union type `AppState` in this scope + --> src\lib.rs:43:38 + | +43 | let state = Arc::new(AppState { + | ^^^^^^^^ not found in this scope + +error[E0433]: failed to resolve: use of unresolved module or unlinked crate `env` + --> src\lib.rs:51:32 + | +51 | let mcp_port = env::var("MCP_PORT") + | ^^^ use of unresolved module or unlinked crate `env` + | + = help: if you wanted to use a crate named `env`, use `cargo add env` to add it to your `Cargo.toml` +help: consider importing this module + | + 6 + use std::env; + | + +warning: unused import: `items::Entity as ItemsEntity` + --> src\mcp.rs:12:30 + | +12 | use crate::entities::{items, items::Entity as ItemsEntity}; + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = note: `#[warn(unused_imports)]` (part of `#[warn(unused)]`) on by default + +warning: unused import: `crate::entities::items` + --> src\lib.rs:6:5 + | +6 | use crate::entities::items; + | ^^^^^^^^^^^^^^^^^^^^^^ + +error[E0282]: type annotations needed + --> src\mcp.rs:38:5 + | +38 | axum::serve(listener, app).await.unwrap(); + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ cannot infer type + +error[E0282]: type annotations needed + --> src\mcp.rs:206:9 + | +206 | let embedding = state.llama.get_embedding(content).await?; + | ^^^^^^^^^ +... +220 | let embedding_bytes: Vec = embedding.iter() + | --------- type must be known at this point + | +help: consider giving `embedding` an explicit type + | +206 | let embedding: /* Type */ = state.llama.get_embedding(content).await?; + | ++++++++++++ + +error[E0282]: type annotations needed + --> src\mcp.rs:221:20 + | +221 | .flat_map(|f| f.to_le_bytes()) + | ^ - type must be known at this point + | +help: consider giving this closure parameter an explicit type + | +221 | .flat_map(|f: /* Type */| f.to_le_bytes()) + | ++++++++++++ + +error[E0282]: type annotations needed + --> src\mcp.rs:288:19 + | +288 | "id": res.try_get::("", "id")?, + | ^^^ cannot infer type + +error: this function depends on never type fallback being `()` + --> src\mcp.rs:300:1 + | +300 | async fn handle_llm_generate(state: &AppState, prompt: &str, n_predict: i32, temperature: f32) -> anyhow::Result { + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + | + = warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in Rust 2024 and in a future release in all editions! + = note: for more information, see + = help: specify the types explicitly +note: in edition 2024, the requirement `!: Serialize` will fail + --> src\mcp.rs:302:8 + | +302 | Ok(serde_json::json!({ + | ________^ +303 | | "content": [{ "type": "text", "text": text }] +304 | | })) + | |______^ + = note: `#[deny(dependency_on_unit_never_type_fallback)]` (part of `#[deny(rust_2024_compatibility)]`) on by default + = note: this error originates in the macro `$crate::json_internal` which comes from the expansion of the macro `serde_json::json` (in Nightly builds, run with -Z macro-backtrace for more info) +help: use `()` annotations to avoid fallback changes + | +301 | let text: () = state.llama.completion(prompt, n_predict, temperature).await?; + | ++++ + +warning: variable does not need to be mutable + --> src\db.rs:63:13 + | +63 | let mut opt = ConnectOptions::new("sqlite::memory:"); + | ----^^^ + | | + | help: remove this `mut` + | + = note: `#[warn(unused_mut)]` (part of `#[warn(unused)]`) on by default + +Some errors have detailed explanations: E0282, E0422, E0432, E0433. +For more information about an error, try `rustc --explain E0282`. +warning: `app` (lib) generated 2 warnings (2 duplicates) +error: could not compile `app` (lib) due to 12 previous errors; 2 warnings emitted +warning: build failed, waiting for other jobs to finish... +warning: `app` (lib test) generated 3 warnings +error: could not compile `app` (lib test) due to 12 previous errors; 3 warnings emitted diff --git a/src-tauri/tests/integration_test.rs b/src-tauri/tests/integration_test.rs new file mode 100644 index 0000000..1e93d26 --- /dev/null +++ b/src-tauri/tests/integration_test.rs @@ -0,0 +1,86 @@ +use app_lib::{db, llama::LlamaClient, mcp, AppState}; +use axum::extract::State; +use axum::Json; +use sea_orm::{ConnectionTrait, DatabaseBackend, DatabaseConnection, Statement}; +use serde_json::json; +use std::sync::Arc; + +#[tokio::test] +async fn test_mcp_integration_flow() { + // 1. Initialize DB (in-memory) + let conn = sea_orm::Database::connect("sqlite::memory:").await.unwrap(); + + // Create items table manually for the test + let db_backend = conn.get_database_backend(); + conn.execute( + db_backend.build( + sea_orm::sea_query::Table::create() + .table(app_lib::entities::items::Entity) + .if_not_exists() + .col( + sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::Id) + .integer() + .not_null() + .auto_increment() + .primary_key(), + ) + .col( + sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::Content) + .string() + .not_null(), + ) + .col( + sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::Path) + .string(), + ) + .col( + sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::CreatedAt) + .string(), + ) + .col( + sea_orm::sea_query::ColumnDef::new(app_lib::entities::items::Column::UpdatedAt) + .string(), + ), + ), + ) + .await + .expect("Failed to create table"); + + // Create vec_items table (virtual) - might fail if vec0 extension is missing, + // so we handle it gracefully in the test or skip it. + // In this test, we just test the tool list and basic error handling which don't need vec0. + + let llama = LlamaClient::new( + "http://localhost:8080".to_string(), + "m".to_string(), + "m".to_string(), + ); + + let state = Arc::new(AppState { + db: conn, + llama: Arc::new(llama), + }); + + // 2. Test tools/list + let req = mcp::JsonRpcRequest { + jsonrpc: "2.0".to_string(), + method: "tools/list".to_string(), + params: json!({}), + id: Some(json!(1)), + }; + + let res = mcp::message_handler(State(state.clone()), Json(req)).await; + assert!(res.result.is_some()); + let tools = res.result.as_ref().unwrap()["tools"].as_array().unwrap(); + assert!(tools.len() >= 4); + + // 3. Test Unknown Tool + let req_unknown = mcp::JsonRpcRequest { + jsonrpc: "2.0".to_string(), + method: "tools/call".to_string(), + params: json!({"name": "unknown"}), + id: Some(json!(2)), + }; + let res_err = mcp::message_handler(State(state.clone()), Json(req_unknown)).await; + assert!(res_err.result.as_ref().unwrap()["error"] == "Unknown tool"); +} diff --git a/src/backend/config.js b/src/backend/config.js new file mode 100644 index 0000000..a7a0010 --- /dev/null +++ b/src/backend/config.js @@ -0,0 +1,53 @@ +/** + * アプリケーション設定・定数 + */ + +export const CONFIG = { + // Electron ウィンドウ + window: { + width: 600, + height: 400, + }, + + // MCP サーバー + mcp: { + name: "sqlite-vec-server", + version: "1.0.0", + defaultPort: process.env.MCP_PORT ? parseInt(process.env.MCP_PORT, 10) : 3000, + ssePath: "/sse", + messagesPath: "/messages", + }, + + // llama.cpp + llama: { + baseUrl: process.env.LLAMA_CPP_BASE_URL || "http://127.0.0.1:8080", + embeddingModel: process.env.LLAMA_CPP_EMBEDDING_MODEL, + completionModel: process.env.LLAMA_CPP_MODEL, + }, + + // ベクトルDB + database: { + filename: "vector.db", + embeddingDim: Number(process.env.VEC_DIM ?? 3), + }, + + // ロギング + logging: { + level: process.env.LOG_LEVEL || "info", // debug, info, warn, error + }, +}; + +/** バージョン情報 */ +export const VERSION = "1.0.0"; + +/** Tray ツールチップ */ +export const TRAY_TOOLTIP = "SQLite Vector MCP Server"; + +/** Tray メニュー */ +export const TRAY_MENU_LABELS = { + open: "Open Window", + quit: "Quit", +}; + +/** ウィンドウタイトル */ +export const WINDOW_TITLE = "SQLite Vector MCP Server"; diff --git a/src/backend/db.js b/src/backend/db.js new file mode 100644 index 0000000..d10209f --- /dev/null +++ b/src/backend/db.js @@ -0,0 +1,121 @@ +import Database from "better-sqlite3"; +import knex from "knex"; +import * as sqlite_vec from "sqlite-vec"; +import { CONFIG } from "./config.js"; +import { Logger } from "./logger.js"; + +let db = null; +let knexDb = null; + +/** + * SQLite Vector を拡張ライブラリとして読み込む + * @throws {Error} sqlite-vec の読み込みに失敗した場合 + */ +function loadVectorExtension() { + try { + sqlite_vec.load(db); + Logger.debug('sqlite-vec extension loaded'); + } catch (error) { + Logger.error('Failed to load sqlite-vec extension', error); + throw error; + } +} + +/** + * テーブルスキーマを初期化 + * @throws {Error} スキーマの初期化に失敗した場合 + */ +function initializeSchema() { + try { + const sql = ` + CREATE TABLE IF NOT EXISTS items ( + id INTEGER PRIMARY KEY, + content TEXT NOT NULL, + path TEXT, + created_at TEXT DEFAULT (datetime('now')), + updated_at TEXT DEFAULT (datetime('now')) + ); + CREATE VIRTUAL TABLE IF NOT EXISTS vec_items USING vec0( + id INTEGER PRIMARY KEY, + embedding FLOAT[${CONFIG.database.embeddingDim}] + ); + CREATE INDEX IF NOT EXISTS idx_items_created_at ON items(created_at); + CREATE INDEX IF NOT EXISTS idx_items_path ON items(path); + `; + db.exec(sql); + Logger.debug('Database schema initialized', { embeddingDim: CONFIG.database.embeddingDim }); + } catch (error) { + Logger.error('Failed to initialize database schema', error); + throw error; + } +} + +/** + * データベースを初期化 + * @throws {Error} データベースの初期化に失敗した場合 + */ +export function initDb() { + try { + Logger.info('Initializing database'); + + // SQLite データベース接続 + db = new Database(CONFIG.database.filename); + db.pragma('journal_mode = WAL'); + Logger.debug('SQLite database connected', { filename: CONFIG.database.filename }); + + // sqlite-vec 拡張を読み込む + loadVectorExtension(); + + // スキーマを初期化 + initializeSchema(); + + // knex 接続を作成 + knexDb = knex({ + client: "better-sqlite3", + connection: { + filename: CONFIG.database.filename, + }, + useNullAsDefault: true, + }); + + Logger.info('Database initialization completed'); + } catch (error) { + Logger.error('Failed to initialize database', error); + throw error; + } +} + +/** + * エンベディング次元数を取得 + * @returns {number} エンベディング次元数 + */ +export function getEmbeddingDim() { + return CONFIG.database.embeddingDim; +} + +/** + * データベース接続を取得(better-sqlite3) + * @returns {Database} SQLiteデータベース接続 + * @throws {Error} データベースが未初期化の場合 + */ +export function getDb() { + if (!db) { + throw new Error('Database not initialized. Call initDb() first.'); + } + return db; +} + +/** + * Knex クエリビルダを取得 + * @returns {Object} Knex クエリビルダ + * @throws {Error} データベースが未初期化の場合 + */ +export function getKnexDb() { + if (!knexDb) { + throw new Error('Knex database not initialized. Call initDb() first.'); + } + return knexDb; +} + +// エクスポート(後方互換性) +export { db as default }; diff --git a/src/backend/electron-main.js b/src/backend/electron-main.js new file mode 100644 index 0000000..35659f8 --- /dev/null +++ b/src/backend/electron-main.js @@ -0,0 +1,140 @@ +import { app, BrowserWindow, Menu, nativeImage, Tray } from 'electron'; +import path from 'path'; +import { fileURLToPath } from 'url'; +import { CONFIG, TRAY_MENU_LABELS, TRAY_TOOLTIP, WINDOW_TITLE } from './config.js'; +import { Logger } from './logger.js'; +import { startMcpServer } from './mcp-server.js'; + +const __dirname = path.dirname(fileURLToPath(import.meta.url)); + +let mainWindow; +let tray; + +/** + * システムトレイを作成・初期化 + * @throws {Error} アイコン読み込みに失敗した場合 + */ +function createTray() { + try { + const isWin = process.platform === 'win32'; + const iconName = isWin ? 'icon.ico' : 'icon.png'; + const iconPath = path.join(__dirname, iconName); + + Logger.debug('Creating tray icon', { iconPath }); + const icon = nativeImage.createFromPath(iconPath); + + if (icon.isEmpty()) { + Logger.warn('Tray icon is empty'); + } + + tray = new Tray(icon); + const contextMenu = Menu.buildFromTemplate([ + { + label: TRAY_MENU_LABELS.open, + click: () => { + Logger.debug('Opening main window from tray'); + mainWindow.show(); + }, + }, + { type: 'separator' }, + { + label: TRAY_MENU_LABELS.quit, + click: () => { + Logger.info('Quit application from tray'); + app.isQuitting = true; + app.quit(); + }, + }, + ]); + + tray.setToolTip(TRAY_TOOLTIP); + tray.setContextMenu(contextMenu); + + tray.on('double-click', () => { + Logger.debug('Tray double-click'); + mainWindow.show(); + }); + + Logger.info('Tray created successfully'); + } catch (error) { + Logger.error('Failed to create tray', error); + throw error; + } +} + +/** + * メインウィンドウを作成・初期化 + * @throws {Error} ウィンドウ作成に失敗した場合 + */ +function createWindow() { + try { + Logger.debug('Creating main window', CONFIG.window); + + mainWindow = new BrowserWindow({ + width: CONFIG.window.width, + height: CONFIG.window.height, + show: true, + webPreferences: { + nodeIntegration: true, + contextIsolation: false, + }, + }); + + const htmlPath = path.join(__dirname, 'index.html'); + mainWindow.loadFile(htmlPath); + mainWindow.setTitle(WINDOW_TITLE); + + mainWindow.on('close', (event) => { + if (!app.isQuitting) { + Logger.debug('Minimizing to tray instead of closing'); + event.preventDefault(); + mainWindow.hide(); + } + return false; + }); + + mainWindow.on('closed', () => { + Logger.debug('Main window closed'); + mainWindow = null; + }); + + Logger.info('Main window created successfully'); + } catch (error) { + Logger.error('Failed to create main window', error); + throw error; + } +} + +/** + * アプリケーション初期化 + */ +app.whenReady().then(async () => { + try { + Logger.info('Electron app is ready'); + createWindow(); + createTray(); + + const mcpPort = CONFIG.mcp.defaultPort; + await startMcpServer(mcpPort); + Logger.info(`MCP server started on port ${mcpPort}`); + } catch (error) { + Logger.error('Failed to initialize application', error); + app.quit(); + } +}); + +/** + * 全ウィンドウが閉じても終了しない(常駐アプリケーション) + */ +app.on('window-all-closed', () => { + Logger.debug('All windows closed, keeping app alive'); + // 常駐のため何もしない +}); + +/** + * 終了前処理 + */ +app.on('before-quit', () => { + Logger.info('Quitting application'); + app.isQuitting = true; +}); diff --git a/src/backend/llama-client.js b/src/backend/llama-client.js new file mode 100644 index 0000000..d7d071c --- /dev/null +++ b/src/backend/llama-client.js @@ -0,0 +1,185 @@ +import { CONFIG } from "./config.js"; +import { Logger } from "./logger.js"; + +/** + * llama.cpp API の基本 URL を取得 + * @returns {string} ベース URL + */ +function getBaseUrl() { + return CONFIG.llama.baseUrl; +} + +/** + * 埋め込みモデル名を取得 + * @returns {string} モデル名 + * @throws {Error} モデルが設定されていない場合 + */ +function getEmbeddingModel() { + const model = CONFIG.llama.embeddingModel; + if (!model) { + throw new Error("LLAMA_CPP_EMBEDDING_MODEL environment variable is not set"); + } + return model; +} + +/** + * 補完モデル名を取得 + * @returns {string} モデル名 + * @throws {Error} モデルが設定されていない場合 + */ +function getCompletionModel() { + const model = CONFIG.llama.completionModel; + if (!model) { + throw new Error("LLAMA_CPP_MODEL environment variable is not set"); + } + return model; +} + +/** + * テキストから埋め込みベクトルを生成 + * @param {string} text - 埋め込み対象のテキスト + * @param {number} timeout - リクエストタイムアウト(ミリ秒) + * @returns {Promise} 埋め込みベクトル + * @throws {Error} API呼び出しまたはレスポンス解析に失敗した場合 + */ +export async function llamaEmbedding(text, timeout = 30000) { + if (!text || typeof text !== "string") { + throw new Error("Text parameter must be a non-empty string"); + } + + const baseUrl = getBaseUrl(); + const model = getEmbeddingModel(); + + try { + Logger.debug("Calling llama.cpp embeddings API", { + baseUrl, + model, + textLength: text.length, + }); + + const controller = new AbortController(); + const timeoutId = setTimeout(() => controller.abort(), timeout); + + const res = await fetch(`${baseUrl}/embeddings`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ + model, + input: text, + }), + signal: controller.signal, + }); + + clearTimeout(timeoutId); + + if (!res.ok) { + const detail = await res.text(); + Logger.error("llama.cpp embeddings API error", { + status: res.status, + detail, + }); + throw new Error(`llama.cpp embeddings error: ${res.status} ${detail}`); + } + + const data = await res.json(); + + // レスポンス形式に応じて埋め込みを抽出 + const embedding = data?.data?.[0]?.embedding ?? data?.embedding; + + if (!Array.isArray(embedding) || embedding.length === 0) { + Logger.error("Invalid embedding response", { data }); + throw new Error("llama.cpp embeddings response is missing or invalid embedding array"); + } + + Logger.debug("Embedding generated successfully", { + dimension: embedding.length, + }); + + return embedding; + } catch (error) { + if (error.name === "AbortError") { + Logger.error("Embeddings request timeout", { timeout }); + throw new Error(`Embeddings request timed out after ${timeout}ms`); + } + throw error; + } +} + +/** + * llama.cpp でテキスト補完を実行 + * @param {string} prompt - プロンプトテキスト + * @param {Object} options - オプション設定 + * @param {number} options.n_predict - 生成するトークン数(デフォルト: 128) + * @param {number} options.temperature - 温度パラメータ(デフォルト: 0.2) + * @param {number} options.timeout - リクエストタイムアウト(ミリ秒) + * @returns {Promise} 生成されたテキスト + * @throws {Error} API呼び出しまたはレスポンス解析に失敗した場合 + */ +export async function llamaCompletion(prompt, options = {}) { + if (!prompt || typeof prompt !== "string") { + throw new Error("Prompt parameter must be a non-empty string"); + } + + const baseUrl = getBaseUrl(); + const model = getCompletionModel(); + const timeout = options.timeout ?? 60000; + + try { + Logger.debug("Calling llama.cpp completion API", { + baseUrl, + model, + promptLength: prompt.length, + n_predict: options.n_predict ?? 128, + temperature: options.temperature ?? 0.2, + }); + + const controller = new AbortController(); + const timeoutId = setTimeout(() => controller.abort(), timeout); + + const res = await fetch(`${baseUrl}/completion`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ + model, + prompt, + n_predict: options.n_predict ?? 128, + temperature: options.temperature ?? 0.2, + stream: false, + }), + signal: controller.signal, + }); + + clearTimeout(timeoutId); + + if (!res.ok) { + const detail = await res.text(); + Logger.error("llama.cpp completion API error", { + status: res.status, + detail, + }); + throw new Error(`llama.cpp completion error: ${res.status} ${detail}`); + } + + const data = await res.json(); + + // レスポンス形式に応じてテキストを抽出 + const text = data?.content ?? data?.completion ?? data?.response; + + if (typeof text !== "string") { + Logger.error("Invalid completion response", { data }); + throw new Error("llama.cpp completion response is missing or invalid text"); + } + + Logger.debug("Completion generated successfully", { + resultLength: text.length, + }); + + return text; + } catch (error) { + if (error.name === "AbortError") { + Logger.error("Completion request timeout", { timeout }); + throw new Error(`Completion request timed out after ${timeout}ms`); + } + throw error; + } +} diff --git a/src/backend/logger.js b/src/backend/logger.js new file mode 100644 index 0000000..184c6c0 --- /dev/null +++ b/src/backend/logger.js @@ -0,0 +1,40 @@ +/** + * ログユーティリティ + */ + +const LOG_LEVELS = { + debug: 0, + info: 1, + warn: 2, + error: 3, +}; + +let currentLogLevel = + LOG_LEVELS[process.env.LOG_LEVEL?.toLowerCase() || "info"] || 1; + +/** + * ログ出力 + * @param {string} level - ログレベル (debug, info, warn, error) + * @param {string} message - ログメッセージ + * @param {*} data - 追加データ + */ +function log(level, message, data = null) { + const levelValue = LOG_LEVELS[level] || LOG_LEVELS.info; + if (levelValue < currentLogLevel) return; + + const timestamp = new Date().toISOString(); + const prefix = `[${timestamp}] [${level.toUpperCase()}]`; + + if (data) { + console.log(`${prefix} ${message}`, data); + } else { + console.log(`${prefix} ${message}`); + } +} + +export const Logger = { + debug: (msg, data) => log("debug", msg, data), + info: (msg, data) => log("info", msg, data), + warn: (msg, data) => log("warn", msg, data), + error: (msg, data) => log("error", msg, data), +}; diff --git a/src/backend/mcp-handlers.js b/src/backend/mcp-handlers.js new file mode 100644 index 0000000..94a6b99 --- /dev/null +++ b/src/backend/mcp-handlers.js @@ -0,0 +1,229 @@ +import { + CallToolRequestSchema, + ListToolsRequestSchema, +} from "@modelcontextprotocol/sdk/types.js"; +import { getDb, getEmbeddingDim, getKnexDb } from "./db.js"; +import { llamaCompletion, llamaEmbedding } from "./llama-client.js"; +import { Logger } from "./logger.js"; +import { getToolDefinitions, TOOL_NAMES } from "./mcp-tools.js"; + +/** + * ベクトルの次元を検証 + * @param {number[]} vector - 検証するベクトル + * @throws {Error} 次元が不正な場合 + */ +function validateEmbeddingDimension(vector) { + const expectedDim = getEmbeddingDim(); + if (!Array.isArray(vector) || vector.length !== expectedDim) { + const error = `埋め込み次元が不正です。期待: ${expectedDim}次元、取得: ${vector?.length || 'undefined'}次元`; + Logger.error(error); + throw new Error(error); + } +} + +/** + * アイテムを追加(テキストから埋め込みを自動生成) + * @param {string} content - テキスト内容 + * @param {string} path - メタデータパス + * @returns {Promise} 処理結果 + */ +async function handleAddItemText(content, path) { + try { + Logger.debug('Adding item from text', { contentLength: content.length }); + const embedding = await llamaEmbedding(content); + validateEmbeddingDimension(embedding); + + const db = getDb(); + const knexDb = getKnexDb(); + + const insertIds = await knexDb("items").insert({ content, path }); + const id = Array.isArray(insertIds) ? insertIds[0] : insertIds; + + db.prepare("INSERT INTO vec_items(id, embedding) VALUES (?, ?)") + .run(id, new Float32Array(embedding)); + + Logger.info(`Item added successfully with id: ${id}`); + return { content: [{ type: "text", text: `Added item with id ${id}` }] }; + } catch (error) { + Logger.error('Failed to add item from text', error); + throw error; + } +} + +/** + * アイテムを追加(ベクトルを直接指定) + * @param {string} content - テキスト内容 + * @param {number[]} vector - 埋め込みベクトル + * @param {string} path - メタデータパス + * @returns {Promise} 処理結果 + */ +async function handleAddItem(content, vector, path) { + try { + Logger.debug('Adding item with vector', { contentLength: content.length }); + validateEmbeddingDimension(vector); + + const db = getDb(); + const knexDb = getKnexDb(); + + const insertIds = await knexDb("items").insert({ content, path }); + const id = Array.isArray(insertIds) ? insertIds[0] : insertIds; + + db.prepare("INSERT INTO vec_items(id, embedding) VALUES (?, ?)") + .run(id, new Float32Array(vector)); + + Logger.info(`Item added successfully with id: ${id}`); + return { content: [{ type: "text", text: `Added item with id ${id}` }] }; + } catch (error) { + Logger.error('Failed to add item', error); + throw error; + } +} + +/** + * テキストから埋め込みを生成して検索 + * @param {string} content - 検索キーワード + * @param {number} limit - 結果の上限 + * @returns {Promise} 検索結果 + */ +async function handleSearchText(content, limit = 10) { + try { + Logger.debug('Searching by text', { contentLength: content.length, limit }); + const embedding = await llamaEmbedding(content); + validateEmbeddingDimension(embedding); + + const db = getDb(); + const results = db.prepare(` + SELECT + i.id, + i.content, + i.path, + i.created_at, + i.updated_at, + v.distance + FROM vec_items v + JOIN items i ON v.id = i.id + WHERE embedding MATCH ? + ORDER BY distance + LIMIT ? + `).all(new Float32Array(embedding), limit); + + Logger.info(`Text search completed, found ${results.length} results`); + return { + content: [{ type: "text", text: JSON.stringify(results, null, 2) }], + }; + } catch (error) { + Logger.error('Failed to search by text', error); + throw error; + } +} + +/** + * ベクトルで直接検索 + * @param {number[]} vector - 検索ベクトル + * @param {number} limit - 結果の上限 + * @returns {Object} 検索結果 + */ +function handleSearchVector(vector, limit = 10) { + try { + Logger.debug('Searching by vector', { limit }); + validateEmbeddingDimension(vector); + + const db = getDb(); + const results = db.prepare(` + SELECT + i.id, + i.content, + i.path, + i.created_at, + i.updated_at, + v.distance + FROM vec_items v + JOIN items i ON v.id = i.id + WHERE embedding MATCH ? + ORDER BY distance + LIMIT ? + `).all(new Float32Array(vector), limit); + + Logger.info(`Vector search completed, found ${results.length} results`); + return { + content: [{ type: "text", text: JSON.stringify(results, null, 2) }], + }; + } catch (error) { + Logger.error('Failed to search by vector', error); + throw error; + } +} + +/** + * LLM で テキスト生成を実行 + * @param {string} prompt - プロンプト + * @param {number} n_predict - 生成トークン数 + * @param {number} temperature - 温度パラメータ + * @returns {Promise} 生成結果 + */ +async function handleLlmGenerate(prompt, n_predict, temperature) { + try { + Logger.debug('Generating text via LLM', { promptLength: prompt.length, n_predict, temperature }); + const text = await llamaCompletion(prompt, { n_predict, temperature }); + Logger.info(`LLM generation completed, result length: ${text.length}`); + return { content: [{ type: "text", text }] }; + } catch (error) { + Logger.error('Failed to generate text via LLM', error); + throw error; + } +} + +/** + * MCP ハンドラーを登録 + * @param {Server} server - MCP サーバーインスタンス + */ +export function registerMcpHandlers(server) { + /** + * ツール一覧を返すハンドラー + */ + server.setRequestHandler(ListToolsRequestSchema, async () => { + Logger.debug('Listing available tools'); + return { + tools: getToolDefinitions(), + }; + }); + + /** + * ツール呼び出しハンドラー + */ + server.setRequestHandler(CallToolRequestSchema, async (request) => { + const toolName = request.params.name; + const args = request.params.arguments; + + try { + Logger.debug(`Tool called: ${toolName}`); + + switch (toolName) { + case TOOL_NAMES.ADD_ITEM_TEXT: + return await handleAddItemText(args.content, args.path); + + case TOOL_NAMES.ADD_ITEM: + return await handleAddItem(args.content, args.vector, args.path); + + case TOOL_NAMES.SEARCH_TEXT: + return await handleSearchText(args.content, args.limit); + + case TOOL_NAMES.SEARCH_VECTOR: + return handleSearchVector(args.vector, args.limit); + + case TOOL_NAMES.LLM_GENERATE: + return await handleLlmGenerate(args.prompt, args.n_predict, args.temperature); + + default: + const error = `不明なツール: ${toolName}`; + Logger.error(error); + throw new Error(error); + } + } catch (error) { + Logger.error(`Tool execution failed: ${toolName}`, error); + throw error; + } + }); + + Logger.info('MCP handlers registered successfully'); +} diff --git a/src/backend/mcp-server.js b/src/backend/mcp-server.js new file mode 100644 index 0000000..8e72448 --- /dev/null +++ b/src/backend/mcp-server.js @@ -0,0 +1,124 @@ +import { Server } from "@modelcontextprotocol/sdk/server/index.js"; +import { SSEServerTransport } from "@modelcontextprotocol/sdk/server/sse.js"; +import express from "express"; +import { CONFIG } from "./config.js"; +import { initDb } from "./db.js"; +import { Logger } from "./logger.js"; +import { registerMcpHandlers } from "./mcp-handlers.js"; + +const app = express(); +let transport = null; +let httpServer = null; + +/** + * 初期化処理を実行 + */ +try { + initDb(); + Logger.info('Database initialized'); +} catch (error) { + Logger.error('Failed to initialize database', error); + process.exit(1); +} + +/** + * MCP サーバーを作成 + */ +const server = new Server( + { + name: CONFIG.mcp.name, + version: CONFIG.mcp.version, + }, + { + capabilities: { + tools: {}, + }, + } +); + +registerMcpHandlers(server); + +/** + * SSE トランスポートエラーハンドラ + * @param {Error} error - エラーオブジェクト + */ +function handleTransportError(error) { + Logger.error('Transport error', error); + transport = null; +} + +/** + * MCP SSE サーバーを起動 + * @param {number} port - ポート番号 + * @returns {Object} HTTPサーバーオブジェクト + * @throws {Error} サーバー起動に失敗した場合 + */ +export function startMcpServer(port = 3000) { + try { + // ポート番号の検証 + if (!Number.isInteger(port) || port < 1 || port > 65535) { + throw new Error(`Invalid port number: ${port}`); + } + + app.use(express.json()); + + /** + * SSE 接続エンドポイント + * @route GET /sse + */ + app.get(CONFIG.mcp.ssePath, async (req, res) => { + try { + Logger.info('SSE connection established'); + transport = new SSEServerTransport(CONFIG.mcp.messagesPath, res); + transport.on('error', handleTransportError); + await server.connect(transport); + } catch (error) { + Logger.error('Failed to establish SSE connection', error); + res.status(500).send('Failed to establish SSE connection'); + } + }); + + /** + * メッセージハンドラエンドポイント + * @route POST /messages + */ + app.post(CONFIG.mcp.messagesPath, async (req, res) => { + try { + if (!transport) { + Logger.warn('No active transport connection'); + return res.status(400).send('No transport connection'); + } + await transport.handlePostMessage(req, res); + } catch (error) { + Logger.error('Failed to handle post message', error); + res.status(500).send('Failed to handle message'); + } + }); + + /** + * ヘルスチェックエンドポイント + * @route GET /health + */ + app.get('/health', (req, res) => { + res.status(200).json({ + status: 'ok', + timestamp: new Date().toISOString(), + transport: transport ? 'connected' : 'disconnected', + }); + }); + + httpServer = app.listen(port, () => { + Logger.info(`MCP SSE Server listening on http://localhost:${port}${CONFIG.mcp.ssePath}`); + }); + + // エラーハンドリング + httpServer.on('error', (error) => { + Logger.error('HTTP server error', error); + }); + + return httpServer; + } catch (error) { + Logger.error('Failed to start MCP server', error); + throw error; + } +} diff --git a/src/backend/mcp-tools.js b/src/backend/mcp-tools.js new file mode 100644 index 0000000..64486bd --- /dev/null +++ b/src/backend/mcp-tools.js @@ -0,0 +1,144 @@ +import { CONFIG } from "./config.js"; +import { Logger } from "./logger.js"; + +/** + * MCP ツール定義を取得 + * @returns {Array} ツール定義の配列 + * @description + * - add_item_text: テキストから埋め込みを自動生成して保存 + * - add_item: テキストと埋め込みベクトルを直接保存 + * - search_text: テキストから埋め込みを自動生成して検索 + * - search_vector: ベクトルで直接検索 + * - llm_generate: llama.cpp でテキスト生成を実行 + */ +export function getToolDefinitions() { + const embeddingDim = CONFIG.database.embeddingDim; + + return [ + { + name: "add_item_text", + description: "テキストから埋め込みを生成して保存します", + inputSchema: { + type: "object", + properties: { + content: { + type: "string", + description: "保存するテキスト内容", + }, + path: { + type: "string", + description: "アイテムのメタデータパス(オプション)", + }, + }, + required: ["content"], + }, + }, + { + name: "add_item", + description: "テキストと埋め込みベクトルを直接保存します", + inputSchema: { + type: "object", + properties: { + content: { + type: "string", + description: "保存するテキスト内容", + }, + path: { + type: "string", + description: "アイテムのメタデータパス(オプション)", + }, + vector: { + type: "array", + items: { type: "number" }, + minItems: embeddingDim, + maxItems: embeddingDim, + description: `埋め込みベクトル (${embeddingDim}次元)`, + }, + }, + required: ["content", "vector"], + }, + }, + { + name: "search_text", + description: "テキストから埋め込みを生成して類似アイテムを検索します", + inputSchema: { + type: "object", + properties: { + content: { + type: "string", + description: "検索キーワード", + }, + limit: { + type: "number", + description: "返す結果の最大数(デフォルト: 10)", + minimum: 1, + maximum: 100, + }, + }, + required: ["content"], + }, + }, + { + name: "search_vector", + description: "ベクトルで類似度検索を実行します", + inputSchema: { + type: "object", + properties: { + vector: { + type: "array", + items: { type: "number" }, + minItems: embeddingDim, + maxItems: embeddingDim, + description: `検索するベクトル (${embeddingDim}次元)`, + }, + limit: { + type: "number", + description: "返す結果の最大数(デフォルト: 10)", + minimum: 1, + maximum: 100, + }, + }, + required: ["vector"], + }, + }, + { + name: "llm_generate", + description: "llama.cpp でテキスト生成を実行します", + inputSchema: { + type: "object", + properties: { + prompt: { + type: "string", + description: "生成するテキストのプロンプト", + }, + n_predict: { + type: "number", + description: "生成するトークン数(デフォルト: 128)", + minimum: 1, + maximum: 2048, + }, + temperature: { + type: "number", + description: "温度パラメータ(0.0-2.0、デフォルト: 0.7)", + minimum: 0, + maximum: 2, + }, + }, + required: ["prompt"], + }, + }, + ]; +} + +/** + * ツール名の定数 + */ +export const TOOL_NAMES = { + ADD_ITEM_TEXT: "add_item_text", + ADD_ITEM: "add_item", + SEARCH_TEXT: "search_text", + SEARCH_VECTOR: "search_vector", + LLM_GENERATE: "llm_generate", +}; + +Logger.debug("MCP tools module loaded"); diff --git a/src/config.js b/src/config.js deleted file mode 100644 index a7a0010..0000000 --- a/src/config.js +++ /dev/null @@ -1,53 +0,0 @@ -/** - * アプリケーション設定・定数 - */ - -export const CONFIG = { - // Electron ウィンドウ - window: { - width: 600, - height: 400, - }, - - // MCP サーバー - mcp: { - name: "sqlite-vec-server", - version: "1.0.0", - defaultPort: process.env.MCP_PORT ? parseInt(process.env.MCP_PORT, 10) : 3000, - ssePath: "/sse", - messagesPath: "/messages", - }, - - // llama.cpp - llama: { - baseUrl: process.env.LLAMA_CPP_BASE_URL || "http://127.0.0.1:8080", - embeddingModel: process.env.LLAMA_CPP_EMBEDDING_MODEL, - completionModel: process.env.LLAMA_CPP_MODEL, - }, - - // ベクトルDB - database: { - filename: "vector.db", - embeddingDim: Number(process.env.VEC_DIM ?? 3), - }, - - // ロギング - logging: { - level: process.env.LOG_LEVEL || "info", // debug, info, warn, error - }, -}; - -/** バージョン情報 */ -export const VERSION = "1.0.0"; - -/** Tray ツールチップ */ -export const TRAY_TOOLTIP = "SQLite Vector MCP Server"; - -/** Tray メニュー */ -export const TRAY_MENU_LABELS = { - open: "Open Window", - quit: "Quit", -}; - -/** ウィンドウタイトル */ -export const WINDOW_TITLE = "SQLite Vector MCP Server"; diff --git a/src/db.js b/src/db.js deleted file mode 100644 index d10209f..0000000 --- a/src/db.js +++ /dev/null @@ -1,121 +0,0 @@ -import Database from "better-sqlite3"; -import knex from "knex"; -import * as sqlite_vec from "sqlite-vec"; -import { CONFIG } from "./config.js"; -import { Logger } from "./logger.js"; - -let db = null; -let knexDb = null; - -/** - * SQLite Vector を拡張ライブラリとして読み込む - * @throws {Error} sqlite-vec の読み込みに失敗した場合 - */ -function loadVectorExtension() { - try { - sqlite_vec.load(db); - Logger.debug('sqlite-vec extension loaded'); - } catch (error) { - Logger.error('Failed to load sqlite-vec extension', error); - throw error; - } -} - -/** - * テーブルスキーマを初期化 - * @throws {Error} スキーマの初期化に失敗した場合 - */ -function initializeSchema() { - try { - const sql = ` - CREATE TABLE IF NOT EXISTS items ( - id INTEGER PRIMARY KEY, - content TEXT NOT NULL, - path TEXT, - created_at TEXT DEFAULT (datetime('now')), - updated_at TEXT DEFAULT (datetime('now')) - ); - CREATE VIRTUAL TABLE IF NOT EXISTS vec_items USING vec0( - id INTEGER PRIMARY KEY, - embedding FLOAT[${CONFIG.database.embeddingDim}] - ); - CREATE INDEX IF NOT EXISTS idx_items_created_at ON items(created_at); - CREATE INDEX IF NOT EXISTS idx_items_path ON items(path); - `; - db.exec(sql); - Logger.debug('Database schema initialized', { embeddingDim: CONFIG.database.embeddingDim }); - } catch (error) { - Logger.error('Failed to initialize database schema', error); - throw error; - } -} - -/** - * データベースを初期化 - * @throws {Error} データベースの初期化に失敗した場合 - */ -export function initDb() { - try { - Logger.info('Initializing database'); - - // SQLite データベース接続 - db = new Database(CONFIG.database.filename); - db.pragma('journal_mode = WAL'); - Logger.debug('SQLite database connected', { filename: CONFIG.database.filename }); - - // sqlite-vec 拡張を読み込む - loadVectorExtension(); - - // スキーマを初期化 - initializeSchema(); - - // knex 接続を作成 - knexDb = knex({ - client: "better-sqlite3", - connection: { - filename: CONFIG.database.filename, - }, - useNullAsDefault: true, - }); - - Logger.info('Database initialization completed'); - } catch (error) { - Logger.error('Failed to initialize database', error); - throw error; - } -} - -/** - * エンベディング次元数を取得 - * @returns {number} エンベディング次元数 - */ -export function getEmbeddingDim() { - return CONFIG.database.embeddingDim; -} - -/** - * データベース接続を取得(better-sqlite3) - * @returns {Database} SQLiteデータベース接続 - * @throws {Error} データベースが未初期化の場合 - */ -export function getDb() { - if (!db) { - throw new Error('Database not initialized. Call initDb() first.'); - } - return db; -} - -/** - * Knex クエリビルダを取得 - * @returns {Object} Knex クエリビルダ - * @throws {Error} データベースが未初期化の場合 - */ -export function getKnexDb() { - if (!knexDb) { - throw new Error('Knex database not initialized. Call initDb() first.'); - } - return knexDb; -} - -// エクスポート(後方互換性) -export { db as default }; diff --git a/src/frontend/icon.ico b/src/frontend/icon.ico new file mode 100644 index 0000000..2ee0da8 --- /dev/null +++ b/src/frontend/icon.ico Binary files differ diff --git a/src/frontend/index.html b/src/frontend/index.html new file mode 100644 index 0000000..c879cde --- /dev/null +++ b/src/frontend/index.html @@ -0,0 +1,38 @@ + + + + + + SQLite Vector MCP Server + + + +

SQLite Vector MCP Server

+

Status: Tauri is running

+
+ + +
+

+ + +
+

Database Info

+
Waiting for data...
+ + diff --git a/src/icon.ico b/src/icon.ico deleted file mode 100644 index 2ee0da8..0000000 --- a/src/icon.ico +++ /dev/null Binary files differ diff --git a/src/index.html b/src/index.html deleted file mode 100644 index 2084690..0000000 --- a/src/index.html +++ /dev/null @@ -1,20 +0,0 @@ - - - - - - SQLite Vector MCP Server - - - -

SQLite Vector MCP Server

-

Status: Running

-

SSE Endpoint: http://localhost:3000/sse

-
-

Database Info

-
Waiting for data...
- - diff --git a/src/llama-client.js b/src/llama-client.js deleted file mode 100644 index d7d071c..0000000 --- a/src/llama-client.js +++ /dev/null @@ -1,185 +0,0 @@ -import { CONFIG } from "./config.js"; -import { Logger } from "./logger.js"; - -/** - * llama.cpp API の基本 URL を取得 - * @returns {string} ベース URL - */ -function getBaseUrl() { - return CONFIG.llama.baseUrl; -} - -/** - * 埋め込みモデル名を取得 - * @returns {string} モデル名 - * @throws {Error} モデルが設定されていない場合 - */ -function getEmbeddingModel() { - const model = CONFIG.llama.embeddingModel; - if (!model) { - throw new Error("LLAMA_CPP_EMBEDDING_MODEL environment variable is not set"); - } - return model; -} - -/** - * 補完モデル名を取得 - * @returns {string} モデル名 - * @throws {Error} モデルが設定されていない場合 - */ -function getCompletionModel() { - const model = CONFIG.llama.completionModel; - if (!model) { - throw new Error("LLAMA_CPP_MODEL environment variable is not set"); - } - return model; -} - -/** - * テキストから埋め込みベクトルを生成 - * @param {string} text - 埋め込み対象のテキスト - * @param {number} timeout - リクエストタイムアウト(ミリ秒) - * @returns {Promise} 埋め込みベクトル - * @throws {Error} API呼び出しまたはレスポンス解析に失敗した場合 - */ -export async function llamaEmbedding(text, timeout = 30000) { - if (!text || typeof text !== "string") { - throw new Error("Text parameter must be a non-empty string"); - } - - const baseUrl = getBaseUrl(); - const model = getEmbeddingModel(); - - try { - Logger.debug("Calling llama.cpp embeddings API", { - baseUrl, - model, - textLength: text.length, - }); - - const controller = new AbortController(); - const timeoutId = setTimeout(() => controller.abort(), timeout); - - const res = await fetch(`${baseUrl}/embeddings`, { - method: "POST", - headers: { "Content-Type": "application/json" }, - body: JSON.stringify({ - model, - input: text, - }), - signal: controller.signal, - }); - - clearTimeout(timeoutId); - - if (!res.ok) { - const detail = await res.text(); - Logger.error("llama.cpp embeddings API error", { - status: res.status, - detail, - }); - throw new Error(`llama.cpp embeddings error: ${res.status} ${detail}`); - } - - const data = await res.json(); - - // レスポンス形式に応じて埋め込みを抽出 - const embedding = data?.data?.[0]?.embedding ?? data?.embedding; - - if (!Array.isArray(embedding) || embedding.length === 0) { - Logger.error("Invalid embedding response", { data }); - throw new Error("llama.cpp embeddings response is missing or invalid embedding array"); - } - - Logger.debug("Embedding generated successfully", { - dimension: embedding.length, - }); - - return embedding; - } catch (error) { - if (error.name === "AbortError") { - Logger.error("Embeddings request timeout", { timeout }); - throw new Error(`Embeddings request timed out after ${timeout}ms`); - } - throw error; - } -} - -/** - * llama.cpp でテキスト補完を実行 - * @param {string} prompt - プロンプトテキスト - * @param {Object} options - オプション設定 - * @param {number} options.n_predict - 生成するトークン数(デフォルト: 128) - * @param {number} options.temperature - 温度パラメータ(デフォルト: 0.2) - * @param {number} options.timeout - リクエストタイムアウト(ミリ秒) - * @returns {Promise} 生成されたテキスト - * @throws {Error} API呼び出しまたはレスポンス解析に失敗した場合 - */ -export async function llamaCompletion(prompt, options = {}) { - if (!prompt || typeof prompt !== "string") { - throw new Error("Prompt parameter must be a non-empty string"); - } - - const baseUrl = getBaseUrl(); - const model = getCompletionModel(); - const timeout = options.timeout ?? 60000; - - try { - Logger.debug("Calling llama.cpp completion API", { - baseUrl, - model, - promptLength: prompt.length, - n_predict: options.n_predict ?? 128, - temperature: options.temperature ?? 0.2, - }); - - const controller = new AbortController(); - const timeoutId = setTimeout(() => controller.abort(), timeout); - - const res = await fetch(`${baseUrl}/completion`, { - method: "POST", - headers: { "Content-Type": "application/json" }, - body: JSON.stringify({ - model, - prompt, - n_predict: options.n_predict ?? 128, - temperature: options.temperature ?? 0.2, - stream: false, - }), - signal: controller.signal, - }); - - clearTimeout(timeoutId); - - if (!res.ok) { - const detail = await res.text(); - Logger.error("llama.cpp completion API error", { - status: res.status, - detail, - }); - throw new Error(`llama.cpp completion error: ${res.status} ${detail}`); - } - - const data = await res.json(); - - // レスポンス形式に応じてテキストを抽出 - const text = data?.content ?? data?.completion ?? data?.response; - - if (typeof text !== "string") { - Logger.error("Invalid completion response", { data }); - throw new Error("llama.cpp completion response is missing or invalid text"); - } - - Logger.debug("Completion generated successfully", { - resultLength: text.length, - }); - - return text; - } catch (error) { - if (error.name === "AbortError") { - Logger.error("Completion request timeout", { timeout }); - throw new Error(`Completion request timed out after ${timeout}ms`); - } - throw error; - } -} diff --git a/src/logger.js b/src/logger.js deleted file mode 100644 index 184c6c0..0000000 --- a/src/logger.js +++ /dev/null @@ -1,40 +0,0 @@ -/** - * ログユーティリティ - */ - -const LOG_LEVELS = { - debug: 0, - info: 1, - warn: 2, - error: 3, -}; - -let currentLogLevel = - LOG_LEVELS[process.env.LOG_LEVEL?.toLowerCase() || "info"] || 1; - -/** - * ログ出力 - * @param {string} level - ログレベル (debug, info, warn, error) - * @param {string} message - ログメッセージ - * @param {*} data - 追加データ - */ -function log(level, message, data = null) { - const levelValue = LOG_LEVELS[level] || LOG_LEVELS.info; - if (levelValue < currentLogLevel) return; - - const timestamp = new Date().toISOString(); - const prefix = `[${timestamp}] [${level.toUpperCase()}]`; - - if (data) { - console.log(`${prefix} ${message}`, data); - } else { - console.log(`${prefix} ${message}`); - } -} - -export const Logger = { - debug: (msg, data) => log("debug", msg, data), - info: (msg, data) => log("info", msg, data), - warn: (msg, data) => log("warn", msg, data), - error: (msg, data) => log("error", msg, data), -}; diff --git a/src/main.js b/src/main.js deleted file mode 100644 index 35659f8..0000000 --- a/src/main.js +++ /dev/null @@ -1,140 +0,0 @@ -import { app, BrowserWindow, Menu, nativeImage, Tray } from 'electron'; -import path from 'path'; -import { fileURLToPath } from 'url'; -import { CONFIG, TRAY_MENU_LABELS, TRAY_TOOLTIP, WINDOW_TITLE } from './config.js'; -import { Logger } from './logger.js'; -import { startMcpServer } from './mcp-server.js'; - -const __dirname = path.dirname(fileURLToPath(import.meta.url)); - -let mainWindow; -let tray; - -/** - * システムトレイを作成・初期化 - * @throws {Error} アイコン読み込みに失敗した場合 - */ -function createTray() { - try { - const isWin = process.platform === 'win32'; - const iconName = isWin ? 'icon.ico' : 'icon.png'; - const iconPath = path.join(__dirname, iconName); - - Logger.debug('Creating tray icon', { iconPath }); - const icon = nativeImage.createFromPath(iconPath); - - if (icon.isEmpty()) { - Logger.warn('Tray icon is empty'); - } - - tray = new Tray(icon); - const contextMenu = Menu.buildFromTemplate([ - { - label: TRAY_MENU_LABELS.open, - click: () => { - Logger.debug('Opening main window from tray'); - mainWindow.show(); - }, - }, - { type: 'separator' }, - { - label: TRAY_MENU_LABELS.quit, - click: () => { - Logger.info('Quit application from tray'); - app.isQuitting = true; - app.quit(); - }, - }, - ]); - - tray.setToolTip(TRAY_TOOLTIP); - tray.setContextMenu(contextMenu); - - tray.on('double-click', () => { - Logger.debug('Tray double-click'); - mainWindow.show(); - }); - - Logger.info('Tray created successfully'); - } catch (error) { - Logger.error('Failed to create tray', error); - throw error; - } -} - -/** - * メインウィンドウを作成・初期化 - * @throws {Error} ウィンドウ作成に失敗した場合 - */ -function createWindow() { - try { - Logger.debug('Creating main window', CONFIG.window); - - mainWindow = new BrowserWindow({ - width: CONFIG.window.width, - height: CONFIG.window.height, - show: true, - webPreferences: { - nodeIntegration: true, - contextIsolation: false, - }, - }); - - const htmlPath = path.join(__dirname, 'index.html'); - mainWindow.loadFile(htmlPath); - mainWindow.setTitle(WINDOW_TITLE); - - mainWindow.on('close', (event) => { - if (!app.isQuitting) { - Logger.debug('Minimizing to tray instead of closing'); - event.preventDefault(); - mainWindow.hide(); - } - return false; - }); - - mainWindow.on('closed', () => { - Logger.debug('Main window closed'); - mainWindow = null; - }); - - Logger.info('Main window created successfully'); - } catch (error) { - Logger.error('Failed to create main window', error); - throw error; - } -} - -/** - * アプリケーション初期化 - */ -app.whenReady().then(async () => { - try { - Logger.info('Electron app is ready'); - createWindow(); - createTray(); - - const mcpPort = CONFIG.mcp.defaultPort; - await startMcpServer(mcpPort); - Logger.info(`MCP server started on port ${mcpPort}`); - } catch (error) { - Logger.error('Failed to initialize application', error); - app.quit(); - } -}); - -/** - * 全ウィンドウが閉じても終了しない(常駐アプリケーション) - */ -app.on('window-all-closed', () => { - Logger.debug('All windows closed, keeping app alive'); - // 常駐のため何もしない -}); - -/** - * 終了前処理 - */ -app.on('before-quit', () => { - Logger.info('Quitting application'); - app.isQuitting = true; -}); diff --git a/src/mcp-handlers.js b/src/mcp-handlers.js deleted file mode 100644 index 94a6b99..0000000 --- a/src/mcp-handlers.js +++ /dev/null @@ -1,229 +0,0 @@ -import { - CallToolRequestSchema, - ListToolsRequestSchema, -} from "@modelcontextprotocol/sdk/types.js"; -import { getDb, getEmbeddingDim, getKnexDb } from "./db.js"; -import { llamaCompletion, llamaEmbedding } from "./llama-client.js"; -import { Logger } from "./logger.js"; -import { getToolDefinitions, TOOL_NAMES } from "./mcp-tools.js"; - -/** - * ベクトルの次元を検証 - * @param {number[]} vector - 検証するベクトル - * @throws {Error} 次元が不正な場合 - */ -function validateEmbeddingDimension(vector) { - const expectedDim = getEmbeddingDim(); - if (!Array.isArray(vector) || vector.length !== expectedDim) { - const error = `埋め込み次元が不正です。期待: ${expectedDim}次元、取得: ${vector?.length || 'undefined'}次元`; - Logger.error(error); - throw new Error(error); - } -} - -/** - * アイテムを追加(テキストから埋め込みを自動生成) - * @param {string} content - テキスト内容 - * @param {string} path - メタデータパス - * @returns {Promise} 処理結果 - */ -async function handleAddItemText(content, path) { - try { - Logger.debug('Adding item from text', { contentLength: content.length }); - const embedding = await llamaEmbedding(content); - validateEmbeddingDimension(embedding); - - const db = getDb(); - const knexDb = getKnexDb(); - - const insertIds = await knexDb("items").insert({ content, path }); - const id = Array.isArray(insertIds) ? insertIds[0] : insertIds; - - db.prepare("INSERT INTO vec_items(id, embedding) VALUES (?, ?)") - .run(id, new Float32Array(embedding)); - - Logger.info(`Item added successfully with id: ${id}`); - return { content: [{ type: "text", text: `Added item with id ${id}` }] }; - } catch (error) { - Logger.error('Failed to add item from text', error); - throw error; - } -} - -/** - * アイテムを追加(ベクトルを直接指定) - * @param {string} content - テキスト内容 - * @param {number[]} vector - 埋め込みベクトル - * @param {string} path - メタデータパス - * @returns {Promise} 処理結果 - */ -async function handleAddItem(content, vector, path) { - try { - Logger.debug('Adding item with vector', { contentLength: content.length }); - validateEmbeddingDimension(vector); - - const db = getDb(); - const knexDb = getKnexDb(); - - const insertIds = await knexDb("items").insert({ content, path }); - const id = Array.isArray(insertIds) ? insertIds[0] : insertIds; - - db.prepare("INSERT INTO vec_items(id, embedding) VALUES (?, ?)") - .run(id, new Float32Array(vector)); - - Logger.info(`Item added successfully with id: ${id}`); - return { content: [{ type: "text", text: `Added item with id ${id}` }] }; - } catch (error) { - Logger.error('Failed to add item', error); - throw error; - } -} - -/** - * テキストから埋め込みを生成して検索 - * @param {string} content - 検索キーワード - * @param {number} limit - 結果の上限 - * @returns {Promise} 検索結果 - */ -async function handleSearchText(content, limit = 10) { - try { - Logger.debug('Searching by text', { contentLength: content.length, limit }); - const embedding = await llamaEmbedding(content); - validateEmbeddingDimension(embedding); - - const db = getDb(); - const results = db.prepare(` - SELECT - i.id, - i.content, - i.path, - i.created_at, - i.updated_at, - v.distance - FROM vec_items v - JOIN items i ON v.id = i.id - WHERE embedding MATCH ? - ORDER BY distance - LIMIT ? - `).all(new Float32Array(embedding), limit); - - Logger.info(`Text search completed, found ${results.length} results`); - return { - content: [{ type: "text", text: JSON.stringify(results, null, 2) }], - }; - } catch (error) { - Logger.error('Failed to search by text', error); - throw error; - } -} - -/** - * ベクトルで直接検索 - * @param {number[]} vector - 検索ベクトル - * @param {number} limit - 結果の上限 - * @returns {Object} 検索結果 - */ -function handleSearchVector(vector, limit = 10) { - try { - Logger.debug('Searching by vector', { limit }); - validateEmbeddingDimension(vector); - - const db = getDb(); - const results = db.prepare(` - SELECT - i.id, - i.content, - i.path, - i.created_at, - i.updated_at, - v.distance - FROM vec_items v - JOIN items i ON v.id = i.id - WHERE embedding MATCH ? - ORDER BY distance - LIMIT ? - `).all(new Float32Array(vector), limit); - - Logger.info(`Vector search completed, found ${results.length} results`); - return { - content: [{ type: "text", text: JSON.stringify(results, null, 2) }], - }; - } catch (error) { - Logger.error('Failed to search by vector', error); - throw error; - } -} - -/** - * LLM で テキスト生成を実行 - * @param {string} prompt - プロンプト - * @param {number} n_predict - 生成トークン数 - * @param {number} temperature - 温度パラメータ - * @returns {Promise} 生成結果 - */ -async function handleLlmGenerate(prompt, n_predict, temperature) { - try { - Logger.debug('Generating text via LLM', { promptLength: prompt.length, n_predict, temperature }); - const text = await llamaCompletion(prompt, { n_predict, temperature }); - Logger.info(`LLM generation completed, result length: ${text.length}`); - return { content: [{ type: "text", text }] }; - } catch (error) { - Logger.error('Failed to generate text via LLM', error); - throw error; - } -} - -/** - * MCP ハンドラーを登録 - * @param {Server} server - MCP サーバーインスタンス - */ -export function registerMcpHandlers(server) { - /** - * ツール一覧を返すハンドラー - */ - server.setRequestHandler(ListToolsRequestSchema, async () => { - Logger.debug('Listing available tools'); - return { - tools: getToolDefinitions(), - }; - }); - - /** - * ツール呼び出しハンドラー - */ - server.setRequestHandler(CallToolRequestSchema, async (request) => { - const toolName = request.params.name; - const args = request.params.arguments; - - try { - Logger.debug(`Tool called: ${toolName}`); - - switch (toolName) { - case TOOL_NAMES.ADD_ITEM_TEXT: - return await handleAddItemText(args.content, args.path); - - case TOOL_NAMES.ADD_ITEM: - return await handleAddItem(args.content, args.vector, args.path); - - case TOOL_NAMES.SEARCH_TEXT: - return await handleSearchText(args.content, args.limit); - - case TOOL_NAMES.SEARCH_VECTOR: - return handleSearchVector(args.vector, args.limit); - - case TOOL_NAMES.LLM_GENERATE: - return await handleLlmGenerate(args.prompt, args.n_predict, args.temperature); - - default: - const error = `不明なツール: ${toolName}`; - Logger.error(error); - throw new Error(error); - } - } catch (error) { - Logger.error(`Tool execution failed: ${toolName}`, error); - throw error; - } - }); - - Logger.info('MCP handlers registered successfully'); -} diff --git a/src/mcp-server.js b/src/mcp-server.js deleted file mode 100644 index 8e72448..0000000 --- a/src/mcp-server.js +++ /dev/null @@ -1,124 +0,0 @@ -import { Server } from "@modelcontextprotocol/sdk/server/index.js"; -import { SSEServerTransport } from "@modelcontextprotocol/sdk/server/sse.js"; -import express from "express"; -import { CONFIG } from "./config.js"; -import { initDb } from "./db.js"; -import { Logger } from "./logger.js"; -import { registerMcpHandlers } from "./mcp-handlers.js"; - -const app = express(); -let transport = null; -let httpServer = null; - -/** - * 初期化処理を実行 - */ -try { - initDb(); - Logger.info('Database initialized'); -} catch (error) { - Logger.error('Failed to initialize database', error); - process.exit(1); -} - -/** - * MCP サーバーを作成 - */ -const server = new Server( - { - name: CONFIG.mcp.name, - version: CONFIG.mcp.version, - }, - { - capabilities: { - tools: {}, - }, - } -); - -registerMcpHandlers(server); - -/** - * SSE トランスポートエラーハンドラ - * @param {Error} error - エラーオブジェクト - */ -function handleTransportError(error) { - Logger.error('Transport error', error); - transport = null; -} - -/** - * MCP SSE サーバーを起動 - * @param {number} port - ポート番号 - * @returns {Object} HTTPサーバーオブジェクト - * @throws {Error} サーバー起動に失敗した場合 - */ -export function startMcpServer(port = 3000) { - try { - // ポート番号の検証 - if (!Number.isInteger(port) || port < 1 || port > 65535) { - throw new Error(`Invalid port number: ${port}`); - } - - app.use(express.json()); - - /** - * SSE 接続エンドポイント - * @route GET /sse - */ - app.get(CONFIG.mcp.ssePath, async (req, res) => { - try { - Logger.info('SSE connection established'); - transport = new SSEServerTransport(CONFIG.mcp.messagesPath, res); - transport.on('error', handleTransportError); - await server.connect(transport); - } catch (error) { - Logger.error('Failed to establish SSE connection', error); - res.status(500).send('Failed to establish SSE connection'); - } - }); - - /** - * メッセージハンドラエンドポイント - * @route POST /messages - */ - app.post(CONFIG.mcp.messagesPath, async (req, res) => { - try { - if (!transport) { - Logger.warn('No active transport connection'); - return res.status(400).send('No transport connection'); - } - await transport.handlePostMessage(req, res); - } catch (error) { - Logger.error('Failed to handle post message', error); - res.status(500).send('Failed to handle message'); - } - }); - - /** - * ヘルスチェックエンドポイント - * @route GET /health - */ - app.get('/health', (req, res) => { - res.status(200).json({ - status: 'ok', - timestamp: new Date().toISOString(), - transport: transport ? 'connected' : 'disconnected', - }); - }); - - httpServer = app.listen(port, () => { - Logger.info(`MCP SSE Server listening on http://localhost:${port}${CONFIG.mcp.ssePath}`); - }); - - // エラーハンドリング - httpServer.on('error', (error) => { - Logger.error('HTTP server error', error); - }); - - return httpServer; - } catch (error) { - Logger.error('Failed to start MCP server', error); - throw error; - } -} diff --git a/src/mcp-tools.js b/src/mcp-tools.js deleted file mode 100644 index 64486bd..0000000 --- a/src/mcp-tools.js +++ /dev/null @@ -1,144 +0,0 @@ -import { CONFIG } from "./config.js"; -import { Logger } from "./logger.js"; - -/** - * MCP ツール定義を取得 - * @returns {Array} ツール定義の配列 - * @description - * - add_item_text: テキストから埋め込みを自動生成して保存 - * - add_item: テキストと埋め込みベクトルを直接保存 - * - search_text: テキストから埋め込みを自動生成して検索 - * - search_vector: ベクトルで直接検索 - * - llm_generate: llama.cpp でテキスト生成を実行 - */ -export function getToolDefinitions() { - const embeddingDim = CONFIG.database.embeddingDim; - - return [ - { - name: "add_item_text", - description: "テキストから埋め込みを生成して保存します", - inputSchema: { - type: "object", - properties: { - content: { - type: "string", - description: "保存するテキスト内容", - }, - path: { - type: "string", - description: "アイテムのメタデータパス(オプション)", - }, - }, - required: ["content"], - }, - }, - { - name: "add_item", - description: "テキストと埋め込みベクトルを直接保存します", - inputSchema: { - type: "object", - properties: { - content: { - type: "string", - description: "保存するテキスト内容", - }, - path: { - type: "string", - description: "アイテムのメタデータパス(オプション)", - }, - vector: { - type: "array", - items: { type: "number" }, - minItems: embeddingDim, - maxItems: embeddingDim, - description: `埋め込みベクトル (${embeddingDim}次元)`, - }, - }, - required: ["content", "vector"], - }, - }, - { - name: "search_text", - description: "テキストから埋め込みを生成して類似アイテムを検索します", - inputSchema: { - type: "object", - properties: { - content: { - type: "string", - description: "検索キーワード", - }, - limit: { - type: "number", - description: "返す結果の最大数(デフォルト: 10)", - minimum: 1, - maximum: 100, - }, - }, - required: ["content"], - }, - }, - { - name: "search_vector", - description: "ベクトルで類似度検索を実行します", - inputSchema: { - type: "object", - properties: { - vector: { - type: "array", - items: { type: "number" }, - minItems: embeddingDim, - maxItems: embeddingDim, - description: `検索するベクトル (${embeddingDim}次元)`, - }, - limit: { - type: "number", - description: "返す結果の最大数(デフォルト: 10)", - minimum: 1, - maximum: 100, - }, - }, - required: ["vector"], - }, - }, - { - name: "llm_generate", - description: "llama.cpp でテキスト生成を実行します", - inputSchema: { - type: "object", - properties: { - prompt: { - type: "string", - description: "生成するテキストのプロンプト", - }, - n_predict: { - type: "number", - description: "生成するトークン数(デフォルト: 128)", - minimum: 1, - maximum: 2048, - }, - temperature: { - type: "number", - description: "温度パラメータ(0.0-2.0、デフォルト: 0.7)", - minimum: 0, - maximum: 2, - }, - }, - required: ["prompt"], - }, - }, - ]; -} - -/** - * ツール名の定数 - */ -export const TOOL_NAMES = { - ADD_ITEM_TEXT: "add_item_text", - ADD_ITEM: "add_item", - SEARCH_TEXT: "search_text", - SEARCH_VECTOR: "search_vector", - LLM_GENERATE: "llm_generate", -}; - -Logger.debug("MCP tools module loaded");